Squash commits for public release
This commit is contained in:
226
boot/arm64/prekernel/drivers/fb.c
Normal file
226
boot/arm64/prekernel/drivers/fb.c
Normal file
@@ -0,0 +1,226 @@
|
||||
#include "fb.h"
|
||||
#include <libboot/devtree/devtree.h>
|
||||
#include <libboot/types.h>
|
||||
|
||||
static char font8x8_basic[128][8];
|
||||
|
||||
static int _cursor_x = 0;
|
||||
static int _cursor_y = 0;
|
||||
static int _fbwidth = 0;
|
||||
static int _fbheight = 0;
|
||||
static int _fbpixels_per_row = 0;
|
||||
static int _fbscale = 1;
|
||||
static uint32_t _basecolor = 0x0;
|
||||
volatile uint32_t* _fb = NULL;
|
||||
|
||||
static void system_cache_invalidate(void* addr, size_t size)
|
||||
{
|
||||
const size_t cache_line_size = 64;
|
||||
size_t start = ROUND_FLOOR((size_t)addr, cache_line_size);
|
||||
size_t end = ROUND_CEIL((size_t)addr + size, cache_line_size);
|
||||
|
||||
asm volatile("isb");
|
||||
asm volatile("dsb sy");
|
||||
for (size_t curaddr = start; curaddr < end; curaddr += cache_line_size) {
|
||||
asm volatile("dc ivac, %0"
|
||||
:
|
||||
: "r"(curaddr));
|
||||
}
|
||||
asm volatile("dsb sy");
|
||||
asm volatile("isb");
|
||||
}
|
||||
|
||||
static void update_cursor_position(char c)
|
||||
{
|
||||
_cursor_x += 8 * _fbscale;
|
||||
|
||||
if (c == '\n' || (_cursor_x + 8 * _fbscale) > _fbwidth) {
|
||||
_cursor_y += 8 * _fbscale;
|
||||
_cursor_x = 0;
|
||||
|
||||
if ((_cursor_y + 8 * _fbscale + 2) > _fbheight) {
|
||||
_cursor_y = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void draw_char_on_screen(char c)
|
||||
{
|
||||
for (int x = 0; x < (8 * _fbscale); x++) {
|
||||
for (int y = 0; y < (8 * _fbscale); y++) {
|
||||
uint32_t ind = (_cursor_x + x) + ((_cursor_y + y) * _fbpixels_per_row);
|
||||
uint32_t clr = _basecolor;
|
||||
if (font8x8_basic[c & 0x7f][y / _fbscale] & (1 << (x / _fbscale))) {
|
||||
clr ^= 0xffffffff;
|
||||
}
|
||||
_fb[ind] = clr;
|
||||
}
|
||||
}
|
||||
system_cache_invalidate((void*)&_fb[_cursor_y * _fbpixels_per_row], (1 + 8 * _fbscale) * _fbpixels_per_row * 4);
|
||||
update_cursor_position(c);
|
||||
}
|
||||
|
||||
int fb_init()
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("simplefb");
|
||||
if (!dev) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
_cursor_x = 0;
|
||||
_cursor_y = 0;
|
||||
_fb = (uint32_t*)dev->aux4;
|
||||
_fbwidth = dev->aux1;
|
||||
_fbheight = dev->aux2;
|
||||
_fbpixels_per_row = dev->aux3;
|
||||
_fbscale = 2;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int fb_reinit_after_map(uintptr_t vaddr)
|
||||
{
|
||||
_fb = (uint32_t*)vaddr;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int fb_put_char(uint8_t c)
|
||||
{
|
||||
if (!_fb) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
draw_char_on_screen((char)c);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static char font8x8_basic[128][8] = {
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0000 (nul)
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0001
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0002
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0003
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0004
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0005
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0006
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0007
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0008
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0009
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000A
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000B
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000C
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000D
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000E
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000F
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0010
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0011
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0012
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0013
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0014
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0015
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0016
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0017
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0018
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0019
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001A
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001B
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001C
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001D
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001E
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001F
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0020 (space)
|
||||
{ 0x18, 0x3C, 0x3C, 0x18, 0x18, 0x00, 0x18, 0x00 }, // U+0021 (!)
|
||||
{ 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0022 (")
|
||||
{ 0x36, 0x36, 0x7F, 0x36, 0x7F, 0x36, 0x36, 0x00 }, // U+0023 (#)
|
||||
{ 0x0C, 0x3E, 0x03, 0x1E, 0x30, 0x1F, 0x0C, 0x00 }, // U+0024 ($)
|
||||
{ 0x00, 0x63, 0x33, 0x18, 0x0C, 0x66, 0x63, 0x00 }, // U+0025 (%)
|
||||
{ 0x1C, 0x36, 0x1C, 0x6E, 0x3B, 0x33, 0x6E, 0x00 }, // U+0026 (&)
|
||||
{ 0x06, 0x06, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0027 (')
|
||||
{ 0x18, 0x0C, 0x06, 0x06, 0x06, 0x0C, 0x18, 0x00 }, // U+0028 (()
|
||||
{ 0x06, 0x0C, 0x18, 0x18, 0x18, 0x0C, 0x06, 0x00 }, // U+0029 ())
|
||||
{ 0x00, 0x66, 0x3C, 0xFF, 0x3C, 0x66, 0x00, 0x00 }, // U+002A (*)
|
||||
{ 0x00, 0x0C, 0x0C, 0x3F, 0x0C, 0x0C, 0x00, 0x00 }, // U+002B (+)
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x0C, 0x06 }, // U+002C (,)
|
||||
{ 0x00, 0x00, 0x00, 0x3F, 0x00, 0x00, 0x00, 0x00 }, // U+002D (-)
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x0C, 0x00 }, // U+002E (.)
|
||||
{ 0x60, 0x30, 0x18, 0x0C, 0x06, 0x03, 0x01, 0x00 }, // U+002F (/)
|
||||
{ 0x3E, 0x63, 0x73, 0x7B, 0x6F, 0x67, 0x3E, 0x00 }, // U+0030 (0)
|
||||
{ 0x0C, 0x0E, 0x0C, 0x0C, 0x0C, 0x0C, 0x3F, 0x00 }, // U+0031 (1)
|
||||
{ 0x1E, 0x33, 0x30, 0x1C, 0x06, 0x33, 0x3F, 0x00 }, // U+0032 (2)
|
||||
{ 0x1E, 0x33, 0x30, 0x1C, 0x30, 0x33, 0x1E, 0x00 }, // U+0033 (3)
|
||||
{ 0x38, 0x3C, 0x36, 0x33, 0x7F, 0x30, 0x78, 0x00 }, // U+0034 (4)
|
||||
{ 0x3F, 0x03, 0x1F, 0x30, 0x30, 0x33, 0x1E, 0x00 }, // U+0035 (5)
|
||||
{ 0x1C, 0x06, 0x03, 0x1F, 0x33, 0x33, 0x1E, 0x00 }, // U+0036 (6)
|
||||
{ 0x3F, 0x33, 0x30, 0x18, 0x0C, 0x0C, 0x0C, 0x00 }, // U+0037 (7)
|
||||
{ 0x1E, 0x33, 0x33, 0x1E, 0x33, 0x33, 0x1E, 0x00 }, // U+0038 (8)
|
||||
{ 0x1E, 0x33, 0x33, 0x3E, 0x30, 0x18, 0x0E, 0x00 }, // U+0039 (9)
|
||||
{ 0x00, 0x0C, 0x0C, 0x00, 0x00, 0x0C, 0x0C, 0x00 }, // U+003A (:)
|
||||
{ 0x00, 0x0C, 0x0C, 0x00, 0x00, 0x0C, 0x0C, 0x06 }, // U+003B (;)
|
||||
{ 0x18, 0x0C, 0x06, 0x03, 0x06, 0x0C, 0x18, 0x00 }, // U+003C (<)
|
||||
{ 0x00, 0x00, 0x3F, 0x00, 0x00, 0x3F, 0x00, 0x00 }, // U+003D (=)
|
||||
{ 0x06, 0x0C, 0x18, 0x30, 0x18, 0x0C, 0x06, 0x00 }, // U+003E (>)
|
||||
{ 0x1E, 0x33, 0x30, 0x18, 0x0C, 0x00, 0x0C, 0x00 }, // U+003F (?)
|
||||
{ 0x3E, 0x63, 0x7B, 0x7B, 0x7B, 0x03, 0x1E, 0x00 }, // U+0040 (@)
|
||||
{ 0x0C, 0x1E, 0x33, 0x33, 0x3F, 0x33, 0x33, 0x00 }, // U+0041 (A)
|
||||
{ 0x3F, 0x66, 0x66, 0x3E, 0x66, 0x66, 0x3F, 0x00 }, // U+0042 (B)
|
||||
{ 0x3C, 0x66, 0x03, 0x03, 0x03, 0x66, 0x3C, 0x00 }, // U+0043 (C)
|
||||
{ 0x1F, 0x36, 0x66, 0x66, 0x66, 0x36, 0x1F, 0x00 }, // U+0044 (D)
|
||||
{ 0x7F, 0x46, 0x16, 0x1E, 0x16, 0x46, 0x7F, 0x00 }, // U+0045 (E)
|
||||
{ 0x7F, 0x46, 0x16, 0x1E, 0x16, 0x06, 0x0F, 0x00 }, // U+0046 (F)
|
||||
{ 0x3C, 0x66, 0x03, 0x03, 0x73, 0x66, 0x7C, 0x00 }, // U+0047 (G)
|
||||
{ 0x33, 0x33, 0x33, 0x3F, 0x33, 0x33, 0x33, 0x00 }, // U+0048 (H)
|
||||
{ 0x1E, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0049 (I)
|
||||
{ 0x78, 0x30, 0x30, 0x30, 0x33, 0x33, 0x1E, 0x00 }, // U+004A (J)
|
||||
{ 0x67, 0x66, 0x36, 0x1E, 0x36, 0x66, 0x67, 0x00 }, // U+004B (K)
|
||||
{ 0x0F, 0x06, 0x06, 0x06, 0x46, 0x66, 0x7F, 0x00 }, // U+004C (L)
|
||||
{ 0x63, 0x77, 0x7F, 0x7F, 0x6B, 0x63, 0x63, 0x00 }, // U+004D (M)
|
||||
{ 0x63, 0x67, 0x6F, 0x7B, 0x73, 0x63, 0x63, 0x00 }, // U+004E (N)
|
||||
{ 0x1C, 0x36, 0x63, 0x63, 0x63, 0x36, 0x1C, 0x00 }, // U+004F (O)
|
||||
{ 0x3F, 0x66, 0x66, 0x3E, 0x06, 0x06, 0x0F, 0x00 }, // U+0050 (P)
|
||||
{ 0x1E, 0x33, 0x33, 0x33, 0x3B, 0x1E, 0x38, 0x00 }, // U+0051 (Q)
|
||||
{ 0x3F, 0x66, 0x66, 0x3E, 0x36, 0x66, 0x67, 0x00 }, // U+0052 (R)
|
||||
{ 0x1E, 0x33, 0x07, 0x0E, 0x38, 0x33, 0x1E, 0x00 }, // U+0053 (S)
|
||||
{ 0x3F, 0x2D, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0054 (T)
|
||||
{ 0x33, 0x33, 0x33, 0x33, 0x33, 0x33, 0x3F, 0x00 }, // U+0055 (U)
|
||||
{ 0x33, 0x33, 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x00 }, // U+0056 (V)
|
||||
{ 0x63, 0x63, 0x63, 0x6B, 0x7F, 0x77, 0x63, 0x00 }, // U+0057 (W)
|
||||
{ 0x63, 0x63, 0x36, 0x1C, 0x1C, 0x36, 0x63, 0x00 }, // U+0058 (X)
|
||||
{ 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0059 (Y)
|
||||
{ 0x7F, 0x63, 0x31, 0x18, 0x4C, 0x66, 0x7F, 0x00 }, // U+005A (Z)
|
||||
{ 0x1E, 0x06, 0x06, 0x06, 0x06, 0x06, 0x1E, 0x00 }, // U+005B ([)
|
||||
{ 0x03, 0x06, 0x0C, 0x18, 0x30, 0x60, 0x40, 0x00 }, // U+005C (\)
|
||||
{ 0x1E, 0x18, 0x18, 0x18, 0x18, 0x18, 0x1E, 0x00 }, // U+005D (])
|
||||
{ 0x08, 0x1C, 0x36, 0x63, 0x00, 0x00, 0x00, 0x00 }, // U+005E (^)
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF }, // U+005F (_)
|
||||
{ 0x0C, 0x0C, 0x18, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0060 (`)
|
||||
{ 0x00, 0x00, 0x1E, 0x30, 0x3E, 0x33, 0x6E, 0x00 }, // U+0061 (a)
|
||||
{ 0x07, 0x06, 0x06, 0x3E, 0x66, 0x66, 0x3B, 0x00 }, // U+0062 (b)
|
||||
{ 0x00, 0x00, 0x1E, 0x33, 0x03, 0x33, 0x1E, 0x00 }, // U+0063 (c)
|
||||
{ 0x38, 0x30, 0x30, 0x3e, 0x33, 0x33, 0x6E, 0x00 }, // U+0064 (d)
|
||||
{ 0x00, 0x00, 0x1E, 0x33, 0x3f, 0x03, 0x1E, 0x00 }, // U+0065 (e)
|
||||
{ 0x1C, 0x36, 0x06, 0x0f, 0x06, 0x06, 0x0F, 0x00 }, // U+0066 (f)
|
||||
{ 0x00, 0x00, 0x6E, 0x33, 0x33, 0x3E, 0x30, 0x1F }, // U+0067 (g)
|
||||
{ 0x07, 0x06, 0x36, 0x6E, 0x66, 0x66, 0x67, 0x00 }, // U+0068 (h)
|
||||
{ 0x0C, 0x00, 0x0E, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0069 (i)
|
||||
{ 0x30, 0x00, 0x30, 0x30, 0x30, 0x33, 0x33, 0x1E }, // U+006A (j)
|
||||
{ 0x07, 0x06, 0x66, 0x36, 0x1E, 0x36, 0x67, 0x00 }, // U+006B (k)
|
||||
{ 0x0E, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+006C (l)
|
||||
{ 0x00, 0x00, 0x33, 0x7F, 0x7F, 0x6B, 0x63, 0x00 }, // U+006D (m)
|
||||
{ 0x00, 0x00, 0x1F, 0x33, 0x33, 0x33, 0x33, 0x00 }, // U+006E (n)
|
||||
{ 0x00, 0x00, 0x1E, 0x33, 0x33, 0x33, 0x1E, 0x00 }, // U+006F (o)
|
||||
{ 0x00, 0x00, 0x3B, 0x66, 0x66, 0x3E, 0x06, 0x0F }, // U+0070 (p)
|
||||
{ 0x00, 0x00, 0x6E, 0x33, 0x33, 0x3E, 0x30, 0x78 }, // U+0071 (q)
|
||||
{ 0x00, 0x00, 0x3B, 0x6E, 0x66, 0x06, 0x0F, 0x00 }, // U+0072 (r)
|
||||
{ 0x00, 0x00, 0x3E, 0x03, 0x1E, 0x30, 0x1F, 0x00 }, // U+0073 (s)
|
||||
{ 0x08, 0x0C, 0x3E, 0x0C, 0x0C, 0x2C, 0x18, 0x00 }, // U+0074 (t)
|
||||
{ 0x00, 0x00, 0x33, 0x33, 0x33, 0x33, 0x6E, 0x00 }, // U+0075 (u)
|
||||
{ 0x00, 0x00, 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x00 }, // U+0076 (v)
|
||||
{ 0x00, 0x00, 0x63, 0x6B, 0x7F, 0x7F, 0x36, 0x00 }, // U+0077 (w)
|
||||
{ 0x00, 0x00, 0x63, 0x36, 0x1C, 0x36, 0x63, 0x00 }, // U+0078 (x)
|
||||
{ 0x00, 0x00, 0x33, 0x33, 0x33, 0x3E, 0x30, 0x1F }, // U+0079 (y)
|
||||
{ 0x00, 0x00, 0x3F, 0x19, 0x0C, 0x26, 0x3F, 0x00 }, // U+007A (z)
|
||||
{ 0x38, 0x0C, 0x0C, 0x07, 0x0C, 0x0C, 0x38, 0x00 }, // U+007B ({)
|
||||
{ 0x18, 0x18, 0x18, 0x00, 0x18, 0x18, 0x18, 0x00 }, // U+007C (|)
|
||||
{ 0x07, 0x0C, 0x0C, 0x38, 0x0C, 0x0C, 0x07, 0x00 }, // U+007D (})
|
||||
{ 0x6E, 0x3B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+007E (~)
|
||||
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } // U+007F
|
||||
};
|
||||
11
boot/arm64/prekernel/drivers/fb.h
Normal file
11
boot/arm64/prekernel/drivers/fb.h
Normal file
@@ -0,0 +1,11 @@
|
||||
#ifndef _BOOT_DRIVERS_SCREEN_H
|
||||
#define _BOOT_DRIVERS_SCREEN_H
|
||||
|
||||
#include <libboot/abi/memory.h>
|
||||
#include <libboot/types.h>
|
||||
|
||||
int fb_init();
|
||||
int fb_reinit_after_map(uintptr_t vaddr);
|
||||
int fb_put_char(uint8_t c);
|
||||
|
||||
#endif // _BOOT_DRIVERS_SCREEN_H
|
||||
23
boot/arm64/prekernel/drivers/uart.c
Normal file
23
boot/arm64/prekernel/drivers/uart.c
Normal file
@@ -0,0 +1,23 @@
|
||||
#include "uart.h"
|
||||
#include <libboot/devtree/devtree.h>
|
||||
|
||||
volatile uint32_t* output = NULL;
|
||||
|
||||
void uart_init()
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("uart");
|
||||
if (!dev) {
|
||||
return;
|
||||
}
|
||||
output = (uint32_t*)dev->region_base;
|
||||
}
|
||||
|
||||
int uart_write(uint8_t data)
|
||||
{
|
||||
if (!output) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
*output = data;
|
||||
return 0;
|
||||
}
|
||||
9
boot/arm64/prekernel/drivers/uart.h
Normal file
9
boot/arm64/prekernel/drivers/uart.h
Normal file
@@ -0,0 +1,9 @@
|
||||
#ifndef _BOOT_DRIVERS_UART_H
|
||||
#define _BOOT_DRIVERS_UART_H
|
||||
|
||||
#include <libboot/types.h>
|
||||
|
||||
void uart_init();
|
||||
int uart_write(uint8_t data);
|
||||
|
||||
#endif // _BOOT_DRIVERS_UART_H
|
||||
66
boot/arm64/prekernel/entry.S
Normal file
66
boot/arm64/prekernel/entry.S
Normal file
@@ -0,0 +1,66 @@
|
||||
.extern main
|
||||
|
||||
.section ".xos_boot_text"
|
||||
prekernel_header:
|
||||
nop // TODO(arm64): Add UEFI support
|
||||
b prekernel_entry
|
||||
|
||||
prekernel_entry:
|
||||
// Calculating location where of start address.
|
||||
adr x20, .
|
||||
adr x22, prekernel_header
|
||||
adr x23, prekernel_entry
|
||||
sub x21, x23, x22
|
||||
sub x20, x20, x21
|
||||
|
||||
// Saving devtree
|
||||
mov x21, x0
|
||||
|
||||
// Setting up prekernel stack.
|
||||
ldr x30, =PREKERNEL_STACK_TOP_OFFSET
|
||||
add x30, x20, x30
|
||||
mov sp, x30
|
||||
|
||||
// Base of kernel
|
||||
mov x0, x20
|
||||
// Header location
|
||||
ldr x1, =PREKERNEL_END_OFFSET
|
||||
add x1, x20, x1
|
||||
// Devtree location
|
||||
mov x2, x21
|
||||
ldr x3, =main
|
||||
add x3, x3, x20
|
||||
blr x3
|
||||
entry_loop:
|
||||
b entry_loop
|
||||
|
||||
.global jump_to_kernel
|
||||
jump_to_kernel:
|
||||
ic iallu
|
||||
dsb sy
|
||||
isb sy
|
||||
blr x1
|
||||
jump_loop:
|
||||
b jump_loop
|
||||
|
||||
.global enable_mmu_el1
|
||||
enable_mmu_el1:
|
||||
dsb sy
|
||||
msr mair_el1, x2
|
||||
msr tcr_el1, x1
|
||||
msr ttbr0_el1, x0
|
||||
msr ttbr1_el1, x3
|
||||
isb sy
|
||||
tlbi vmalle1
|
||||
isb sy
|
||||
ic iallu
|
||||
isb sy
|
||||
mrs x3, sctlr_el1
|
||||
orr x3, x3, #1
|
||||
orr x3, x3, #4
|
||||
and x3, x3, #(~2)
|
||||
msr sctlr_el1, x3
|
||||
ic iallu
|
||||
dsb sy
|
||||
isb sy
|
||||
ret
|
||||
159
boot/arm64/prekernel/main.c
Normal file
159
boot/arm64/prekernel/main.c
Normal file
@@ -0,0 +1,159 @@
|
||||
#include "drivers/fb.h"
|
||||
#include "drivers/uart.h"
|
||||
#include "vm.h"
|
||||
#include <libboot/abi/memory.h>
|
||||
#include <libboot/abi/rawimage.h>
|
||||
#include <libboot/devtree/devtree.h>
|
||||
#include <libboot/elf/elf_lite.h>
|
||||
#include <libboot/log/log.h>
|
||||
#include <libboot/mem/alloc.h>
|
||||
#include <libboot/mem/mem.h>
|
||||
|
||||
#define DEBUG_BOOT
|
||||
#define EARLY_FB
|
||||
|
||||
extern void jump_to_kernel(void*, uintptr_t);
|
||||
static void* bootdesc_paddr;
|
||||
static void* bootdesc_vaddr;
|
||||
static size_t kernel_vaddr = 0;
|
||||
static size_t kernel_paddr = 0;
|
||||
static size_t kernel_size = 0;
|
||||
|
||||
#define LAUNCH_SERVER_PATH "/System/launch_server"
|
||||
|
||||
static int alloc_init(uintptr_t base, rawimage_header_t* riheader)
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("ram");
|
||||
if (!dev) {
|
||||
#ifdef EARLY_FB
|
||||
log("Can't find RAM in devtree");
|
||||
#endif
|
||||
while (1) { };
|
||||
}
|
||||
|
||||
// Currenlty we setting the allocation after the raw image.
|
||||
// Allocator is used to alloc paddrs for kernel, vm and other.
|
||||
uintptr_t start_addr = ROUND_CEIL(base + riheader->rawimage_size, page_size());
|
||||
size_t free_space = dev->region_size - (start_addr - dev->region_base);
|
||||
malloc_init((void*)start_addr, free_space);
|
||||
|
||||
#if defined(DEBUG_BOOT) && defined(EARLY_FB)
|
||||
log("malloc inited %llx %llx", start_addr, free_space);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
static size_t memory_layout_size()
|
||||
{
|
||||
return 0 + 1; // Including the trailing element.
|
||||
}
|
||||
|
||||
static int preserve_alloc_init(size_t kernsize)
|
||||
{
|
||||
const size_t devtreesize = ROUND_CEIL(devtree_size(), page_size());
|
||||
const size_t memmap = ROUND_CEIL(memory_layout_size() * sizeof(memory_layout_t), page_size());
|
||||
const size_t bootargsstruct = ROUND_CEIL(sizeof(boot_args_t), page_size());
|
||||
const size_t bootargssize = devtreesize + memmap + bootargsstruct;
|
||||
|
||||
// 32 tables should be enough for initial mappings.
|
||||
const size_t prekernelvmsize = 32 * page_size();
|
||||
const size_t total = ROUND_CEIL(kernsize + bootargssize + prekernelvmsize, page_size());
|
||||
|
||||
return palloc_init(total, 2 << 20);
|
||||
}
|
||||
|
||||
static memory_boot_desc_t memory_boot_desc_init()
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("ram");
|
||||
if (!dev) {
|
||||
#ifdef EARLY_FB
|
||||
log("Can't find RAM in devtree");
|
||||
#endif
|
||||
while (1) { };
|
||||
}
|
||||
|
||||
memory_boot_desc_t res;
|
||||
const size_t memlayout_size = memory_layout_size();
|
||||
memory_layout_t* mem_layout_paddr = palloc_aligned(memlayout_size * sizeof(memory_layout_t), page_size());
|
||||
|
||||
// Init of trailing element.
|
||||
mem_layout_paddr[memlayout_size - 1].flags = MEMORY_LAYOUT_FLAG_TERMINATE;
|
||||
memory_layout_t* mem_layout_vaddr = paddr_to_vaddr(mem_layout_paddr, kernel_paddr, kernel_vaddr);
|
||||
#if defined(DEBUG_BOOT) && defined(EARLY_FB)
|
||||
log("initing MEMLAYOUT %lx of %d elems", mem_layout_vaddr, memlayout_size);
|
||||
#endif
|
||||
|
||||
res.ram_base = dev->region_base;
|
||||
res.ram_size = dev->region_size;
|
||||
res.reserved_areas = mem_layout_vaddr;
|
||||
return res;
|
||||
}
|
||||
|
||||
static void load_kernel(void* kenrelstart)
|
||||
{
|
||||
kernel_size = elf_get_kernel_size(kenrelstart);
|
||||
kernel_size = ROUND_CEIL(kernel_size, page_size());
|
||||
|
||||
int err = preserve_alloc_init(kernel_size);
|
||||
if (err) {
|
||||
#ifdef EARLY_FB
|
||||
log("add assert");
|
||||
#endif
|
||||
while (1) { }
|
||||
}
|
||||
|
||||
int res = elf_load_kernel(kenrelstart, kernel_size, &kernel_vaddr, &kernel_paddr);
|
||||
#if defined(DEBUG_BOOT) && defined(EARLY_FB)
|
||||
log("kernel %lx %lx %lx", kernel_vaddr, kernel_paddr, kernel_size);
|
||||
#endif
|
||||
|
||||
void* odt_paddr = palloc_aligned(devtree_size(), page_size());
|
||||
memcpy(odt_paddr, devtree_start(), devtree_size());
|
||||
void* odt_vaddr = paddr_to_vaddr(odt_paddr, kernel_paddr, kernel_vaddr);
|
||||
#if defined(DEBUG_BOOT) && defined(EARLY_FB)
|
||||
log("copying ODT %lx -> %lx of %d", devtree_start(), odt_vaddr, devtree_size());
|
||||
#endif
|
||||
|
||||
boot_args_t boot_args;
|
||||
boot_args.vaddr = kernel_vaddr;
|
||||
boot_args.paddr = kernel_paddr;
|
||||
boot_args.kernel_data_size = 0x0; // Sets up later
|
||||
boot_args.mem_boot_desc = memory_boot_desc_init();
|
||||
boot_args.devtree = odt_vaddr;
|
||||
boot_args.fb_boot_desc.vaddr = 0; // Marking fb as invalid.
|
||||
memcpy(boot_args.init_process, LAUNCH_SERVER_PATH, sizeof(LAUNCH_SERVER_PATH));
|
||||
|
||||
bootdesc_paddr = palloc_aligned(sizeof(boot_args), page_size());
|
||||
memcpy(bootdesc_paddr, &boot_args, sizeof(boot_args));
|
||||
bootdesc_vaddr = paddr_to_vaddr(bootdesc_paddr, kernel_paddr, kernel_vaddr);
|
||||
#if defined(DEBUG_BOOT) && defined(EARLY_FB)
|
||||
log("copying BOOTDESC %lx -> %lx of %d", &boot_args, bootdesc_vaddr, sizeof(boot_args));
|
||||
#endif
|
||||
}
|
||||
|
||||
static int log_merged_output(uint8_t ch)
|
||||
{
|
||||
uart_write(ch);
|
||||
fb_put_char(ch);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int main(uintptr_t base, rawimage_header_t* riheader, void* devtree)
|
||||
{
|
||||
devtree_init((void*)(base + riheader->devtree_off), riheader->devtree_size);
|
||||
uart_init();
|
||||
fb_init();
|
||||
log_init(log_merged_output);
|
||||
alloc_init(base, riheader);
|
||||
|
||||
load_kernel((void*)(base + riheader->kern_off));
|
||||
vm_setup(base, bootdesc_paddr, riheader);
|
||||
|
||||
#ifdef DEBUG_BOOT
|
||||
log("Preboot done: booting to OS@%llx", ((boot_args_t*)bootdesc_vaddr)->vaddr);
|
||||
#endif
|
||||
|
||||
((boot_args_t*)bootdesc_vaddr)->kernel_data_size = ROUND_CEIL(palloc_used_size(), page_size());
|
||||
jump_to_kernel((void*)bootdesc_vaddr, ((boot_args_t*)bootdesc_vaddr)->vaddr);
|
||||
return 0;
|
||||
}
|
||||
168
boot/arm64/prekernel/vm.c
Normal file
168
boot/arm64/prekernel/vm.c
Normal file
@@ -0,0 +1,168 @@
|
||||
#include "vm.h"
|
||||
#include "drivers/fb.h"
|
||||
#include <libboot/abi/kernel.h>
|
||||
#include <libboot/abi/rawimage.h>
|
||||
#include <libboot/devtree/devtree.h>
|
||||
#include <libboot/log/log.h>
|
||||
#include <libboot/mem/alloc.h>
|
||||
#include <libboot/mem/mem.h>
|
||||
|
||||
// #define DEBUG_VM
|
||||
|
||||
static uint64_t* global_page_table_0;
|
||||
static uint64_t* global_page_table_1;
|
||||
|
||||
// This is setup for 4KB pages
|
||||
static const int tg0 = 0b00;
|
||||
static const int tg1 = 0b10;
|
||||
static const int t0sz = 25;
|
||||
static const int t1sz = 25;
|
||||
static const uint64_t kernel_base = 0xffffffffffffffff - ((1ull << (64 - t1sz)) - 1);
|
||||
|
||||
static uint64_t* new_ptable(boot_args_t* args)
|
||||
{
|
||||
uint64_t* res = (uint64_t*)palloc_aligned(page_size(), page_size());
|
||||
memset(res, 0, page_size());
|
||||
#ifdef DEBUG_VM
|
||||
log(" alloc ptable %llx %llx", (uint64_t)res, page_size());
|
||||
#endif
|
||||
return res;
|
||||
}
|
||||
|
||||
// Huge page impl is not suitable for Apl, where we need percise mappings.
|
||||
static void map4kb_1gb(boot_args_t* args, size_t phyz, size_t virt)
|
||||
{
|
||||
const size_t page_covers = (1ull << PTABLE_LV2_VADDR_OFFSET);
|
||||
const size_t page_mask = page_covers - 1;
|
||||
|
||||
if ((phyz & page_mask) != 0 || (virt & page_mask) != 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Mapping from level2, as needed.
|
||||
uint64_t* page_table = global_page_table_0;
|
||||
if (virt >= kernel_base) {
|
||||
page_table = global_page_table_1;
|
||||
virt -= kernel_base;
|
||||
}
|
||||
|
||||
uint64_t pdesc = 0x00000000000701;
|
||||
pdesc |= (uintptr_t)phyz;
|
||||
page_table[VM_VADDR_OFFSET_AT_LEVEL(virt, PTABLE_LV2_VADDR_OFFSET, VMM_LV2_ENTITY_COUNT)] = pdesc;
|
||||
}
|
||||
|
||||
static void map4kb_2mb(boot_args_t* args, size_t phyz, size_t virt)
|
||||
{
|
||||
const size_t page_covers = (1ull << PTABLE_LV1_VADDR_OFFSET);
|
||||
const size_t page_mask = page_covers - 1;
|
||||
|
||||
if ((phyz & page_mask) != 0 || (virt & page_mask) != 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Mapping from level2, as needed.
|
||||
uint64_t* page_table = global_page_table_0;
|
||||
if (virt >= kernel_base) {
|
||||
page_table = global_page_table_1;
|
||||
virt -= kernel_base;
|
||||
}
|
||||
|
||||
uint64_t ptable_desc = page_table[VM_VADDR_OFFSET_AT_LEVEL(virt, PTABLE_LV2_VADDR_OFFSET, VMM_LV2_ENTITY_COUNT)];
|
||||
if (ptable_desc == 0) {
|
||||
uint64_t* nptbl = new_ptable(args);
|
||||
uint64_t pdesc = 0x00000000000003;
|
||||
pdesc |= (uintptr_t)nptbl;
|
||||
page_table[VM_VADDR_OFFSET_AT_LEVEL(virt, PTABLE_LV2_VADDR_OFFSET, VMM_LV2_ENTITY_COUNT)] = pdesc;
|
||||
ptable_desc = pdesc;
|
||||
}
|
||||
|
||||
page_table = (uint64_t*)(((ptable_desc >> 12) << 12) & 0xffffffffffff);
|
||||
uint64_t pdesc = 0x00000000000701;
|
||||
pdesc |= (uintptr_t)phyz;
|
||||
page_table[VM_VADDR_OFFSET_AT_LEVEL(virt, PTABLE_LV1_VADDR_OFFSET, VMM_LV1_ENTITY_COUNT)] = pdesc;
|
||||
}
|
||||
|
||||
static void map_uart(boot_args_t* args)
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("uart");
|
||||
if (!dev) {
|
||||
return;
|
||||
}
|
||||
|
||||
uint64_t paddr = dev->region_base;
|
||||
const size_t page_covers = (1ull << PTABLE_LV2_VADDR_OFFSET);
|
||||
paddr &= ~(page_covers - 1);
|
||||
|
||||
#ifdef DEBUG_VM
|
||||
log("mapping %lx %lx", paddr, paddr);
|
||||
#endif
|
||||
map4kb_1gb(args, paddr, paddr);
|
||||
}
|
||||
|
||||
static void map_fb(boot_args_t* args)
|
||||
{
|
||||
devtree_entry_t* dev = devtree_find_device("simplefb");
|
||||
if (!dev) {
|
||||
return;
|
||||
}
|
||||
|
||||
uint64_t paddr = dev->region_base;
|
||||
const size_t page_covers = (1ull << PTABLE_LV2_VADDR_OFFSET);
|
||||
paddr &= ~(page_covers - 1);
|
||||
|
||||
#ifdef DEBUG_VM
|
||||
log("mapping %lx %lx", paddr, 0xfc0000000ULL);
|
||||
#endif
|
||||
map4kb_1gb(args, paddr, 0xfc0000000ULL);
|
||||
|
||||
args->fb_boot_desc.paddr = dev->region_base;
|
||||
args->fb_boot_desc.vaddr = 0xfc0000000ULL + (args->fb_boot_desc.paddr - paddr);
|
||||
args->fb_boot_desc.width = dev->aux1;
|
||||
args->fb_boot_desc.height = dev->aux2;
|
||||
args->fb_boot_desc.pixels_per_row = dev->aux3;
|
||||
}
|
||||
|
||||
void vm_setup(uintptr_t base, boot_args_t* args, rawimage_header_t* riheader)
|
||||
{
|
||||
// This implementation is a stub, supporting only 4kb pages for now.
|
||||
// We should support 16kb pages for sure on modern Apls.
|
||||
global_page_table_0 = (uint64_t*)palloc_aligned(page_size(), page_size());
|
||||
memset(global_page_table_0, 0, page_size());
|
||||
|
||||
global_page_table_1 = (uint64_t*)palloc_aligned(page_size(), page_size());
|
||||
memset(global_page_table_1, 0, page_size());
|
||||
|
||||
const size_t map_range_2mb = (2 << 20);
|
||||
const size_t map_range_1gb = (1 << 30);
|
||||
|
||||
// Mapping kernel vaddr to paddr
|
||||
size_t kernel_size_to_map = palloc_total_size() + shadow_area_size();
|
||||
size_t kernel_range_count_to_map = (kernel_size_to_map + (map_range_2mb - 1)) / map_range_2mb;
|
||||
for (size_t i = 0; i < kernel_range_count_to_map; i++) {
|
||||
#ifdef DEBUG_VM
|
||||
log("mapping %lx %lx", args->paddr + i * map_range_2mb, args->vaddr + i * map_range_2mb);
|
||||
#endif
|
||||
map4kb_2mb(args, args->paddr + i * map_range_2mb, args->vaddr + i * map_range_2mb);
|
||||
}
|
||||
|
||||
// Mapping RAM
|
||||
size_t ram_base = ROUND_FLOOR(args->mem_boot_desc.ram_base, map_range_1gb);
|
||||
size_t ram_size = args->mem_boot_desc.ram_size;
|
||||
size_t ram_range_count_to_map = (ram_size + (map_range_1gb - 1)) / map_range_1gb;
|
||||
for (size_t i = 0; i < ram_range_count_to_map; i++) {
|
||||
#ifdef DEBUG_VM
|
||||
log("mapping %lx %lx", ram_base + i * map_range_1gb, ram_base + i * map_range_1gb);
|
||||
#endif
|
||||
map4kb_1gb(args, ram_base + i * map_range_1gb, ram_base + i * map_range_1gb);
|
||||
}
|
||||
|
||||
// The initial boot requires framebuffer and uart to be mapped.
|
||||
// Checking this and mapping devices.
|
||||
map_uart(args);
|
||||
map_fb(args);
|
||||
|
||||
extern void enable_mmu_el1(uint64_t ttbr0, uint64_t tcr, uint64_t mair, uint64_t ttbr1);
|
||||
enable_mmu_el1((uint64_t)global_page_table_0, 0x135003500 | (tg0 << 14) | (tg1 << 30) | (t1sz << 16) | t0sz, 0x04ff, (uint64_t)global_page_table_1);
|
||||
|
||||
fb_reinit_after_map(args->fb_boot_desc.vaddr);
|
||||
}
|
||||
25
boot/arm64/prekernel/vm.h
Normal file
25
boot/arm64/prekernel/vm.h
Normal file
@@ -0,0 +1,25 @@
|
||||
#ifndef _BOOT_VM_H
|
||||
#define _BOOT_VM_H
|
||||
|
||||
#include <libboot/abi/memory.h>
|
||||
#include <libboot/abi/rawimage.h>
|
||||
#include <libboot/types.h>
|
||||
|
||||
#define VMM_LV0_ENTITY_COUNT (512)
|
||||
#define VMM_LV1_ENTITY_COUNT (512)
|
||||
#define VMM_LV2_ENTITY_COUNT (512)
|
||||
#define VMM_LV3_ENTITY_COUNT (512)
|
||||
|
||||
#define PTABLE_LV_TOP (2)
|
||||
#define PTABLE_LV0_VADDR_OFFSET (12)
|
||||
#define PTABLE_LV1_VADDR_OFFSET (21)
|
||||
#define PTABLE_LV2_VADDR_OFFSET (30)
|
||||
#define PTABLE_LV3_VADDR_OFFSET (39)
|
||||
|
||||
#define VM_VADDR_OFFSET_AT_LEVEL(vaddr, off, ent) ((vaddr >> off) % ent)
|
||||
|
||||
static size_t page_size() { return 0x1000; }
|
||||
|
||||
void vm_setup(uintptr_t base, boot_args_t* args, rawimage_header_t* riheader);
|
||||
|
||||
#endif // _BOOT_VM_H
|
||||
Reference in New Issue
Block a user