2023-09-05 07:46:32 +00:00
|
|
|
Kernel: Barebones
|
|
|
|
=================
|
|
|
|
|
2014-10-15 17:00:21 +00:00
|
|
|
*Concepts you may want to Google beforehand: kernel, ELF format, makefile*
|
|
|
|
|
|
|
|
**Goal: Create a simple kernel and a bootsector capable of booting it**
|
|
|
|
|
|
|
|
The kernel
|
|
|
|
----------
|
|
|
|
|
|
|
|
Our C kernel will just print an 'X' on the top left corner of the screen. Go ahead
|
|
|
|
and open `kernel.c`.
|
|
|
|
|
|
|
|
You will notice a dummy function that does nothing. That function will force us
|
|
|
|
to create a kernel entry routine which does not point to byte 0x0 in our kernel, but
|
|
|
|
to an actual label which we know that launches it. In our case, function `main()`.
|
|
|
|
|
|
|
|
`i386-elf-gcc -ffreestanding -c kernel.c -o kernel.o`
|
|
|
|
|
|
|
|
That routine is coded on `kernel_entry.asm`. Read it and you will learn how to
|
|
|
|
use `[extern]` declarations in assembly. To compile this file, instead of generating
|
|
|
|
a binary, we will generate an `elf` format file which will be linked with `kernel.o`
|
|
|
|
|
|
|
|
`nasm kernel_entry.asm -f elf -o kernel_entry.o`
|
|
|
|
|
|
|
|
|
|
|
|
The linker
|
|
|
|
----------
|
|
|
|
|
|
|
|
A linker is a very powerful tool and we only started to benefit from it.
|
|
|
|
|
|
|
|
To link both object files into a single binary kernel and resolve label references,
|
|
|
|
run:
|
|
|
|
|
|
|
|
`i386-elf-ld -o kernel.bin -Ttext 0x1000 kernel_entry.o kernel.o --oformat binary`
|
|
|
|
|
|
|
|
Notice how our kernel will be placed not at `0x0` in memory, but at `0x1000`. The
|
|
|
|
bootsector will need to know this address too.
|
|
|
|
|
|
|
|
|
|
|
|
The bootsector
|
|
|
|
--------------
|
|
|
|
|
|
|
|
It is very similar to the one in lesson 10. Open `bootsect.asm` and examine the code.
|
|
|
|
Actually, if you remove all the lines used to print messages on the screen, it accounts
|
|
|
|
to a couple dozen lines.
|
|
|
|
|
|
|
|
Compile it with `nasm bootsect.asm -f bin -o bootsect.bin`
|
|
|
|
|
|
|
|
|
|
|
|
Putting it all together
|
|
|
|
-----------------------
|
|
|
|
|
|
|
|
Now what? We have two separate files for the bootsector and the kernel?
|
|
|
|
|
|
|
|
Can't we just "link" them together into a single file? Yes, we can, and it's easy,
|
|
|
|
just concatenate them:
|
|
|
|
|
|
|
|
`cat bootsect.bin kernel.bin > os-image.bin`
|
|
|
|
|
|
|
|
|
|
|
|
Run!
|
|
|
|
----
|
|
|
|
|
|
|
|
You can now run `os-image.bin` with qemu.
|
|
|
|
|
|
|
|
Remember that if you find disk load errors you may need to play with the disk numbers
|
2014-10-15 17:02:48 +00:00
|
|
|
or qemu parameters (floppy = `0x0`, hdd = `0x80`). I usually use `qemu-system-i386 -fda os-image.bin`
|
2014-10-15 17:00:21 +00:00
|
|
|
|
|
|
|
You will see four messages:
|
|
|
|
|
|
|
|
- "Started in 16-bit Real Mode"
|
|
|
|
- "Loading kernel into memory"
|
|
|
|
- (Top left) "Landed in 32-bit Protected Mode"
|
|
|
|
- (Top left, overwriting previous message) "X"
|
|
|
|
|
|
|
|
Congratulations!
|
|
|
|
|
|
|
|
|
|
|
|
Makefile
|
|
|
|
--------
|
|
|
|
|
|
|
|
As a last step, we will tidy up the compilation process with a Makefile. Open the `Makefile`
|
|
|
|
script and examine its contents. If you don't know what a Makefile is, now is a good time
|
|
|
|
to Google and learn it, as this will save us a lot of time in the future.
|