2017-11-09 23:21:49 +00:00
|
|
|
import serial
|
|
|
|
|
2017-11-06 21:44:17 +00:00
|
|
|
type
|
|
|
|
GDTEntry{.packed.} = object
|
|
|
|
limit_low: uint16
|
|
|
|
base_low: uint16
|
|
|
|
base_middle: uint8
|
|
|
|
access: uint8
|
2017-11-09 23:21:49 +00:00
|
|
|
flags_and_limit_mid: uint8
|
2017-11-06 21:44:17 +00:00
|
|
|
base_high: uint8
|
2017-11-09 23:21:49 +00:00
|
|
|
GDTPointer{.packed.} = object
|
2017-11-06 21:44:17 +00:00
|
|
|
limit: uint16 # Size of GDT
|
|
|
|
base: uint32 # GDT memory location
|
2017-11-09 23:21:49 +00:00
|
|
|
AccessByte = uint8
|
|
|
|
Flags = uint8
|
2017-11-21 19:20:05 +00:00
|
|
|
GDT{.packed.} = object
|
|
|
|
descriptor: GDTPointer
|
|
|
|
entries: array[0..2, GDTEntry]
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-21 19:20:05 +00:00
|
|
|
const
|
|
|
|
gdtAddress: uint32 = 0x00000800
|
|
|
|
gdt = cast[ptr GDT](gdtAddress)
|
2017-11-06 21:44:17 +00:00
|
|
|
|
|
|
|
{.push stackTrace:off.}
|
2017-11-21 19:20:05 +00:00
|
|
|
proc gdtFlush(){.inline,asmNoStackFrame.} =
|
2017-11-06 21:44:17 +00:00
|
|
|
asm """
|
2017-11-09 23:21:49 +00:00
|
|
|
cli
|
2017-11-21 19:20:05 +00:00
|
|
|
lgdtl [`0x00000800`]
|
|
|
|
movw $0x10, %ax
|
|
|
|
movw %ax, %ds
|
|
|
|
movw %ax, %es
|
|
|
|
movw %ax, %fs
|
|
|
|
movw %ax, %gs
|
|
|
|
ljmp $0x08, $next
|
|
|
|
next:
|
2017-11-06 21:44:17 +00:00
|
|
|
"""
|
|
|
|
{.pop.}
|
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
proc gdtSetGate(num: int, base: uint16, limit: uint32, access: AccessByte, flags: Flags) =
|
2017-11-06 21:44:17 +00:00
|
|
|
# Setup the descriptor base address
|
2018-04-17 19:28:25 +01:00
|
|
|
gdt.entries[num].base_low = uint16(base and 0xFFFF)
|
|
|
|
gdt.entries[num].base_middle = uint8((base shr 16) and 0xFF)
|
|
|
|
gdt.entries[num].base_high = uint8((base shr 24) and 0xFF)
|
2017-11-06 21:44:17 +00:00
|
|
|
|
|
|
|
# Setup the descriptor limits
|
2018-04-17 19:28:25 +01:00
|
|
|
gdt.entries[num].limit_low = uint16(limit and 0xFFFF)
|
|
|
|
gdt.entries[num].flags_and_limit_mid = uint8(((limit shr 16) and 0x0F) or flags)
|
2017-11-21 19:20:05 +00:00
|
|
|
gdt.entries[num].access = access
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
proc createAccessByte(priv: range[0..3], directionConform: bool, readWrite: bool): AccessByte =
|
|
|
|
result = 0b10011000'u8 # Present and executable bit
|
|
|
|
result = result or (uint8(priv) shl 5)
|
|
|
|
if directionConform:
|
|
|
|
result = result or 0b00000100'u8
|
|
|
|
if readWrite:
|
|
|
|
result = result or 0b00000010'u8
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
proc createFlags(granularity: bool, size: bool): Flags =
|
|
|
|
if granularity:
|
|
|
|
result = result or 0b10000000'u8
|
|
|
|
if size:
|
|
|
|
result = result or 0b01000000'u8
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
proc gdtInstall*() =
|
2017-11-21 19:20:05 +00:00
|
|
|
gdt.descriptor.limit = uint16(sizeof(GDTEntry) * gdt.entries.len) - 1;
|
|
|
|
gdt.descriptor.base = cast[uint32](gdt.entries.addr)
|
2017-11-09 23:21:49 +00:00
|
|
|
# NULL descriptor
|
|
|
|
gdtSetGate(0, 0, 0, 0, 0);
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
# The second entry is our Code Segment
|
|
|
|
gdtSetGate(1, 0'u16, 0xFFFFFFFF'u32, createAccessByte(0, false, true), createFlags(true, true));
|
2017-11-06 21:44:17 +00:00
|
|
|
|
2017-11-09 23:21:49 +00:00
|
|
|
# The third entry is our Data Segment
|
|
|
|
gdtSetGate(2, 0'u16, 0xFFFFFFFF'u32, createAccessByte(0, false, true), createFlags(true, true));
|
|
|
|
serial.write("Flushing GDT.\L")
|
2017-11-21 19:20:05 +00:00
|
|
|
gdtFlush()
|