; Copyright 2015 Philipp Oppermann ; ; Licensed under the Apache License, Version 2.0 (the "License"); ; you may not use this file except in compliance with the License. ; You may obtain a copy of the License at ; ; http://www.apache.org/licenses/LICENSE-2.0 ; ; Unless required by applicable law or agreed to in writing, software ; distributed under the License is distributed on an "AS IS" BASIS, ; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ; See the License for the specific language governing permissions and ; limitations under the License. global start section .text bits 32 start: mov esp, stack_top call check_multiboot call check_cpuid call check_long_mode call setup_page_tables call enable_paging ; print `OK` to screen mov dword [0xb8000], 0x2f4b2f4f hlt setup_page_tables: ; map first P4 entry to P3 table mov eax, p3_table or eax, 0b11 ; present + writable mov [p4_table], eax ; map first P3 entry to a huge page that starts at address 0 mov dword [p3_table], 0b10000011 ; present + writable + huge ret enable_paging: ; load P4 to cr3 register (cpu uses this to access the P4 table) mov eax, p4_table mov cr3, eax ; enable PAE-flag in cr4 (Physical Address Extension) mov eax, cr4 or eax, 1 << 5 mov cr4, eax ; set the long mode bit in the EFER MSR (model specific register) mov ecx, 0xC0000080 rdmsr or eax, 1 << 8 wrmsr ; enable paging in the cr0 register mov eax, cr0 or eax, 1 << 31 or eax, 1 << 16 mov cr0, eax ret ; Prints `ERR: ` and the given error code to screen and hangs. ; parameter: error code (in ascii) in al error: mov dword [0xb8000], 0x4f524f45 mov dword [0xb8004], 0x4f3a4f52 mov dword [0xb8008], 0x4f204f20 mov byte [0xb800a], al hlt ; Throw error 0 if eax doesn't contain the Multiboot 2 magic value (0x36d76289). check_multiboot: cmp eax, 0x36d76289 jne .no_multiboot ret .no_multiboot: mov al, "0" jmp error ; Throw error 1 if the CPU doesn't support the CPUID command. check_cpuid: pushfd ; Store the FLAGS-register. pop eax ; Restore the A-register. mov ecx, eax ; Set the C-register to the A-register. xor eax, 1 << 21 ; Flip the ID-bit, which is bit 21. push eax ; Store the A-register. popfd ; Restore the FLAGS-register. pushfd ; Store the FLAGS-register. pop eax ; Restore the A-register. push ecx ; Store the C-register. popfd ; Restore the FLAGS-register. xor eax, ecx ; Do a XOR-operation on the A-register and the C-register. jz .no_cpuid ; The zero flag is set, no CPUID. ret ; CPUID is available for use. .no_cpuid: mov al, "1" jmp error ; Throw error 2 if the CPU doesn't support Long Mode. check_long_mode: mov eax, 0x80000000 ; Set the A-register to 0x80000000. cpuid ; CPU identification. cmp eax, 0x80000001 ; Compare the A-register with 0x80000001. jb .no_long_mode ; It is less, there is no long mode. mov eax, 0x80000000 ; Set the A-register to 0x80000000. cpuid ; CPU identification. cmp eax, 0x80000001 ; Compare the A-register with 0x80000001. jb .no_long_mode ; It is less, there is no long mode. ret .no_long_mode: mov al, "2" jmp error section .bss align 4096 p4_table: resb 4096 p3_table: resb 4096 stack_bottom: resb 64 stack_top: