summaryrefslogtreecommitdiff
path: root/software/demo_gpio/demo_gpio.c
blob: 3b0b6f5ebb5347ca2501df312cab3c4f6c94516b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
// See LICENSE for license details.

#include <stdio.h>

#include "shared.h"
#include "plic.h"
#include <string.h>

void reset_demo (void);

// Structures for registering different interrupt handlers
// for different parts of the application.
typedef void (*function_ptr_t) (void);

void no_interrupt_handler (void) {};

function_ptr_t g_ext_interrupt_handlers[9];

function_ptr_t g_m_timer_interrupt_handler = no_interrupt_handler;

// Instance data for the PLIC.

plic_instance_t g_plic;

// Simple variables for LEDs, buttons, etc.
volatile unsigned int* g_outputs  = (unsigned int *) (GPIO_BASE_ADDR + GPIO_OUT_OFFSET);
volatile unsigned int* g_inputs   = (unsigned int *) (GPIO_BASE_ADDR + GPIO_IN_OFFSET);
volatile unsigned int* g_tristates = (unsigned int *) (GPIO_BASE_ADDR + GPIO_TRI_OFFSET);


/*Entry Point for PLIC Interrupt Handler*/
void handle_m_ext_interrupt(){
  plic_source int_num  = PLIC_claim_interrupt(&g_plic);
  if ((int_num >=1 ) && (int_num <= 8)) {
    g_ext_interrupt_handlers[int_num]();
  }
  else {
    _exit(1 + (uintptr_t) int_num);
  }
  PLIC_complete_interrupt(&g_plic, int_num);
}


/*Entry Point for Machine Timer Interrupt Handler*/
void handle_m_time_interrupt(){

  clear_csr(mie, MIP_MTIP);
  
  // Reset the timer for 3s in the future.
  // This also clears the existing timer interrupt.
  
  volatile uint64_t * mtime       = (uint64_t*) MTIME_ADDR;
  volatile uint64_t * mtimecmp    = (uint64_t*) MTIMECMP_BASE_ADDR;
  uint64_t now = *mtime;
  uint64_t then = now + 3 * CLOCK_FREQUENCY / RTC_PRESCALER;
  *mtimecmp = then;
 
  // read the current value of the LEDS and invert them.
  uint32_t leds = *g_outputs;
		   
  *g_outputs = (~leds) & ((0xF << RED_LEDS_OFFSET)   |
			  (0xF << GREEN_LEDS_OFFSET) |
			  (0xF << BLUE_LEDS_OFFSET));
  

  // Re-enable the timer interrupt.
  set_csr(mie, MIP_MTIP);
 
}


/*Entry Point for Machine Timer Interrupt  Handler*/
void handle_m_timer_interrupt(){
  g_m_timer_interrupt_handler();
}

const char * instructions_msg = " \
\n\
                SIFIVE, INC.\n\
\n\
         5555555555555555555555555\n\
        5555                   5555\n\
       5555                     5555\n\
      5555                       5555\n\
     5555       5555555555555555555555\n\
    5555       555555555555555555555555\n\
   5555                             5555\n\
  5555                               5555\n\
 5555                                 5555\n\
5555555555555555555555555555          55555\n\
 55555           555555555           55555\n\
   55555           55555           55555\n\
     55555           5           55555\n\
       55555                   55555\n\
         55555               55555\n\
           55555           55555\n\
             55555       55555\n\
               55555   55555\n\
                 555555555\n\
                   55555\n\
                     5\n\
\n\
 SiFive E-Series Coreplex Demo on Arty Board\n \
 \n\
 ";

void print_instructions() {
  
  write (STDOUT_FILENO, instructions_msg, strlen(instructions_msg));
  
}

void button_0_handler(void) {

  // Rainbow LEDs!
  * g_outputs = (0x9 << RED_LEDS_OFFSET) | (0xA << GREEN_LEDS_OFFSET) | (0xC << BLUE_LEDS_OFFSET);

};


void reset_demo (){

    // Disable the machine & timer interrupts until setup is done.
  
    clear_csr(mie, MIP_MEIP);
    clear_csr(mie, MIP_MTIP);
  
    g_ext_interrupt_handlers[INT_DEVICE_BUTTON_0] = button_0_handler;
    g_ext_interrupt_handlers[INT_DEVICE_BUTTON_1] = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_BUTTON_2] = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_BUTTON_3] = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_JA_7]     = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_JA_8]     = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_JA_9]     = no_interrupt_handler;
    g_ext_interrupt_handlers[INT_DEVICE_JA_10]    = no_interrupt_handler;
    
    print_instructions();

    PLIC_enable_interrupt (&g_plic, INT_DEVICE_BUTTON_0);
       
    // Set the machine timer to go off in 3 seconds.
    // The
    volatile uint64_t * mtime       = (uint64_t*) MTIME_ADDR;
    volatile uint64_t * mtimecmp    = (uint64_t*) MTIMECMP_BASE_ADDR;
    uint64_t now = *mtime;
    uint64_t then = now + 3*CLOCK_FREQUENCY / RTC_PRESCALER;
    *mtimecmp = then;

    // Enable the Machine-External bit in MIE
    set_csr(mie, MIP_MEIP);

    // Enable the Machine-Timer bit in MIE
    set_csr(mie, MIP_MTIP);
    
    // Enable interrupts in general.
    set_csr(mstatus, MSTATUS_MIE);

}

int main(int argc, char **argv)
{
  // Set up the GPIOs such that the inputs' tristate are set.
  // The boot ROM actually does this, but still good.
  
  * g_tristates = (0xF << BUTTONS_OFFSET) | (0xF << SWITCHES_OFFSET) | (0xF << JA_IN_OFFSET);
  
  * g_outputs = (0xF << RED_LEDS_OFFSET);


  /**************************************************************************
   * Set up the PLIC
   * 
   *************************************************************************/
  PLIC_init(&g_plic, PLIC_BASE_ADDR, PLIC_NUM_SOURCES, PLIC_NUM_PRIORITIES);

  reset_demo();

  while (1);

  return 0;

}