initial commit

This commit is contained in:
2025-03-11 05:17:34 +08:00
commit e906741ee9
27 changed files with 924 additions and 0 deletions

72
Makefile Normal file
View File

@@ -0,0 +1,72 @@
MAKE := make
ARCH := aarch64
GNU := $(ARCH)-linux-gnu
QEMU := qemu-system-$(ARCH)
QEMUFLAGS += -M raspi3b -display none -serial null -serial pty
CC := $(GNU)-gcc
CCFLAGS += -Wall -Wextra -O0 \
-nostdinc -nostdlib -nostartfiles -nodefaultlibs \
-Wno-unused-parameter -Wno-main -g
LD := $(GNU)-ld
LDFLAGS += -g -nostdlib
OBJCOPY := $(GNU)-objcopy
#TARGET := kernel8
#TARGET_DIR := ./kernel
ELF := $(TARGET).elf
IMG := $(TARGET).img
LD_SCRIPT := $(TARGET_DIR)/linker.ld
INCLUDE += -Iinclude
LIB_DIR := ./lib
MISC_DIR := ./misc
#QEMUFLAGS += -dtb $(MISC_DIR)/bcm2710-rpi-3-b-plus.dtb
BOOTLOADER_DIR := ./bootloader
ROOTFS_DIR := ./rootfs
CPIO := initramfs.cpio
QEMUFLAGS += -initrd $(CPIO)
SRCS := $(shell find $(TARGET_DIR) -name '*.[cS]') \
$(shell find $(LIB_DIR) -name '*.c')
OBJS := $(SRCS:%=%.o)
.PHONY: all build clean clean_target run
all:
$(MAKE) build TARGET_DIR=./kernel TARGET=kernel8
$(MAKE) build TARGET_DIR=./bootloader TARGET=bootloader
build: $(IMG)
$(CPIO): $(shell find $(ROOTFS_DIR))
cd $(ROOTFS_DIR) && find . | cpio -o -H newc > ../$@
$(IMG): $(ELF)
$(OBJCOPY) -O binary $< $@
$(ELF): $(LD_SCRIPT) $(OBJS)
$(LD) -o $@ -T $^ $(LDFLAGS)
%.S.o: %.S
mkdir -p $(dir $@)
$(CC) -c $< -o $@ $(INCLUDE) $(CCFLAGS)
%.c.o: %.c
mkdir -p $(dir $@)
$(CC) -c $< -o $@ $(INCLUDE) $(CCFLAGS)
clean:
$(MAKE) clean_target TARGET_DIR=./kernel TARGET=kernel8
$(MAKE) clean_target TARGET_DIR=./bootloader TARGET=bootloader
clean_target:
-rm $(OBJS) $(ELF) $(IMG) $(CPIO)
run: all $(CPIO)
$(QEMU) -kernel bootloader.img $(QEMUFLAGS)
#$(QEMU) -kernel kernel8.img $(QEMUFLAGS)

56
bootloader/Start.S Normal file
View File

@@ -0,0 +1,56 @@
.section ".text.boot"
.global _start
_start:
mrs x1, mpidr_el1
and x1, x1, #3
cbz x1, run
wait:
wfe
b wait
run:
mov x20, x0
adr x0, _start
mov sp, x0
adr x0, __text_start
ldr x1, =__new_text_start
cmp x0, x1
b.gt relocate
adr x0, __bss_start
adr x1, __bss_end
bl memzero
mov x0, x20
bl main
b wait
relocate:
// move text section
ldr x0, =__new_text_start
adr x1, __text_start
adr x2, __text_end
sub x2, x2, x1
bl memcpy
mov x19, x0
// move rodata section
ldr x0, =__new_ro_start
adr x1, __rodata_start
adr x2, __rodata_end
sub x2, x2, x1
bl memcpy
// move data section
ldr x0, =__new_data_start
adr x1, __data_start
adr x2, __data_end
sub x2, x2, x1
bl memcpy
br x19
b wait

50
bootloader/linker.ld Normal file
View File

