| 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
 | /*
References:
  https://github-wiki-see.page/m/raspberrypi/firmware/wiki/Mailbox-property-interface
  https://jsandler18.github.io/extra/prop-channel.html
  https://jsandler18.github.io/extra/mailbox.html
*/
#pragma once
#include <stdint.h>
// The channel takes the lower 4 bits; the data pointer the upper 28. So a mail
// must be aligned to a 16-byte boundary.
#define MAIL_ALIGN 16
// Type prefix for data arrays used as mail.
#define MAIL_T volatile __attribute__((aligned(MAIL_ALIGN)))
enum
{
  PROPERTY_CHANNEL = 8,
};
enum
{
  MAILBOX_REQUEST  = 0,
  MAILBOX_RESPONSE = 1,
};
enum
{
  MAILBOX_DELIVERY_ERROR = 0,
  MAILBOX_ERROR          = 0x80000001,
  MAILBOX_SUCCESS        = 0x80000000,
};
enum
{
  TAG_END                                  = 0,
  TAG_FRAMEBUFFER_ALLOCATE                 = 0x00040001,
  TAG_FRAMEBUFFER_RELEASE                  = 0x00048001,
  TAG_FRAMEBUFFER_SET_PHYSICAL_SCREEN_SIZE = 0x00048003,
  TAG_FRAMEBUFFER_SET_VIRTUAL_SCREEN_SIZE  = 0x00048004,
  TAG_FRAMEBUFFER_SET_DEPTH                = 0x00048005,
};
typedef struct Tag {
  union {
    uint32_t all;
    struct {
      uint16_t command : 12; // Command.
      uint8_t  type    :  4; // Command type.
      uint8_t  device  :  4; // Hardware device.
      uint16_t zeroes  : 12; // Reserved.
    } id;
  };
  uint32_t size;    // Buffer size.
  uint32_t code;    // Request/response code.
  uint32_t data[];  // Buffer data.
} Tag;
typedef struct __attribute__((aligned(MAIL_ALIGN))) Mail {
  uint32_t size;    // Buffer size.
  uint32_t code;    // Request/response code.
  Tag      tags[];  // Variable quantity.
} Mail;
void        mbox_init();
const Mail* mbox_read(uint8_t channel);
uint32_t    mbox_write(uint8_t channel, volatile const void* mail);
 |