mirror of
https://github.com/cfenollosa/os-tutorial.git
synced 2024-10-27 20:34:19 +00:00
93 lines
3.9 KiB
Markdown
93 lines
3.9 KiB
Markdown
*Concepts you may want to Google beforehand: freestanding, uint32_t, size_t*
|
|
|
|
**Goal: Fix miscellaneous issues with our code**
|
|
|
|
The OSDev wiki has a section [which describes some issues with
|
|
JamesM's tutorial](http://wiki.osdev.org/James_Molloy%27s_Tutorial_Known_Bugs).
|
|
Since we followed his tutorial for lessons 18-22 (interrupts through malloc), we'll
|
|
need to make sure we fix any of the issues before moving on.
|
|
|
|
1. Wrong CFLAGS
|
|
---------------
|
|
|
|
We add `-ffreestanding` when compiling `.o` files, which includes `kernel_entry.o` and thus
|
|
`kernel.bin` and `os-image.bin`.
|
|
|
|
Before, we disabled libgcc (not libc) through the use of `-nostdlib` and we didn't re-enable
|
|
it for linking. Since this is tricky, we'll delete `-nostdlib`
|
|
|
|
`-nostdinc` was also passed to gcc, but we will need it for step 3, so let's delete it.
|
|
|
|
|
|
2. kernel.c `main()` function
|
|
-----------------------------
|
|
|
|
Modify `kernel/kernel.c` and change `main()` to `kernel_main()` since gcc recognizes "main" as
|
|
a special keyword and we don't want to mess with that.
|
|
|
|
Change `boot/kernel_entry.asm` to point to the new name accordingly.
|
|
|
|
To fix the `i386-elf-ld: warning: cannot find entry symbol _start; defaulting to 0000000000001000`
|
|
warning message, add a `global _start;` and define the `_start:` label in `boot/kernel_entry.asm`.
|
|
|
|
|
|
3. Reinvented datatypes
|
|
-----------------------
|
|
|
|
It looks like it was a bad idea to define non-standard data types like `u32` and such, since
|
|
C99 introduces standard fixed-width data types like `uint32_t`
|
|
|
|
We need to include `<stdint.h>` which works even in `-ffreestanding` (but requires stdlibs)
|
|
and use those data types instead of our own, then delete them on `type.h`
|
|
|
|
We also delete the underscores around `__asm__` and `__volatile__` since they aren't needed.
|
|
|
|
|
|
4. Improperly aligned `kmalloc`
|
|
-------------------------------
|
|
|
|
First, since `kmalloc` uses a size parameter, we'll use the correct data type `size_t` instead
|
|
of `u32int_t`. `size_t` should be used for all parameters which "count" stuff and cannot be
|
|
negative. Include `<stddef.h>`.
|
|
|
|
We will fix our `kmalloc` in the future, making it a proper memory manager and aligning data types.
|
|
For now, it will always return a new page-aligned memory block.
|
|
|
|
|
|
5. Missing functions
|
|
--------------------
|
|
|
|
We will implement the missing `mem*` functions in following lessons
|
|
|
|
|
|
6. Interrupt handlers
|
|
---------------------
|
|
`cli` is redundant, since we already established on the IDT entries if interrupts
|
|
are enabled within a handler using the `idt_gate_t` flags.
|
|
|
|
`sti` is also redundant, as `iret` loads the eflags value from the stack, which contains a
|
|
bit telling whether interrupts are on or off.
|
|
In other words the interrupt handler automatically restores interrupts whether or not
|
|
interrupts were enabled before this interrupt
|
|
|
|
On `cpu/isr.h`, `struct registers_t` has a couple issues.
|
|
First, the alleged `esp` is renamed to `useless`.
|
|
The value is useless because it has to do with the current stack context, not what was interrupted.
|
|
Then, we rename `useresp` to `esp`
|
|
|
|
We add `cld` just before `call isr_handler` on `cpu/interrupt.asm` as suggested
|
|
by the osdev wiki.
|
|
|
|
There is a final, important issue with `cpu/interrupt.asm`. The common stubs create an instance
|
|
of `struct registers` on the stack and then call the C handler. But that breaks the ABI, since
|
|
the stack belongs to the called function and they may change them as they please. It is needed
|
|
to pass the struct as a pointer.
|
|
|
|
To achieve this, edit `cpu/isr.h` and `cpu/isr.c` and change `registers_t r` into `registers_t *t`,
|
|
then, instead of accessing the fields of the struct via `.`, access the fields of the pointer via `->`.
|
|
Finally, in `cpu/interrupt.asm`, and add a `push esp` before calling both `isr_handler` and
|
|
`irq_handler` -- remember to also `pop eax` to clear the pointer afterwards.
|
|
|
|
Both current callbacks, the timer and the keyboard, also need to be changed to use a pointer to
|
|
`registers_t`.
|