summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/kernel.c5
-rw-r--r--src/mailbox.c73
-rw-r--r--src/mailbox.h38
-rw-r--r--src/mmio.c4
-rw-r--r--src/mmio.h5
-rw-r--r--src/uart.c19
6 files changed, 127 insertions, 17 deletions
diff --git a/src/kernel.c b/src/kernel.c
index a9a2d11..4296aa7 100644
--- a/src/kernel.c
+++ b/src/kernel.c
@@ -1,3 +1,4 @@
1#include <mailbox.h>
1#include <mmio.h> 2#include <mmio.h>
2#include <raspi.h> 3#include <raspi.h>
3#include <uart.h> 4#include <uart.h>
@@ -10,7 +11,9 @@ static void halt() {
10 11
11void main() { 12void main() {
12 const int raspi = raspi_init(); 13 const int raspi = raspi_init();
13 mmio_init(raspi); 14 mmio_init(raspi); // Must be initialized before other peripherals.
15
16 mbox_init();
14 uart_init(raspi); 17 uart_init(raspi);
15 18
16 uart_print("Hello world!\n"); 19 uart_print("Hello world!\n");
diff --git a/src/mailbox.c b/src/mailbox.c
new file mode 100644
index 0000000..eaf0955
--- /dev/null
+++ b/src/mailbox.c
@@ -0,0 +1,73 @@
1#include <mailbox.h>
2
3#include <mmio.h>
4
5#include <stdbool.h>
6#include <stdint.h>
7
8enum
9{
10 MAILBOX = 0xB880 // Mailbox address relative to MMIO base address.
11};
12
13typedef uint32_t Message;
14
15static inline uint8_t msg_channel(Message msg) {
16 return msg & 0xf;
17}
18
19static inline const void* msg_data(Message msg) {
20 // The data field is actually a pointer to the data.
21 return (const void*)((uintptr_t)(msg >> 4));
22}
23
24static inline Message msg_make(uint8_t channel, const void* data) {
25 return ((uintptr_t)(data) << 4) | (channel & 0xf);
26}
27
28typedef struct Mailbox {
29 Message inbox; // 0x00
30 uint32_t unused_1; // 0x04
31 uint32_t unused_2; // 0x08
32 uint32_t unused_3; // 0x0C
33 uint32_t unused_4; // 0x10
34 uint32_t unused_5; // 0x14
35 uint32_t status; // 0x18
36 uint32_t unused_6; // 0x1C
37 Message outbox; // 0x20
38} Mailbox;
39
40static inline bool inbox_empty(const volatile Mailbox* pMailbox) {
41 return (pMailbox->status & 0x40000000) != 0;
42}
43
44static inline bool outbox_full(const volatile Mailbox* pMailbox) {
45 return (pMailbox->status & 0x80000000) != 0;
46}
47
48static volatile Mailbox* pMailbox;
49
50void mbox_init() {
51 pMailbox = mmio_get_peripheral(MAILBOX);
52}
53
54const Mail* mbox_read(uint8_t channel) {
55 Message msg;
56 // Read until we get a message from the desired channel.
57 // Discard any other messages.
58 do {
59 // Wait until there is mail to receive.
60 while (inbox_empty(pMailbox));
61 // Read the mail.
62 msg = pMailbox->inbox;
63 } while (msg_channel(msg) != channel);
64 return (const Mail*)(msg_data(msg));
65}
66
67void mbox_write(uint8_t channel, const void* mail) {
68 // Wait until the outbox is clear.
69 while (outbox_full(pMailbox));
70 // Send the mail.
71 pMailbox->outbox = msg_make(channel, mail);
72}
73
diff --git a/src/mailbox.h b/src/mailbox.h
new file mode 100644
index 0000000..e96a4ed
--- /dev/null
+++ b/src/mailbox.h
@@ -0,0 +1,38 @@
1#pragma once
2
3#include <stdint.h>
4
5// The channel takes the lower 4 bits; the data pointer the upper 28. So a mail
6// must be aligned to a 16-byte boundary.
7#define MAIL_ALIGN 16
8
9enum
10{
11 PROPERTY_CHANNEL = 8,
12};
13
14typedef struct Tag {
15 union {
16 uint32_t all;
17 struct {
18 uint16_t command : 12; // Command.
19 uint8_t type : 4; // Command type.
20 uint8_t device : 4; // Hardware device.
21 uint16_t zeroes : 12; // Reserved.
22 } id;
23 };
24 uint32_t size; // Buffer size.
25 uint32_t code; // Request/response code.
26 uint32_t data[1]; // Buffer data.
27} Tag;
28
29typedef struct __attribute__((aligned(MAIL_ALIGN))) Mail {
30 uint32_t size; // Buffer size.
31 uint32_t code; // Request/response code.
32 Tag tags[1]; // Variable quantity.
33} Mail;
34
35void mbox_init();
36const Mail* mbox_read(uint8_t channel);
37void mbox_write(uint8_t channel, const void* mail);
38
diff --git a/src/mmio.c b/src/mmio.c
index e082d66..b2abe98 100644
--- a/src/mmio.c
+++ b/src/mmio.c
@@ -30,3 +30,7 @@ void mmio_write(uint32_t reg, uint32_t val) {
30 *REG_ADDR(reg) = val; 30 *REG_ADDR(reg) = val;
31} 31}
32 32
33void* mmio_get_peripheral(uint32_t peripheral_base) {
34 return MMIO_BASE + peripheral_base;
35}
36
diff --git a/src/mmio.h b/src/mmio.h
index b566bbe..4d111d6 100644
--- a/src/mmio.h
+++ b/src/mmio.h
@@ -2,7 +2,8 @@
2 2
3#include <stdint.h> 3#include <stdint.h>
4 4
5void mmio_init(int raspi); 5void mmio_init(int raspi);
6uint32_t mmio_read(uint32_t reg); 6uint32_t mmio_read(uint32_t reg);
7void mmio_write(uint32_t reg, uint32_t val); 7void mmio_write(uint32_t reg, uint32_t val);
8void* mmio_get_peripheral(uint32_t peripheral_base);
8 9
diff --git a/src/uart.c b/src/uart.c
index bb2a0ea..3b107f4 100644
--- a/src/uart.c
+++ b/src/uart.c
@@ -5,6 +5,7 @@ References:
5#include <uart.h> 5#include <uart.h>
6 6
7#include <gpio.h> 7#include <gpio.h>
8#include <mailbox.h>
8#include <mmio.h> 9#include <mmio.h>
9 10
10enum 11enum
@@ -31,15 +32,10 @@ enum
31 UART0_ITIP = (UART0_BASE + 0x84), 32 UART0_ITIP = (UART0_BASE + 0x84),
32 UART0_ITOP = (UART0_BASE + 0x88), 33 UART0_ITOP = (UART0_BASE + 0x88),
33 UART0_TDR = (UART0_BASE + 0x8C), 34 UART0_TDR = (UART0_BASE + 0x8C),
34 // The offsets for Mailbox registers.
35 MBOX_BASE = 0xB880,
36 MBOX_READ = (MBOX_BASE + 0x00),
37 MBOX_STATUS = (MBOX_BASE + 0x18),
38 MBOX_WRITE = (MBOX_BASE + 0x20)
39}; 35};
40 36
41// A mailbox message with set clock rate of PL011 to 3MHz tag. 37// A mailbox message with set clock rate of PL011 to 3MHz tag.
42static volatile unsigned int __attribute__((aligned(16))) mbox[9] = { 38static const uint32_t __attribute__((aligned(MAIL_ALIGN))) UART_SET_CLK[9] = {
43 9*4, 0, 0x38002, 12, 8, 2, 3000000, 0, 0 39 9*4, 0, 0x38002, 12, 8, 2, 3000000, 0, 0
44}; 40};
45 41
@@ -77,14 +73,9 @@ void uart_init(int raspi) {
77 // For Raspi3 and 4 the UART_CLOCK is system-clock dependent by default. 73 // For Raspi3 and 4 the UART_CLOCK is system-clock dependent by default.
78 // Set it to 3Mhz so that we can consistently set the baud rate 74 // Set it to 3Mhz so that we can consistently set the baud rate
79 if (raspi >= 3) { 75 if (raspi >= 3) {
80 // UART_CLOCK = 30000000; 76 // Send message over property channel to configure UART clock.
81 unsigned int r = (unsigned int) (((uint64_t)(&mbox) & ~0xF) | 8); 77 mbox_write(PROPERTY_CHANNEL, UART_SET_CLK);
82 // Wait until we can talk to the VC. 78 mbox_read(PROPERTY_CHANNEL);
83 while (mmio_read(MBOX_STATUS) & 0x80000000);
84 // Send our message to property channel and wait for the response.
85 mmio_write(MBOX_WRITE, r);
86 while ((mmio_read(MBOX_STATUS) & 0x40000000) ||
87 (mmio_read(MBOX_READ) != r));
88 } 79 }
89 80
90 // Divider = 3000000 / (16 * 115200) = 1.627 = ~1. 81 // Divider = 3000000 / (16 * 115200) = 1.627 = ~1.