In the last blog posting I have talked about how to read files from a FAT12 partition and print out its loaded content to the screen. In today’s blog posting I want to talk more about how to implement a Second Stage Boot Loader that can be loaded into memory for its execution. But let’s talk first, why we need a Second Stage Boot Loader.
Second Stage Boot Loader – Why?
In my first blog posting about OS development, we have talked about how the BIOS loads the boot sector at the memory location 0x7C00 and executes it. The restriction of the boot sector is that it has only a size of 512 bytes (one disk sector) and within that size you can’t implement that much functionality with that restriction.
Therefore, you need a Second Stage Boot Loader that gets loaded into memory and prepares the CPU and the actual OS kernel for its execution. A Second Stage Boot Loader must implement the following tasks:
- Getting date and time information from the BIOS
- Detecting the Memory Map
- Retrieving the supported Video Modes
- Enabling the A20 Gate
- Switching the CPU into x32 Protected Mode or x64 Long Mode
- Loading the real OS Kernel from Disk
- Executing the OS Kernel
As you can see from this list, it would be impossible to do all these things within 512 bytes – or even 510 bytes when you subtract the magical byte pattern 0xAA55 at the end of the boot sector.
Implementing the Second Stage Boot Loader
Let’s talk now how to implement the Second Stage Boot Loader. In my case the Second Stage Boot Loader is x16 based so that I still have access to the BIOS interrupts – otherwise you can’t implement the above-mentioned tasks. One of the first things that the Second Stage Boot Loader implements is getting the date and time information from the BIOS.
This information will then be passed over to the Kernel who must keep them up to date – as precise as possible. The following listing shows the GetDate and GetTime functions that uses the BIOS interrupt 0x1A to get this information.
;================================================= ; This function retrieves the date from the BIOS. ;================================================= GetDate: ; Get the current date from the BIOS MOV AH, 0x4 INT 0x1A ; Century PUSH CX MOV AL, CH CALL Bcd2Decimal MOV [Year1], AX POP CX ; Year MOV AL, CL CALL Bcd2Decimal MOV [Year2], AX ; Month MOV AL, DH CALL Bcd2Decimal MOV [Month], AX ; Day MOV AL, DL CALL Bcd2Decimal MOV [Day], AX ; Calculate the whole year (e.g. "20" * 100 + "22" = 2022) MOV AX, [Year1] MOV BX, 100 MUL BX MOV BX, [Year2] ADD AX, BX MOV [Year], AX RET ;================================================= ; This function retrieves the time from the BIOS. ;================================================= GetTime: ; Get the current time from the BIOS MOV AH, 0x2 INT 0x1A ; Hour PUSH CX MOV AL, CH CALL Bcd2Decimal MOV [Hour], AX POP CX ; Minute MOV AL, CL CALL Bcd2Decimal MOV [Minute], AX ; Second MOV AL, DH CALL Bcd2Decimal MOV [Second], AX RET
As you can see from the previous listing, the BIOS interrupt 0x1A returns the current date and time as BCD numbers, therefore the function Bcd2Decimal does the conversion to decimal values:
;========================================================== ; This function converts a BCD number to a decimal number. ;========================================================== Bcd2Decimal: MOV CL, AL SHR AL, 4 MOV CH, 10 MUL CH AND CL, 0Fh ADD AL, CL RET
Afterwards the retrieved information is printed to the screen through the functions PrintString and PrintDecimal.
;================================================ ; This function prints a whole string, where the ; input string is stored in the register "SI" ;================================================ PrintString: ; Set the TTY mode MOV AH, 0xE INT 10 ; Set the input string MOV AL, [SI] CMP AL, 0 JE PrintString_End INT 0x10 INC SI JMP PrintString PrintString_End: RET ;================================================ ; This function prints out a decimal number ; that is stored in the register AX. ;================================================ PrintDecimal: MOV CX, 0 MOV DX, 0 PrintDecimal_Start: CMP AX ,0 JE PrintDecimal_Print MOV BX, 10 DIV BX PUSH DX INC CX XOR DX, DX JMP PrintDecimal_Start PrintDecimal_Print: CMP CX, 0 JE PrintDecimal_Exit POP DX ; Add 48 so that it represents the ASCII value of digits MOV AL, DL ADD AL, 48 MOV AH, 0xE INT 0x10 DEC CX JMP PrintDecimal_Print PrintDecimal_Exit: RET
Over the next blog postings, the implementation of the Second Stage Boot Loader will be enhanced with the previous mentioned tasks.
Loading and executing the Second Stage Boot Loader
The Second Stage Boot Loader code will be compiled by NASM into a raw binary file and then it will be added as an additional file to our final floppy image. Our boot loader code can then read that file with the implementation from the last blog posting into memory for its code execution. The following code shows how the Second Stage Boot Loader is loaded into memory – the variable KAOSLDR_OFFSET currently points to the memory location 0x2000.
; Load the KAOSLDR.BIN file into memory MOV CX, 11 LEA SI, [SecondStageFileName] LEA DI, [FileName] REP MOVSB MOV WORD [Loader_Offset], KAOSLDR_OFFSET CALL LoadRootDirectory ; Execute the KAOSLDR.BIN file... CALL KAOSLDR_OFFSET
Executing the Second Stage Boot Loader code is quite easy: we just do a simple CALL to the memory location KAOSLDR_OFFSET.
Summary
Reading an additional binary file into memory for its execution is not that complicated – if we are in x16 Real Mode, who offers us the necessary BIOS interrupts. Today we have used this functionality to load and execution the Second Stage Boot Loader of our OS. This additional boot loader code is necessary because we can’t do that many things within the limited boot sector size of 512 bytes. In the next blog posting we will continue with our Second Stage Boot Loader code and will finally switch the CPU into x64 Long Mode. Stay tuned…