2015-03-20 18:20:44 +00:00
|
|
|
|
|
|
|
**Goal: Clean the code a bit and parse user input**
|
|
|
|
|
2018-09-30 03:42:23 +00:00
|
|
|
In this lesson we will do two things. First, we will clean up the code a bit, so it is ready
|
2015-03-20 18:20:44 +00:00
|
|
|
for further lessons. During the previous ones I tried to put things in the most predictable places,
|
|
|
|
but it is also a good exercise to know when the code base is growing and adapt it to current
|
|
|
|
and further needs.
|
|
|
|
|
|
|
|
|
|
|
|
Code cleaning
|
|
|
|
-------------
|
|
|
|
|
|
|
|
First of all, we will quickly start to need more utility functions
|
|
|
|
for handling strings and so on. In a regular OS, this is called the C library,
|
|
|
|
or libc for short.
|
|
|
|
|
|
|
|
Right now we have a `utils.c` which we will split into `mem.c` and `string.c`, with their respective headers.
|
|
|
|
|
|
|
|
Second, we will create a new function `irq_install()` so that the kernel
|
|
|
|
only needs to perform one call to initialize all the IRQs. That function
|
|
|
|
is akin to `isr_install()` and placed on the same `irq.c`.
|
|
|
|
While we're here, we will disable the `kprint()` on `timer_callback()`
|
|
|
|
to avoid filling the screen with junk, now that we know that it works
|
|
|
|
properly.
|
|
|
|
|
|
|
|
There is not a clear distinction between `cpu/` and `drivers/`.
|
|
|
|
Keep in mind that I'm
|
|
|
|
creating this tutorial while following many others, and each of them
|
|
|
|
has a distinct folder structure. The only change we will do for now is to
|
|
|
|
move `drivers/ports.*` into `cpu/` since it is clearly cpu-dependent code.
|
|
|
|
`boot/` is also CPU-dependent code, but we will not mess with it until
|
|
|
|
we implement the boot sequence for a different machine.
|
|
|
|
|
2015-03-23 15:08:34 +00:00
|
|
|
There are more switches for the `CFLAGS` on the `Makefile`, since we will now
|
|
|
|
start creating higher-level functions for our C library and we don't want
|
|
|
|
the compiler to include any external code if we make a mistake with a declaration.
|
2018-09-25 12:26:55 +00:00
|
|
|
We also added some flags to turn warnings into errors, since an apparently minor mistake
|
2015-03-23 15:08:34 +00:00
|
|
|
converting pointers can blow up later on. This also forced us to modify some misc pointer
|
|
|
|
declarations in our code.
|
2015-03-20 18:20:44 +00:00
|
|
|
|
2015-03-23 19:03:41 +00:00
|
|
|
Finally, we'll add a macro to avoid warning-errors on unused parameters on `libc/function.h`
|
|
|
|
|
2015-03-20 18:20:44 +00:00
|
|
|
Keyboard characters
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
How to access the typed characters, then?
|
|
|
|
|
|
|
|
- When a key is pressed, the callback gets the ASCII code via a new
|
|
|
|
arrays which are defined at the beginning of `keyboard.c`
|
|
|
|
- The callback then appends that character to a buffer, `key_buffer`
|
|
|
|
- It is also printed on the screen
|
|
|
|
- When the OS wants to read user input, it calls `libc/io.c:readline()`
|
|
|
|
|
|
|
|
`keyboard.c` also parses backspace, by removing the last element
|
|
|
|
of the key buffer, and deleting it from the screen, by calling
|
|
|
|
`screen.c:kprint_backspace()`. For this we needed to modify a bit
|
|
|
|
`print_char()` to not advance the offset when printing a backspace
|
|
|
|
|
|
|
|
|
|
|
|
Responding to user input
|
|
|
|
------------------------
|
|
|
|
|
|
|
|
The keyboard callback checks for a newline, and then calls the kernel,
|
|
|
|
telling it that the user has input something. Out final libc function
|
|
|
|
is `strcmp()`, which compares two strings and returns 0 if they
|
|
|
|
are equal. If the user inputs "END", we halt the CPU.
|
|
|
|
|
|
|
|
This is the most basic shell ever, but you should be proud, because
|
|
|
|
we implemented it from scratch. Do you realize how cool this is?
|
|
|
|
|
|
|
|
If you want to, expand `kernel.c` to parse more stuff. In the future,
|
|
|
|
when we have a filesystem, we will allow the user to run some basic commands.
|