forked from Bananymous/banan-os
				
			Kernel: Paging uses now PAE with 2 MiB pages
This commit is contained in:
		
							parent
							
								
									a12ddfa12a
								
							
						
					
					
						commit
						d6623def32
					
				|  | @ -211,7 +211,7 @@ namespace APIC | ||||||
| 	static void ParseMADT(RSDPDescriptor* rsdp) | 	static void ParseMADT(RSDPDescriptor* rsdp) | ||||||
| 	{ | 	{ | ||||||
| 		RSDT* root = (RSDT*)(rsdp->revision == 2 ? ((RSDPDescriptor20*)rsdp)->xsdt_address : rsdp->rsdt_address); | 		RSDT* root = (RSDT*)(rsdp->revision == 2 ? ((RSDPDescriptor20*)rsdp)->xsdt_address : rsdp->rsdt_address); | ||||||
| 		Paging::MapRSDP((uint32_t)root & 0xFFC00000); | 		Paging::MapPage((uint32_t)root); | ||||||
| 		uint32_t sdt_entry_count = (root->header.length - sizeof(root->header)) / (rsdp->revision == 2 ? 8 : 4); | 		uint32_t sdt_entry_count = (root->header.length - sizeof(root->header)) / (rsdp->revision == 2 ? 8 : 4); | ||||||
| 
 | 
 | ||||||
| 		for (uint32_t i = 0; i < sdt_entry_count; i++) | 		for (uint32_t i = 0; i < sdt_entry_count; i++) | ||||||
|  | @ -372,10 +372,8 @@ namespace APIC | ||||||
| 		if (s_local_apic == 0 || s_io_apic == 0) | 		if (s_local_apic == 0 || s_io_apic == 0) | ||||||
| 			return false; | 			return false; | ||||||
| 
 | 
 | ||||||
| 		if ((s_io_apic & 0xFFC00000) != (s_local_apic & 0xFFC00000)) | 		Paging::MapPage(s_io_apic); | ||||||
| 			Kernel::panic("lapic and ioapic are not in the same 4 MiB are"); | 		Paging::MapPage(s_local_apic); | ||||||
| 		 |  | ||||||
| 		Paging::MapAPIC(s_io_apic & 0xFFC00000); |  | ||||||
| 
 | 
 | ||||||
| 		// Enable Local APIC
 | 		// Enable Local APIC
 | ||||||
| 		SetMSR(IA32_APIC_BASE, (s_local_apic & 0xFFFFF000) | IA32_APIC_BASE_ENABLE, 0); | 		SetMSR(IA32_APIC_BASE, (s_local_apic & 0xFFFFF000) | IA32_APIC_BASE_ENABLE, 0); | ||||||
|  |  | ||||||
|  | @ -1,66 +1,88 @@ | ||||||
|  | #include <kernel/CPUID.h> | ||||||
| #include <kernel/Paging.h> | #include <kernel/Paging.h> | ||||||
| #include <kernel/panic.h> | #include <kernel/panic.h> | ||||||
| 
 | 
 | ||||||
|  | #include <string.h> | ||||||
|  | 
 | ||||||
|  | #define PRESENT		(1 << 0) | ||||||
|  | #define READ_WRITE	(1 << 1) | ||||||
|  | #define PAGE_SIZE	(1 << 7) | ||||||
|  | 
 | ||||||
| namespace Paging | namespace Paging | ||||||
| { | { | ||||||
| 
 | 
 | ||||||
| 	static uint32_t s_page_directory[1024] __attribute__((aligned(4096))); | 	static uint64_t s_page_directory_pointer_table[4]	__attribute__((aligned(0x20))); | ||||||
| 	static uint32_t s_page_table[1024] __attribute__((aligned(4096))); | 	static uint64_t s_page_directory[4 * 512]			__attribute__((aligned(4096))); | ||||||
| 	static uint32_t s_page_table_framebuffer[1024] __attribute__((aligned(4096))); | 
 | ||||||
| 	static uint32_t s_page_table_rsdp[1024] __attribute__((aligned(4096))); | 	static bool HasRequirements() | ||||||
| 	static uint32_t s_page_table_apic[1024] __attribute__((aligned(4096))); | 	{ | ||||||
|  | 		uint32_t ecx, edx; | ||||||
|  | 		CPUID::GetFeatures(ecx, edx); | ||||||
|  | 		if (!(edx & CPUID::Features::EDX_PAE)) | ||||||
|  | 		{ | ||||||
|  | 			derrorln("PAE not supported, halting"); | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	void Initialize() | 	void Initialize() | ||||||
| 	{ | 	{ | ||||||
| 		for (uint32_t i = 0; i < 1024; i++) | 		if (!HasRequirements()) | ||||||
| 			s_page_directory[i] = 0x00000002; | 			asm volatile("hlt"); | ||||||
| 
 | 
 | ||||||
| 		for (uint32_t i = 0; i < 1024; i++) | 		// Identity map first 2 (2 MiB) pages
 | ||||||
| 			s_page_table[i] = (i << 12) | 0x03; | 		memset(s_page_directory, 0x00, sizeof(s_page_directory)); | ||||||
| 		s_page_directory[0] = (uint32_t)s_page_table | 0x03; | 		s_page_directory[0] = (0x00 << 21) | PAGE_SIZE | READ_WRITE | PRESENT; | ||||||
|  | 		s_page_directory[1] = (0x01 << 21) | PAGE_SIZE | READ_WRITE | PRESENT; | ||||||
|  | 
 | ||||||
|  | 		// Initialize PDPTEs
 | ||||||
|  | 		for (int i = 0; i < 4; i++) | ||||||
|  | 			s_page_directory_pointer_table[i] = (uint64_t)(&s_page_directory[512 * i]) | PRESENT; | ||||||
| 
 | 
 | ||||||
| 		asm volatile( | 		asm volatile( | ||||||
| 			"movl %%eax, %%cr3;" | 			// Enable PAE
 | ||||||
|  | 			"movl %%cr4, %%eax;" | ||||||
|  | 			"orl $0x20, %%eax;" | ||||||
|  | 			"movl %%eax, %%cr4;" | ||||||
|  | 
 | ||||||
|  | 			// Load PDPT address to cr3
 | ||||||
|  | 			"movl %0, %%cr3;" | ||||||
|  | 			 | ||||||
|  | 			// Enable paging
 | ||||||
| 			"movl %%cr0, %%eax;" | 			"movl %%cr0, %%eax;" | ||||||
| 			"orl $0x80000001, %%eax;" | 			"orl $0x80000000, %%eax;" | ||||||
| 			"movl %%eax, %%cr0;" | 			"movl %%eax, %%cr0;" | ||||||
| 			:: "a"(s_page_directory) | 
 | ||||||
|  | 			:: "r" (s_page_directory_pointer_table) | ||||||
|  | 			: "eax" | ||||||
| 		); | 		); | ||||||
|  | 
 | ||||||
|  | 		dprintln("Paging enabled"); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	static void MapPDE(uint32_t address, uint32_t* pt) | 	void MapPage(uintptr_t address) | ||||||
| 	{ | 	{ | ||||||
| 		if ((address & 0xffc00000) != address) | 		if (address != ((address >> 21) << 21)) | ||||||
| 			Kernel::panic("Trying to map non 4 MiB aligned address"); |  | ||||||
| 
 |  | ||||||
| 		uint32_t pd_index = address >> 22; |  | ||||||
| 
 |  | ||||||
| 		if (!(s_page_directory[pd_index] & (1 << 0))) |  | ||||||
| 		{ | 		{ | ||||||
| 			// Identity map the whole page table
 | 			dprintln("aligning 0x{8H} to 2 MiB boundary -> 0x{8H}", address, (address >> 21) << 21); | ||||||
| 			for (uint32_t i = 0; i < 1024; i++) | 			address = (address >> 21) << 21; | ||||||
| 				pt[i] = address | (i << 12) | 0x03; |  | ||||||
| 			// Set the pde to point to page table
 |  | ||||||
| 			s_page_directory[pd_index] = (uint32_t)pt | 0x03; |  | ||||||
| 			// Flush TLB
 |  | ||||||
| 			for (uint32_t i = 0; i < 1024; i++) |  | ||||||
| 				asm volatile("invlpg (%0)" :: "r" (address | (i << 12)) : "memory"); |  | ||||||
| 		} | 		} | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	void MapFramebuffer(uint32_t address) | 		uint32_t pde = address >> 21; | ||||||
| 	{ |  | ||||||
| 		MapPDE(address, s_page_table_framebuffer); |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	void MapRSDP(uint32_t address) | 		dprintln("mapping pde {} (0x{8H} - 0x{8H})", pde, pde << 21, ((pde + 1) << 21) - 1); | ||||||
| 	{ |  | ||||||
| 		MapPDE(address, s_page_table_rsdp); |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	void MapAPIC(uint32_t address) | 		if (s_page_directory[pde] & PRESENT) | ||||||
| 	{ | 		{ | ||||||
| 		MapPDE(address, s_page_table_apic); | 			dprintln("page already mapped"); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		// Map and flush the given address
 | ||||||
|  | 		s_page_directory[pde] = address | PAGE_SIZE | READ_WRITE | PRESENT; | ||||||
|  | 		asm volatile("invlpg (%0)" :: "r"(address) : "memory"); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | @ -95,7 +95,7 @@ namespace VESA | ||||||
| 		s_height	= framebuffer.height; | 		s_height	= framebuffer.height; | ||||||
| 		s_mode		= framebuffer.type; | 		s_mode		= framebuffer.type; | ||||||
| 
 | 
 | ||||||
| 		Paging::MapFramebuffer(framebuffer.addr & 0xFFC00000); | 		Paging::MapPage(framebuffer.addr); | ||||||
| 
 | 
 | ||||||
| 		if (s_mode == MULTIBOOT_FRAMEBUFFER_TYPE_GRAPHICS) | 		if (s_mode == MULTIBOOT_FRAMEBUFFER_TYPE_GRAPHICS) | ||||||
| 		{ | 		{ | ||||||
|  |  | ||||||
|  | @ -7,8 +7,6 @@ namespace Paging | ||||||
| 
 | 
 | ||||||
| 	void Initialize(); | 	void Initialize(); | ||||||
| 
 | 
 | ||||||
| 	void MapFramebuffer(uint32_t address); | 	void MapPage(uintptr_t address); | ||||||
| 	void MapRSDP(uint32_t address); |  | ||||||
| 	void MapAPIC(uint32_t address); |  | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
		Loading…
	
		Reference in New Issue