@@ -0,0 +1,50 @@
ENTRY(_start)
MEMORY
{
NEWTEXT (rx) : ORIGIN = 0x10000, LENGTH = 128K
NEWRO (r) : ORIGIN = 0x30000, LENGTH = 128K
NEWDATA (rw) : ORIGIN = 0x50000, LENGTH = 64K
NEWBSS (rw) : ORIGIN = 0x60000, LENGTH = 64K
TEXT (rx) : ORIGIN = 0x80000, LENGTH = 128K
RO (r) : ORIGIN = 0xa0000, LENGTH = 128K
DATA (rw) : ORIGIN = 0xc0000, LENGTH = 64K
BSS (rw) : ORIGIN = 0xd0000, LENGTH = 64K
RAM (rw) : ORIGIN = 0xf0000, LENGTH = 8M
}
SECTIONS
{
.text : {
__text_start = .;
KEEP(*(.text.boot))
*(.text)
__text_end = .;
} >TEXT
.rodata : {
__rodata_start = .;
*(.rodata)
__rodata_end = .;
} >RO
.data : {
__data_start = .;
*(.data)
__data_end = .;
} >DATA
.bss : {
__bss_start = .;
*(.bss)
__bss_end = .;
} >BSS
__stack_end = ORIGIN(RAM) + LENGTH(RAM);
__new_text_start = ORIGIN(NEWTEXT);
__new_ro_start = ORIGIN(NEWRO);
__new_data_start = ORIGIN(NEWDATA);
__new_bss_start = ORIGIN(NEWBSS);
}
__kernel = 0x80000;
__heap_start = ORIGIN(RAM);
__heap_end = ORIGIN(RAM) + LENGTH(RAM) - 2M;
__bss_size = (__bss_end - __bss_start)>>3;

21
bootloader/main.c Normal file
View File

@@ -0,0 +1,21 @@
#include <stddef.h>
#include <shell.h>
#include <uart.h>
extern byte_t __kernel[];
byte_t *kernel = __kernel;
void main()
{
uart_init();
uart_getc();
uart_puts("loaded addr: ");
uart_hex((unsigned int)(unsigned long)main);
uart_puts(ENDL);
int shell_cont = 1;
while (shell_cont) {
shell_cont = shell();
}
}

46
include/gpio.h Normal file
View File

@@ -0,0 +1,46 @@
#pragma once
/*
* Copyright (C) 2018 bzt (bztsrc@github)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
*/
#define MMIO_BASE 0x3F000000
#define GPFSEL0 ((volatile unsigned int*)(MMIO_BASE+0x00200000))
#define GPFSEL1 ((volatile unsigned int*)(MMIO_BASE+0x00200004))
#define GPFSEL2 ((volatile unsigned int*)(MMIO_BASE+0x00200008))
#define GPFSEL3 ((volatile unsigned int*)(MMIO_BASE+0x0020000C))
#define GPFSEL4 ((volatile unsigned int*)(MMIO_BASE+0x00200010))
#define GPFSEL5 ((volatile unsigned int*)(MMIO_BASE+0x00200014))
#define GPSET0 ((volatile unsigned int*)(MMIO_BASE+0x0020001C))
#define GPSET1 ((volatile unsigned int*)(MMIO_BASE+0x00200020))
#define GPCLR0 ((volatile unsigned int*)(MMIO_BASE+0x00200028))
#define GPLEV0 ((volatile unsigned int*)(MMIO_BASE+0x00200034))
#define GPLEV1 ((volatile unsigned int*)(MMIO_BASE+0x00200038))
#define GPEDS0 ((volatile unsigned int*)(MMIO_BASE+0x00200040))
#define GPEDS1 ((volatile unsigned int*)(MMIO_BASE+0x00200044))
#define GPHEN0 ((volatile unsigned int*)(MMIO_BASE+0x00200064))
#define GPHEN1 ((volatile unsigned int*)(MMIO_BASE+0x00200068))
#define GPPUD ((volatile unsigned int*)(MMIO_BASE+0x00200094))
#define GPPUDCLK0 ((volatile unsigned int*)(MMIO_BASE+0x00200098))
#define GPPUDCLK1 ((volatile unsigned int*)(MMIO_BASE+0x0020009C))

46
include/initrd.h Normal file
View File

@@ -0,0 +1,46 @@
#pragma once
#include <stddef.h>
typedef struct {
char c_magic[6];
char c_ino[8];
char c_mode[8];
char c_uid[8];
char c_gid[8];
char c_nlink[8];
char c_mtime[8];
char c_filesize[8];
char c_devmajor[8];
char c_devminor[8];
char c_rdevmajor[8];
char c_rdevminor[8];
char c_namesize[8];
char c_check[8];
}__attribute__((packed)) cpio_newc_header_t;
typedef struct file_node {
struct file_node *l, *r;
int rand;
int node_size;
int ino;
int mode;
int uid;
int gid;
int nlink;
int mtime;
int filesize;
int devmajor;
int devminor;
int rdevmajor;
int rdevminor;
int namesize;
char *filename;
byte_t *filecontent;
} file_node_t;
file_node_t *initrd_init(void);
int initrd_ls(void);
file_node_t *initrd_get(file_node_t *root, const char *filename);

