Jason Xu 9c5c2813a0 Update QEMU command in all README and Makefile | 2 years ago | |
---|---|---|
.. | ||
Makefile | 5 years ago | |
Makefile.clang | 2 years ago | |
Makefile.gcc | 2 years ago | |
OLVASSEL.md | 2 years ago | |
README.md | 2 years ago | |
gpio.h | 6 years ago | |
kernel8.img | 4 years ago | |
link.ld | 6 years ago | |
main.c | 6 years ago | |
mbox.c | 6 years ago | |
mbox.h | 6 years ago | |
sprintf.c | 6 years ago | |
sprintf.h | 6 years ago | |
start.S | 3 years ago | |
uart.c | 3 years ago | |
uart.h | 6 years ago |
Before we can improve our exception handler, we are going to need some functions very well known from the C library. Since we are programming bare metal, we don't have libc, therefore we have to implement printf() on our own.
$ qemu-system-aarch64 -M raspi3b -kernel kernel8.img -serial stdio
Hello World!
This is character 'A', a hex number: 7FFF and in decimal: 32767
Padding test: '00007FFF', ' -123'
The interesting part. We heavily rely on our compiler's features to handle variable length argument list. As usual in these tutorials, it's not a fully featured, but rather a bare minimum implementation. Supports '%s', '%c', '%d' and '%x'. Padding is limited, only right alignment with leading zeros for hex and spaces for decimal.
sprintf(dst, fmt, ...)
same as printf, but stores result in a string
vsprintf(dst, fmt, va)
a variant that receives an argument list parameter instead of a variable length list of arguments.
printf(fmt, ...)
the good old C library function. Uses the sprintf function above and then outputs the string
in the same way as uart_puts() did. Since we have '%x', uart_hex() became unnecessary, therefore removed.
Although we are not going to use floats and doubles, gcc built-ins might. So we have to enable the FPU
coprocessor to avoid "undefined instruction" exceptions. Also, in a lack of a proper exception handler,
we have a dummy exc_handler
stub this time.
We test our printf implementation.