summaryrefslogtreecommitdiff
path: root/src/mainboard/emulation/spike-riscv/spike_util.c
blob: b34ff4a0d023c0d8250e08dbc635d7d574a92006 (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
#include <spike_util.h>

uintptr_t htif_interrupt(uintptr_t mcause, uintptr_t* regs) {
        uintptr_t fromhost = swap_csr(mfromhost, 0);
        if (!fromhost)
        return 0;

        uintptr_t dev = FROMHOST_DEV(fromhost);
        uintptr_t cmd = FROMHOST_CMD(fromhost);
        uintptr_t data = FROMHOST_DATA(fromhost);

        sbi_device_message* m = HLS()->device_request_queue_head;
        sbi_device_message* prev = 0x0;
        unsigned long i, n;
        for (i = 0, n = HLS()->device_request_queue_size; i < n; i++) {
                /*
                if (!supervisor_paddr_valid(m, sizeof(*m))
                && EXTRACT_FIELD(read_csr(mstatus), MSTATUS_PRV1) != PRV_M)
                panic("htif: page fault");
                */

                sbi_device_message* next = (void*)m->sbi_private_data;
                if (m->dev == dev && m->cmd == cmd) {
                        m->data = data;

                        // dequeue from request queue
                        if (prev)
                        prev->sbi_private_data = (uintptr_t)next;
                        else
                        HLS()->device_request_queue_head = next;
                        HLS()->device_request_queue_size = n-1;
                        m->sbi_private_data = 0;

                        // enqueue to response queue
                        if (HLS()->device_response_queue_tail)
                        {
                                HLS()->device_response_queue_tail->sbi_private_data = (uintptr_t)m;
                        }
                        else
                        {
                                HLS()->device_response_queue_head = m;
                        }
                        HLS()->device_response_queue_tail = m;

                        // signal software interrupt
                        set_csr(mip, MIP_SSIP);
                        return 0;
                }

                prev = m;
                m = (void*)atomic_read(&m->sbi_private_data);
        }
        //HLT();
        return 0;
        //panic("htif: no record");
}

uintptr_t mcall_console_putchar(uint8_t ch)
{
        while (swap_csr(mtohost, TOHOST_CMD(1, 1, ch)) != 0);
        while (1) {
                uintptr_t fromhost = read_csr(mfromhost);
                if (FROMHOST_DEV(fromhost) != 1 || FROMHOST_CMD(fromhost) != 1) {
                if (fromhost)
                htif_interrupt(0, 0);
                continue;
        }
        write_csr(mfromhost, 0);
        break;
        }
        return 0;
}

void testPrint(void) {
        /* Print a test command to check Spike console output */
        mcall_console_putchar('h');
        mcall_console_putchar('e');
        mcall_console_putchar('l');
        mcall_console_putchar('l');
        mcall_console_putchar('o');
        mcall_console_putchar('\n');
}