7
include/kmalloc.h Normal file
View File

@@ -0,0 +1,7 @@
#pragma once
#include <stddef.h>
void *simple_alloc(size_t size);
void *kmalloc(size_t size);
void kfree(void *ptr);

48
include/mbox.h Normal file
View File

@@ -0,0 +1,48 @@
/*
* Copyright (C) 2018 bzt (bztsrc@github)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
*/
/* a properly aligned buffer */
extern volatile unsigned int mbox[36];
#define MBOX_REQUEST 0
/* channels */
#define MBOX_CH_POWER 0
#define MBOX_CH_FB 1
#define MBOX_CH_VUART 2
#define MBOX_CH_VCHIQ 3
#define MBOX_CH_LEDS 4
#define MBOX_CH_BTNS 5
#define MBOX_CH_TOUCH 6
#define MBOX_CH_COUNT 7
#define MBOX_CH_PROP 8
/* tags */
#define MBOX_TAG_GETSERIAL 0x10004
#define MBOX_TAG_LAST 0
#define MBOX_TAG_BOARD_REVISION 0x00010002
int mbox_call(unsigned char ch);
unsigned int get_board_revision(void);

8
include/random.h Normal file
View File

@@ -0,0 +1,8 @@
#pragma once
extern const int _random_a;
extern const int _random_c;
extern const int _random_m;
extern int seed;
int random(void);

9
include/shell.h Normal file
View File

@@ -0,0 +1,9 @@
#pragma once
void help(void);
void hello(void);
void hwinfo(void);
void reboot(void);
int // is continue
shell(void);

9
include/stddef.h Normal file
View File

@@ -0,0 +1,9 @@
#pragma once
#define true 1
#define false 0
#define ENDL "\r\n"
typedef long unsigned int size_t;
typedef unsigned char byte_t;
typedef unsigned long long int uint64_t;

7
include/string.h Normal file
View File

@@ -0,0 +1,7 @@
#pragma once
#include <stddef.h>
int strcmp(const char *lhs, const char *rhs);
void *memcpy(void *dest, const void *src, size_t count);
void *memzero(void *start, void *end);

10
include/uart.h Normal file
View File

@@ -0,0 +1,10 @@
#pragma once
#include <stddef.h>
void uart_init();
void uart_send(unsigned int c);
byte_t uart_getb();
char uart_getc();
void uart_puts(char *s);
void uart_hex(unsigned int d);

32
kernel/Start.S Normal file
View File

@@ -0,0 +1,32 @@
.section ".text.boot"
.global _start
_start:
// read cpu id, stop slave cores
mrs x1, mpidr_el1
and x1, x1, #3
cbz x1, 2f
// cpu id > 0, stop
1:
wfe
b 1b
2:// cpu id == 0
// set top of stack just before our code (stack grows to a lower address per AAPCS64)
ldr x1, =_start
mov sp, x1
// clear bss
ldr x1, =__bss_start
ldr w2, =__bss_size
3:
cbz w2, 4f
str xzr, [x1], #8
sub w2, w2, #1
cbnz w2, 3b
4:
// jump to C code, should not return
bl main
// for failsafe, halt this core too
b 1b

33
kernel/linker.ld Normal file
View File

@@ -0,0 +1,33 @@
ENTRY(_start)
MEMORY
{
TEXT (rx) : ORIGIN = 0x80000, LENGTH = 128K
RO (r) : ORIGIN = 0xa0000, LENGTH = 128K
DATA (rw) : ORIGIN = 0x100000, LENGTH = 512K
RAM (rw) : ORIGIN = 0x180000, LENGTH = 8M
}
SECTIONS
{
.text : {
KEEP(*(.text.boot))
*(.text)
} >TEXT
.rodata : {
*(.rodata)
} >RO
.data : {
*(.data)
} >DATA
.bss : {
__bss_start = .;
*(.bss)
__bss_end = .;
} >DATA
__stack_end = ORIGIN(RAM) + LENGTH(RAM);
}
__kernel = 0x80000;
__heap_start = ORIGIN(RAM);
__heap_end = ORIGIN(RAM) + LENGTH(RAM) - 2M;
__bss_size = (__bss_end - __bss_start)>>3;

