Coming here after 10 days or so, but the time off the tracks was really necessary.
I did a very dire mistake and kept emulating for 80206
never actually realisizing that it is only a 16 bit microprocessor.
I was supposed to use 80306
.
A lot of time and energy rendered useless because I was trying to achieve and run 32 bit instructions on a 16 bit machine.
Not Cool.
So I decided to get off the track and devote time to reading, learning and understanding the structure of the processor. So that's what I did.
Now, with a little bit of confidence in this I was ready to move ahead.
In the last update I wrote the GDT because I techinically messed it up, but we managed to write our own bare minimum GDT with taking help from almost entirety of the internet.
Now I have to boot into 32 Bit protected mode. Before we move ahead, you need to know about Interupts & Pipelining; and you need to know about them properly to be able to understand what is happening.
But to save time, I'll give you an overview of what they are. You can read about them later on when you see feel like (if).
Interputs in a modern digital computer is basically a flag, that once raised by the processor which basically a response to an event which requires attention from the software/program. Consider an interupt like the break
statement in the C/C++ language. It breaks the current loop or flow of control and forces the program out of it.
An interupt does the same thing but with the processor. It is a signal sent to the processor which "interupts" the current process. Both hardware and software can create interupts.
Pipelining on the other hand, you can say is the precursor to multi-processing and multi-threading.
More information, and better information available here
Pipelining is a technique where multiple instructions are bundled or overlapped in exec
.
Pipelining is an implementation technique where multiple instructions are overlapped in execution. The computer pipeline is divided in stages. Each stage completes a part of an instruction in parallel. The stages are connected one to the next to form a pipe - instructions enter at one end, progress through the stages, and exit at the other end.
Pipelining does not decrease the time for individual instruction execution. Instead, it increases instruction throughput. The throughput of the instruction pipeline is determined by how often an instruction exits the pipeline.
In a nutshell, the following steps are supposed to be taken in order boot into 32-Bit Protected Mode.
1. Disable interrupts
2. Load our GDT
3. Set a bit on the CPU control register cr0
4. Flush the CPU pipeline by issuing a carefully crafted far 5. jump
6. Update all the segment registers
7. Update the stack
8. Call to a well-known label which contains the first useful 9. code in 32 bits
With that out of the way, let us understand the code. The steps above are achived in the switch file & once we'll call the main file.
[bits 16]
switch_to_pm:
cli ; 1. disable interrupts
lgdt [gdt_descriptor] ; 2. load the GDT descriptor
mov eax, cr0
or eax, 0x1 ; 3. set 32-bit mode bit in cr0
mov cr0, eax
jmp CODE_SEG:init_pm ; 4. far jump by using a different segment
[bits 32]
init_pm: ; we are now using 32-bit instructions
mov ax, DATA_SEG ; 5. update the segment registers
mov ds, ax
mov ss, ax
mov es, ax
mov fs, ax
mov gs, ax
mov ebp, 0x90000 ; 6. update the stack right at the top of the free space
mov esp, ebp
call BEGIN_PM ; 7. Call a well-known label with useful code
[org 0x7c00] ; bootloader offset
mov bp, 0x9000 ; set the stack
mov sp, bp
mov bx, MSG_REAL_MODE
call print ; This will be written after the BIOS messages
call switch_to_pm
jmp $ ; this will actually never be executed
%include "boot_sect_print.asm"
%include "32bit-gdt.asm"
%include "32bit-print.asm"
%include "32bit-switch.asm"
[bits 32]
BEGIN_PM: ; after the switch we will get here
mov ebx, MSG_PROT_MODE
call print_string_pm ; Note that this will be written at the top left corner
jmp $
MSG_REAL_MODE db "Started in 16-bit real mode", 0
MSG_PROT_MODE db "Loaded 32-bit protected mode", 0
; Check the conf you get, you should get 32
; The following below is the bootsector code, simplest bootloader.
; I decided to copy a rather professionally written code instead of using mine to be sure I don't mess up my computer again.
; bootsector
times 510-($-$$) db 0
dw 0xaa55