flush_gdt:
lgdt [gdtr]
jmp 0x08:complete_flush
complete_flush:
mov ax, 0x10
mov ds, ax
mov es, ax
mov fs, ax
mov gs, ax
mov ss, ax
ret
I am unable to understand what this code does . flush_gdt
is a label okay , then lgdt [gdtr]
loads the 48-bit
pointer in gdtr
register and after that from jmp 0x08:complet_flush
.
What does jmp instruction do ? and then why are we moving 0x10 to ax and then subsequently to other registers
x86 supports two virtual memory schemes (read about it here):
Most operating systems want to to use paging and don't want the segmentation, but its must and can't just be disabled.
So the trick is to disable its effect as it wasn't there. This can usually be done by creating 4 large overlapped segments descriptors (beside the null segment):
all these segments starts from
0x00000000
up to0xffffffff
, so you end up with overlapped large segments that is privileged code and data, and non-privileged code and data in the same time. This should open up the virtual memory and disable the segmentation effect.The processor uses the segment selectors (segment registers
cs
,ds
,ss
...) to find out the right segment (once again, the segmentation is must).Every segment selector is 16 bit size and has the following layout (source):
The first two bits indicates that privilege level, x86 supports 4 levels, but only two of them actually used (
00
highest, and11
lowest).The third bit indicates the table should be used, mostly
0
, the GDT.If you interpreted the
0x08
that is loaded incs
, it will be in binary:and the
0x10
that is loaded inds
,ss
, ... :If you read the segment selectors of any user mode program you should see that the
cs
value is27
(0x1b
) which means:and the data selectors
ds
,ss
, ..., should store 35 (0x23
):The data segments selectors (registers), can be easily modified using simple
mov
instruction, but thecs
can't be used withmov
, so you usejmp 0x08:OFFSET
to load the segment configurations into the the code segment selector.