12
kernel/main.c Normal file
View File

@@ -0,0 +1,12 @@
#include <uart.h>
#include <shell.h>
void main()
{
uart_init();
int shell_cont = 1;
while (shell_cont) {
shell_cont = shell();
}
}

88
lib/initrd.c Normal file
View File

@@ -0,0 +1,88 @@
#include <stddef.h>
#include <initrd.h>
#include <kmalloc.h>
#include <random.h>
#include <string.h>
#define nullnode ((file_node_t *)0)
void _init_node(file_node_t *node) {
node->l = nullnode;
node->r = nullnode;
node->rand = random();
node->node_size = 1;
}
void _pull_from(file_node_t *to, file_node_t *from)
{
if (!from)
return;
to->node_size += from->node_size;
}
void _pull(file_node_t *node)
{
node->node_size = 1;
_pull_from(node, node->l);
_pull_from(node, node->r);
}
file_node_t *_merge(file_node_t *a, file_node_t *b)
{
if (!a || !b)
return a ?: b;
if (a->rand < b->rand) {
a->r = _merge(a->r, b);
_pull(a);
return a;
}
b->l = _merge(a, b->l);
_pull(b);
return b;
}
void _split(file_node_t *rt, const char *s, file_node_t **a, file_node_t **b)
{
if (!rt) {
*a = *b = nullnode;
return;
}
if (strcmp(s, rt->filename) < 0) {
*a = rt;
_split((*a)->r, s, &(*a)->r, b);
_pull(*a);
} else {
*b = rt;
_split((*b)->l, s, a, &(*b)->l);
_pull(*b);
}
}
file_node_t *initrd_init(void)
{
// TODO
return nullnode;
}
file_node_t *_node_bs(file_node_t *cur, const char *s)
{
if (!cur)
return nullnode;
int cmp = strcmp(s, cur->filename);
if (cmp < 0)
return _node_bs(cur->r, s);
if (cmp > 0)
return _node_bs(cur->l, s);
return cur; // cmp == 0
}
file_node_t *initrd_get(file_node_t *root, const char *filename)
{
return _node_bs(root, filename);
}

37
lib/kmalloc.c Normal file
View File

@@ -0,0 +1,37 @@
#include <kmalloc.h>
#include <stddef.h>
extern void *__heap_start;
extern void *__heap_end;
void *_heap_top = (void *)0;
// simple 8-byte aligned linear allocation
void *simple_alloc(size_t size)
{
if (!_heap_top) {
_heap_top = __heap_start;
}
if (size & 0xff)
size = (size & ~0xff) + 0x100;
if ((uint64_t)_heap_top + size >= (uint64_t)__heap_end)
return (void *)0;
void *ret = _heap_top;
_heap_top = (void *)((uint64_t)_heap_top + size);
return ret;
}
void *kmalloc(size_t size)
{
return simple_alloc(size);
}
void kfree(void *ptr)
{
// not implemented for now
return;
}

89
lib/mbox.c Normal file
View File

