initial commit
This commit is contained in:
commit
1e778fc163
30
Makefile
Normal file
30
Makefile
Normal file
|
@ -0,0 +1,30 @@
|
|||
GCCPATH = /opt/gcc-arm/bin
|
||||
|
||||
CC = $(GCCPATH)/aarch64-none-elf-gcc
|
||||
LD = $(GCCPATH)/aarch64-none-elf-ld
|
||||
OBJCOPY = $(GCCPATH)/aarch64-none-elf-objcopy
|
||||
GCCFLAGS = -Wall -O2 -ffreestanding -nostdinc -nostdlib -nostartfiles
|
||||
|
||||
OUTDIR = ./out/bin
|
||||
|
||||
HFILES = $(wildcard *.h include/*.h)
|
||||
CFILES = $(wildcard *.c src/*.c)
|
||||
_OFILES = $(CFILES:.c=.o)
|
||||
OFILES = $(_OFILES:src/%=out/%)
|
||||
|
||||
all: clean kernel8.img
|
||||
|
||||
out/%.o: src/%.c $(HFILES)
|
||||
@echo "Building $@"
|
||||
@mkdir -p $(@D)
|
||||
@$(CC) -Iinclude -c $< -o $@ $(GCCLFAGS)
|
||||
|
||||
kernel8.img: $(OFILES)
|
||||
@echo "Linking kernel"
|
||||
@mkdir -p bin
|
||||
@$(LD) -nostdlib $(OFILES) -T link.ld -o out/kernel8.elf
|
||||
@$(OBJCOPY) -O binary out/kernel8.elf bin/kernel8.img
|
||||
@echo "Done!"
|
||||
|
||||
clean:
|
||||
@/bin/rm -rf out/ > /dev/null 2> /dev/null || true
|
6
burn.sh
Executable file
6
burn.sh
Executable file
|
@ -0,0 +1,6 @@
|
|||
#!/bin/bash
|
||||
|
||||
mount -t drvfs f: /mnt/f
|
||||
cp bin/kernel8.img /mnt/f
|
||||
umount /mnt/f
|
||||
echo "Done!"
|
6
include/convert.h
Normal file
6
include/convert.h
Normal file
|
@ -0,0 +1,6 @@
|
|||
#pragma once
|
||||
|
||||
#include "stdint.h"
|
||||
|
||||
int stoi(const char* string);
|
||||
char* itoa(int number, char* string);
|
15
include/gpio.h
Normal file
15
include/gpio.h
Normal file
|
@ -0,0 +1,15 @@
|
|||
#include "stdint.h"
|
||||
|
||||
// Functions for peeking/poking into memory
|
||||
void poke(int64_t addr, uint32_t val);
|
||||
uint32_t peek(int64_t addr);
|
||||
|
||||
|
||||
int32_t gpio_call(uint32_t pin, uint32_t reg, uint32_t value, uint32_t field_size);
|
||||
|
||||
uint32_t gpio_set (uint32_t pin, uint32_t value);
|
||||
uint32_t gpio_clear (uint32_t pin, uint32_t value);
|
||||
uint32_t gpio_fselect(uint32_t pin, uint32_t value);
|
||||
uint32_t gpio_pull (uint32_t pin, uint32_t value);
|
||||
|
||||
void gpio_setAltMode5(uint32_t pin);
|
11
include/io.h
Normal file
11
include/io.h
Normal file
|
@ -0,0 +1,11 @@
|
|||
#pragma once
|
||||
|
||||
#include "stdint.h"
|
||||
|
||||
void uart_init();
|
||||
|
||||
void uart_putchar(char c);
|
||||
void uart_puts(const char* string);
|
||||
|
||||
char uart_getchar();
|
||||
char* uart_gets(char* string);
|
13
include/stdint.h
Normal file
13
include/stdint.h
Normal file
|
@ -0,0 +1,13 @@
|
|||
#pragma once
|
||||
|
||||
typedef char int8_t;
|
||||
typedef short int16_t;
|
||||
typedef int int32_t;
|
||||
typedef long int64_t;
|
||||
|
||||
typedef unsigned char uint8_t;
|
||||
typedef unsigned short uint16_t;
|
||||
typedef unsigned int uint32_t;
|
||||
typedef unsigned long uint64_t;
|
||||
|
||||
typedef uint64_t size_t;
|
19
link.ld
Normal file
19
link.ld
Normal file
|
@ -0,0 +1,19 @@
|
|||
SECTIONS
|
||||
{
|
||||
. = 0x80000;
|
||||
.text : { KEEP(*(.text.boot)) *(.text .text.* .gnu.linkonce.t*) }
|
||||
.rodata : { *(.rodata .rodata.* .gnu.linkonce.r*) }
|
||||
PROVIDE(_data = .);
|
||||
.data : { *(.data .data.* .gnu.linkonce.d*) }
|
||||
.bss (NOLOAD) : {
|
||||
. = ALIGN(16);
|
||||
__bss_start = .;
|
||||
*(.bss .bss.*)
|
||||
*(COMMON)
|
||||
__bss_end = .;
|
||||
}
|
||||
_end = .;
|
||||
|
||||
/DISCARD/ : { *(.comment) *(.gnu*) *(.note*) *(.eh_frame*) }
|
||||
}
|
||||
__bss_size = (__bss_end - __bss_start) >> 3;
|
29
src/boot.S
Normal file
29
src/boot.S
Normal file
|
@ -0,0 +1,29 @@
|
|||
.section ".text.boot"
|
||||
|
||||
.global _start
|
||||
|
||||
_start:
|
||||
# Check if on main core
|
||||
mrs x1, mpidr_el1
|
||||
and x1, x1, #3
|
||||
cbz x1, 2f
|
||||
|
||||
1: # Infinite loop
|
||||
b 1b
|
||||
|
||||
2: # Set up stack
|
||||
ldr x1, =_start
|
||||
mov sp, x1
|
||||
|
||||
# memzero the BSS section
|
||||
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 main() in C
|
||||
ldr x4, =_start
|
||||
bl main
|
||||
b 1b
|
54
src/convert.c
Normal file
54
src/convert.c
Normal file
|
@ -0,0 +1,54 @@
|
|||
#include "convert.h"
|
||||
#include "io.h"
|
||||
|
||||
int stoi(const char* string)
|
||||
{
|
||||
int output = 0;
|
||||
int32_t sign = 1;
|
||||
if(*string == '-')
|
||||
{
|
||||
string++;
|
||||
sign = -1;
|
||||
}
|
||||
|
||||
|
||||
while(*string >= '0' && *string <= '9')
|
||||
{
|
||||
output *= 10;
|
||||
output += sign * (*string++ - '0');
|
||||
}
|
||||
|
||||
return output;
|
||||
}
|
||||
|
||||
char* itoa(int number, char* string)
|
||||
{
|
||||
volatile char* temp = (volatile char*)0x40000000;
|
||||
int i = 0;
|
||||
if(number == 0)
|
||||
{
|
||||
temp[i++] = '0';
|
||||
}
|
||||
else
|
||||
{
|
||||
for(int n = (number < 0 ? -number : number); n > 0; n /= 10)
|
||||
{
|
||||
temp[i++] = '0' + (n % 10);
|
||||
}
|
||||
}
|
||||
|
||||
if(number < 0)
|
||||
{
|
||||
temp[i++] = '-';
|
||||
}
|
||||
|
||||
temp[i] = '\0';
|
||||
string[i] = '\0';
|
||||
|
||||
while(*temp)
|
||||
{
|
||||
string[--i] = *temp++;
|
||||
}
|
||||
|
||||
return string;
|
||||
}
|
76
src/gpio.c
Normal file
76
src/gpio.c
Normal file
|
@ -0,0 +1,76 @@
|
|||
#include "gpio.h"
|
||||
|
||||
void poke(int64_t addr, uint32_t val)
|
||||
{
|
||||
*(volatile uint32_t*)addr = val;
|
||||
}
|
||||
|
||||
uint32_t peek(int64_t addr)
|
||||
{
|
||||
return *(volatile uint32_t*)addr;
|
||||
}
|
||||
|
||||
enum
|
||||
{
|
||||
PERIPHERAL_BASE = 0xFE000000,
|
||||
|
||||
GPFSEL0 = PERIPHERAL_BASE + 0x200000,
|
||||
GPSET0 = PERIPHERAL_BASE + 0x20001C,
|
||||
GPCLR0 = PERIPHERAL_BASE + 0x200028,
|
||||
GPPUPPDN0 = PERIPHERAL_BASE + 0x2000E4
|
||||
};
|
||||
|
||||
enum
|
||||
{
|
||||
MAX_PINS = 57
|
||||
};
|
||||
|
||||
int32_t gpio_call(uint32_t pin, uint32_t reg, uint32_t value, uint32_t field_size)
|
||||
{
|
||||
// Calculate a mask for the field. The mask is a string of 1's that is field_size long
|
||||
uint32_t field_mask = (1 << field_size) - 1;
|
||||
|
||||
// Can't access pins that don't exist
|
||||
if(pin > MAX_PINS) return 0;
|
||||
|
||||
// Can't write values that exceed the width of the field
|
||||
if(value > field_mask) return 0;
|
||||
|
||||
uint32_t fields_per_register = 32 / field_size; // Number of fields per register
|
||||
uint32_t register_addr = reg + (pin / fields_per_register) * 4; // Offset of register to base register. E.g. GPSET1 = GPSET0 + 4 bytes
|
||||
uint32_t field_offset = (pin % fields_per_register) * field_size; // Number of bits the field is offset into the register
|
||||
|
||||
uint32_t current_value = peek(register_addr);
|
||||
current_value &= ~(field_mask << field_offset); // zero the field
|
||||
current_value |= value << field_offset; // set the field
|
||||
poke(register_addr, current_value); // poke back into memory
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
uint32_t gpio_set(uint32_t pin, uint32_t value)
|
||||
{
|
||||
return gpio_call(pin, GPSET0, value, 1);
|
||||
}
|
||||
|
||||
uint32_t gpio_clear(uint32_t pin, uint32_t value)
|
||||
{
|
||||
return gpio_call(pin, GPCLR0, value, 1);
|
||||
}
|
||||
|
||||
uint32_t gpio_fselect(uint32_t pin, uint32_t value)
|
||||
{
|
||||
return gpio_call(pin, GPFSEL0, value, 3);
|
||||
}
|
||||
|
||||
uint32_t gpio_pull(uint32_t pin, uint32_t value)
|
||||
{
|
||||
return gpio_call(pin, GPPUPPDN0, value, 2);
|
||||
}
|
||||
|
||||
void gpio_setAltMode5(uint32_t pin)
|
||||
{
|
||||
// UART1 requires the pins to float
|
||||
gpio_pull(pin, 0);
|
||||
gpio_fselect(pin, 2);
|
||||
}
|
79
src/io.c
Normal file
79
src/io.c
Normal file
|
@ -0,0 +1,79 @@
|
|||
#include "io.h"
|
||||
#include "gpio.h"
|
||||
|
||||
// Mini UART registers
|
||||
enum
|
||||
{
|
||||
AUX_BASE = 0xFE000000 + 0x215000,
|
||||
|
||||
AUX_ENABLES = AUX_BASE + 0x04,
|
||||
AUX_MU_IO_REG = AUX_BASE + 0x40,
|
||||
AUX_MU_IER_REG = AUX_BASE + 0x44,
|
||||
AUX_MU_IIR_REG = AUX_BASE + 0x48,
|
||||
AUX_MU_LCR_REG = AUX_BASE + 0x4C,
|
||||
AUX_MU_MCR_REG = AUX_BASE + 0x50,
|
||||
AUX_MU_LSR_REG = AUX_BASE + 0x54,
|
||||
AUX_MU_CNTL_REG = AUX_BASE + 0x60,
|
||||
AUX_MU_STAT_REG = AUX_BASE + 0x64,
|
||||
AUX_MU_BAUD_REG = AUX_BASE + 0x68,
|
||||
|
||||
AUX_UART_CLOCK = 500000000,
|
||||
UART_MAX_QUEUE = 16 * 1024
|
||||
};
|
||||
|
||||
void uart_init()
|
||||
{
|
||||
poke(AUX_ENABLES, 1); // Enable UART1
|
||||
poke(AUX_MU_LCR_REG, 3); // 8 bit communication
|
||||
poke(AUX_MU_MCR_REG, 0); // Pull RTS high
|
||||
poke(AUX_MU_IER_REG, 0);
|
||||
poke(AUX_MU_IIR_REG, 0xC6); // Disable interrupts
|
||||
poke(AUX_MU_BAUD_REG, AUX_UART_CLOCK / (112500 * 8) - 1);
|
||||
gpio_setAltMode5(14);
|
||||
gpio_setAltMode5(15);
|
||||
poke(AUX_MU_CNTL_REG, 3);
|
||||
}
|
||||
|
||||
uint32_t uart_isReceiveFIFOReady() { return peek(AUX_MU_LSR_REG) & 1; }
|
||||
uint32_t uart_isTransmitFIFOReady() { return peek(AUX_MU_LSR_REG) & (1 << 5); }
|
||||
|
||||
void uart_writeBlocking(uint8_t ch)
|
||||
{
|
||||
while(!uart_isTransmitFIFOReady());
|
||||
poke(AUX_MU_IO_REG, (uint32_t)ch);
|
||||
}
|
||||
|
||||
char uart_getchar()
|
||||
{
|
||||
while(!uart_isReceiveFIFOReady());
|
||||
char in = (char)peek(AUX_MU_IO_REG);
|
||||
uart_putchar(in);
|
||||
if(in == '\r')
|
||||
uart_writeBlocking('\n');
|
||||
return in;
|
||||
}
|
||||
|
||||
char* uart_gets(char* string)
|
||||
{
|
||||
char* start = string;
|
||||
do
|
||||
{
|
||||
*string = uart_getchar();
|
||||
} while(*string++ != '\r');
|
||||
|
||||
*(string - 1) = '\0';
|
||||
return start;
|
||||
}
|
||||
|
||||
void uart_putchar(char ch)
|
||||
{
|
||||
if(ch == '\n')
|
||||
uart_writeBlocking('\r');
|
||||
uart_writeBlocking(ch);
|
||||
}
|
||||
|
||||
void uart_puts(const char* string)
|
||||
{
|
||||
while(*string)
|
||||
uart_putchar(*string++);
|
||||
}
|
93
src/kernel.c
Normal file
93
src/kernel.c
Normal file
|
@ -0,0 +1,93 @@
|
|||
#include "io.h"
|
||||
#include "convert.h"
|
||||
|
||||
void print_clippy();
|
||||
|
||||
void main()
|
||||
{
|
||||
const char* hex_chars = "0123456789ABCDEF";
|
||||
register int* start_address asm ("x4");
|
||||
long int start = (long int)start_address;
|
||||
|
||||
uart_init();
|
||||
print_clippy();
|
||||
uart_puts("Boot successful! \n");
|
||||
uart_puts("Started execution at 0x");
|
||||
for(int i = 0; i < 8; i++)
|
||||
{
|
||||
uart_putchar(hex_chars[start & 0xF]);
|
||||
start >>= 4;
|
||||
}
|
||||
|
||||
char buffer[1024];
|
||||
for(;;)
|
||||
{
|
||||
/*
|
||||
uart_puts("\n\n📎 ");
|
||||
int a = stoi(uart_gets(buffer));
|
||||
int b = -a;
|
||||
|
||||
uart_puts("a: ");
|
||||
uart_puts(itoa(a, buffer));
|
||||
uart_puts("\nb: ");
|
||||
uart_puts(itoa(b, buffer));
|
||||
*/
|
||||
|
||||
|
||||
uart_puts("\n\nPlease enter first operand\n");
|
||||
uart_puts("📎 ");
|
||||
int a = stoi(uart_gets(buffer));
|
||||
|
||||
uart_puts("\nPlease enter second operand\n");
|
||||
uart_puts("📎 ");
|
||||
int b = stoi(uart_gets(buffer));
|
||||
|
||||
uart_puts("\n");
|
||||
uart_puts(itoa(a, buffer));
|
||||
uart_puts(" * ");
|
||||
uart_puts(itoa(b, buffer));
|
||||
uart_puts(" = ");
|
||||
uart_puts(itoa(a * b, buffer));
|
||||
|
||||
|
||||
// uart_puts(buffer);
|
||||
}
|
||||
|
||||
__asm("hlt");
|
||||
}
|
||||
|
||||
void print_clippy()
|
||||
{
|
||||
uart_puts(" ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ \n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠠⡀⢄⢀⢀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⢐⠌⡢⢑⠌⠢⢑⢐⠅⡢⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⢤⣤⣴⢴⡦⣧⣌⠄⠀⠀⠀⠀⠀⠑⡰⠨⡊⠄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ \n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢰⣟⣯⣷⣻⣽⣯⢷⣟⡆⠀⠀⠀⠀⠀⠀⠐⢅⢊⢌⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⢘⠨⡊⢔⠈⠉⠁⠀⠀⠀⠀⠀⠀⡤⡤⣅⣆⡢⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠊⠔⠌⠂⠀⠀⠀⠀⠀⠀⠀⠀⠘⢟⣿⢽⣷⣻⢷⢦⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠠⢐⣨⡴⣤⣢⡈⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⠩⡊⡛⢟⣿⣻⠂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⠈⣴⢿⡽⣯⢿⣞⡿⣎⠀⠀⠀⠀⠀⠀⠀⠀⢀⠀⠂⠑⠈⠄⠀⠈⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠄⠠⣻⣽⢿⡽⣯⢿⣽⣻⠀⡀⠀⠀⠀⠀⠠⣈⣤⢦⣦⣂⣁⠀⠄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠄⠀⠙⣾⣻⣽⣻⣽⢷⠋⠀⠀⠀⠀⠀⠈⣴⣟⣾⣻⣞⣿⣺⢧⠀⠈⢀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠀⡀⠁⠍⠑⠁⠁⠀⠀⠀⠀⠀⠀⡈⣾⣗⣯⣷⣻⣗⣿⣻⠀⠁⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠁⠠⡠⢐⠀⠈⠀⠀⠀⠀⠀⠀⠀⠀⠱⣯⡷⣟⣾⡽⣞⠏⡀⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡑⢌⢂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠁⠈⠙⠙⠓⢉⠈⢀⠀⠐⠀⠀⠠⠠⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡪⢐⠔⠀⠀⠀⠀⠠⡠⡀⠀⠀⠀⠀⠀⡅⠢⡁⠀⠀⠀⠀⠀⢀⢌⠪⠨⠂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡪⢐⠌⠀⠀⠀⠀⡑⡐⢌⠀⠀⠀⠀⠐⢌⠪⡀⠀⠀⠀⠀⠠⡑⢄⠕⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⡪⢐⠡⠀⠀⠀⠀⠌⡌⠢⠀⠀⠀⠀⠈⡢⢑⠄⠀⠀⠀⠠⡑⢌⠂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢌⠢⡡⠀⠀⠀⠀⠡⡊⢌⠀⠀⠀⠀⠀⡪⢐⠌⠀⠀⠀⢌⢌⠢⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠢⡑⡐⡀⠀⠀⠀⠡⢊⠔⡀⠀⠀⠀⠀⢌⠢⡑⠀⠀⠀⡢⠢⠡⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢈⠢⡑⡀⠀⠀⠀⠈⡢⢑⠄⠀⠀⠀⠀⢐⠅⡊⠄⠀⠀⡊⢌⢊⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠠⢑⠔⡐⡀⠀⠀⠀⢌⢂⠪⠀⠀⠀⠀⢐⠅⡪⠀⠀⠀⢌⠢⠡⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠡⡊⠔⢄⠀⠀⠀⠀⠪⠨⡊⢄⢀⢀⠢⡑⡐⠁⠀⠀⠠⡑⠅⠅⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠐⠌⡪⢐⠄⠀⠀⠀⠈⠈⠢⠡⠢⡡⢑⠐⠁⠀⠀⠀⠀⡪⠨⡂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠨⢂⢅⠢⠀⠀⠀⠀⠀⠀⠁⠁⠀⠀⠀⠀⠀⠀⠀⠀⢌⢌⠢⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠡⢂⠕⡡⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢅⠢⡡⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠐⠡⠢⡡⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠨⡂⢕⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠈⢊⠔⡑⢄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢄⠣⠨⡂⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠊⢔⠡⡡⡁⡀⢀⠀⠀⡀⢀⠌⢌⠢⠡⡃⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠑⡐⠌⡌⡂⡊⡢⢊⠔⢌⠢⠑⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠀⠈⠂⠈⠂⠈⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ \n");
|
||||
uart_puts("⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀ ⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀\n");
|
||||
|
||||
}
|
Loading…
Reference in a new issue