bookstack

Shell

Goal: Clean the code a bit and parse user input

In this lesson we will do two things. First, we will clean up the code a bit, so it is ready 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.

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. We also added some flags to turn warnings into errors, since an apparently minor mistake converting pointers can blow up later on. This also forced us to modify some misc pointer declarations in our code.

Finally, we’ll add a macro to avoid warning-errors on unused parameters on libc/function.h

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.