@@ -0,0 +1,89 @@
/*
* Copyright (C) 2018 bzt (bztsrc@github)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
*/
#include <gpio.h>
#include <mbox.h>
/* mailbox message buffer */
volatile unsigned int __attribute__((aligned(16))) mbox[36];
#define VIDEOCORE_MBOX (MMIO_BASE+0x0000B880)
#define MBOX_READ ((volatile unsigned int*)(VIDEOCORE_MBOX+0x0))
#define MBOX_POLL ((volatile unsigned int*)(VIDEOCORE_MBOX+0x10))
#define MBOX_SENDER ((volatile unsigned int*)(VIDEOCORE_MBOX+0x14))
#define MBOX_STATUS ((volatile unsigned int*)(VIDEOCORE_MBOX+0x18))
#define MBOX_CONFIG ((volatile unsigned int*)(VIDEOCORE_MBOX+0x1C))
#define MBOX_WRITE ((volatile unsigned int*)(VIDEOCORE_MBOX+0x20))
#define MBOX_RESPONSE 0x80000000
#define MBOX_FULL 0x80000000
#define MBOX_EMPTY 0x40000000
#define MAILBOX_BASE MMIO_BASE + 0xb880
#define REQUEST_CODE 0x00000000
#define REQUEST_SUCCEED 0x80000000
#define REQUEST_FAILED 0x80000001
#define TAG_REQUEST_CODE 0x00000000
#define END_TAG 0x00000000
/**
* Make a mailbox call. Returns 0 on failure, non-zero on success
*/
int mbox_call(unsigned char ch)
{
unsigned int r = (((unsigned int)((unsigned long)&mbox)&~0xF) | (ch&0xF));
/* wait until we can write to the mailbox */
do{asm volatile("nop");}while(*MBOX_STATUS & MBOX_FULL);
/* write the address of our message to the mailbox with channel identifier */
*MBOX_WRITE = r;
/* now wait for the response */
while(1) {
/* is there a response? */
do{asm volatile("nop");}while(*MBOX_STATUS & MBOX_EMPTY);
/* is it a response to our message? */
if(r == *MBOX_READ)
/* is it a valid successful response? */
return mbox[1]==MBOX_RESPONSE;
}
return 0;
}
unsigned int get_board_revision(void)
{
mbox[0] = 36 * 4;
mbox[1] = REQUEST_CODE;
// tags
mbox[2] = MBOX_TAG_BOARD_REVISION;
mbox[3] = 4;
mbox[4] = TAG_REQUEST_CODE;
mbox[5] = 0; // value buffer
mbox[6] = END_TAG;
mbox_call(MBOX_CH_PROP);
return mbox[5];
}

12
lib/random.c Normal file
View File

@@ -0,0 +1,12 @@
#include <random.h>
const int _random_a = 100003;
const int _random_c = 114514 + 33;
const int _random_m = 1000000007;
int seed = 0;
int random(void)
{
seed = _random_a * (seed + _random_c) % _random_m;
return seed;
}

75
lib/shell.c Normal file
View File

@@ -0,0 +1,75 @@
#include <stddef.h>
#include <string.h>
#include <uart.h>
#include <mbox.h>
#include <shell.h>
#define INPUT_BUFLEN 1000
void help (void)
{
uart_puts(
"help : print this help menu" ENDL
"hello : print Hello World!" ENDL
"hwinfo: print hardware info" ENDL
"reboot: reboot the device" ENDL
);
}
void hello (void)
{
uart_puts("hello, world" ENDL);
}
void hwinfo (void)
{
unsigned int val = get_board_revision();
uart_puts("hwinfo: ");
uart_hex(val);
uart_puts(ENDL);
}
#define PM_PASSWORD 0x5a000000
#define PM_RSTC 0x3F10001c
#define PM_WDOG 0x3F100024
void set(long addr, unsigned int value) {
volatile unsigned int* point = (unsigned int*)addr;
*point = value;
}
void reset(int tick) { // reboot after watchdog timer expire
set(PM_RSTC, PM_PASSWORD | 0x20); // full reset
set(PM_WDOG, PM_PASSWORD | tick); // number of watchdog tick
}
void reboot(void)
{
reset(1 << 16);
}
int shell (void)
{
uart_puts("# ");
char buf[INPUT_BUFLEN], ch;
int sz = 0;
while ((ch = uart_getc()) != '\n') {
buf[sz++] = ch;
uart_send(ch);
}
uart_puts(ENDL);
buf[sz] = '\0';
if (strcmp(buf, "help") == 0) {
help();
} else if (strcmp(buf, "hello") == 0) {
hello();
} else if (strcmp(buf, "hwinfo") == 0) {
hwinfo();
} else if (strcmp(buf, "reboot") == 0) {
reboot();
}
return true;
}

35
lib/string.c Normal file
View File

@@ -0,0 +1,35 @@
#include <stddef.h>
#include <string.h>
int strcmp(const char *a, const char *b)
{
while (*a != '\0' && *b != '\0') {
if (*a != *b)
return (*a > *b) - (*a < *b);
a++, b++;
}
if (*a == '\0' && *b == '\0')
return 0;
if (*a == '\0')
return -1;
return 1;
}
void *memcpy(void *dest, const void *src, size_t count)
{
void *ret = dest;
for (const byte_t *p = src; count > 0; --count, ++p, ++dest)
*(byte_t *)dest = *p;
return ret;
}
void *memzero(void *start, void *end)
{
void *ret = start;
for (byte_t *p = start; p < (byte_t *)end; ++p)
*p = 0;
return ret;
}

103
lib/uart.c Normal file
View File

@@ -0,0 +1,103 @@
#include <uart.h>
#include <gpio.h>
/* Auxilary mini UART registers */
#define AUX_ENABLE ((volatile unsigned int*)(MMIO_BASE+0x00215004))
#define AUX_MU_IO ((volatile unsigned int*)(MMIO_BASE+0x00215040))
#define AUX_MU_IER ((volatile unsigned int*)(MMIO_BASE+0x00215044))
#define AUX_MU_IIR ((volatile unsigned int*)(MMIO_BASE+0x00215048))
#define AUX_MU_LCR ((volatile unsigned int*)(MMIO_BASE+0x0021504C))
#define AUX_MU_MCR ((volatile unsigned int*)(MMIO_BASE+0x00215050))
#define AUX_MU_LSR ((volatile unsigned int*)(MMIO_BASE+0x00215054))
#define AUX_MU_MSR ((volatile unsigned int*)(MMIO_BASE+0x00215058))
#define AUX_MU_SCRATCH ((volatile unsigned int*)(MMIO_BASE+0x0021505C))
#define AUX_MU_CNTL ((volatile unsigned int*)(MMIO_BASE+0x00215060))
#define AUX_MU_STAT ((volatile unsigned int*)(MMIO_BASE+0x00215064))
#define AUX_MU_BAUD ((volatile unsigned int*)(MMIO_BASE+0x00215068))
/**
* Set baud rate and characteristics (115200 8N1) and map to GPIO
*/
void uart_init()
{
register unsigned int r;
/* initialize UART */
*AUX_ENABLE |=1; // enable UART1, AUX mini uart
*AUX_MU_IER = 0;
*AUX_MU_CNTL = 0;
*AUX_MU_LCR = 3; // 8 bits
*AUX_MU_MCR = 0;
*AUX_MU_IER = 0;
*AUX_MU_IIR = 0xc6; // disable interrupts
*AUX_MU_BAUD = 270; // 115200 baud
/* map UART1 to GPIO pins */
r=*GPFSEL1;
r&=~((7<<12)|(7<<15)); // gpio14, gpio15
r|=(2<<12)|(2<<15); // alt5
*GPFSEL1 = r;
*GPPUD = 0; // enable pins 14 and 15
r=150; while(r--) { asm volatile("nop"); }
*GPPUDCLK0 = (1<<14)|(1<<15);
r=150; while(r--) { asm volatile("nop"); }
*GPPUDCLK0 = 0; // flush GPIO setup
*AUX_MU_CNTL = 3; // enable Tx, Rx
}
/**
* Send a character
*/
void uart_send(unsigned int c) {
/* wait until we can send */
do{asm volatile("nop");}while(!(*AUX_MU_LSR&0x20));
/* write the character to the buffer */
*AUX_MU_IO=c;
}
byte_t uart_getb()
{
byte_t r;
/* wait until something is in the buffer */
do{asm volatile("nop");}while(!(*AUX_MU_LSR&0x01));
/* read it and return */
r=(byte_t)(*AUX_MU_IO);
return r;
}
/**
* Receive a character
*/
char uart_getc() {
char r = (char)uart_getb();
/* convert carrige return to newline */
return r=='\r'?'\n':r;
}
/**
* Display a string
*/
void uart_puts(char *s) {
while(*s) {
/* convert newline to carrige return + newline */
if(*s=='\n')
uart_send('\r');
uart_send(*s++);
}
}
/**
* Display a binary value in hexadecimal
*/
void uart_hex(unsigned int d) {
unsigned int n;
int c;
for(c=28;c>=0;c-=4) {
// get highest tetrad
n=(d>>c)&0xF;
// 0-9 => '0'-'9', 10-15 => 'A'-'F'
n+=n>9?0x37:0x30;
uart_send(n);
}
}

Binary file not shown.

3
misc/config.txt Normal file
View File

@@ -0,0 +1,3 @@
initramfs initramfs.cpio 0x20000000
kernel=bootloader.img
arm_64bit=1

1
rootfs/poop.txt Normal file
View File

@@ -0,0 +1 @@
poop

15
scripts/upload.py Normal file
View File

@@ -0,0 +1,15 @@
from pwn import *
kernel = open("kernel8.img", "rb").read()
r = serialtube("/dev/ttyS5", convert_newlines=False)
input("@")
r.send(str(len(kernel))+"\n")
#print(r.recv())
#r.interactive()
print(r.recvuntil(b") : "))
r.send(kernel)
r.interactive()