Made the page table allocation/filling mechanism a bit more flexible and less error prone.
Also, freeing regions is now supported (but still only the last allocated region can be freed, as that's currently good enough). git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@14366 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
a26988784a
commit
0eecdd8334
@ -59,6 +59,7 @@ struct extended_memory {
|
|||||||
|
|
||||||
static const uint32 kDefaultPageTableFlags = 0x07; // present, user, R/W
|
static const uint32 kDefaultPageTableFlags = 0x07; // present, user, R/W
|
||||||
static const size_t kMaxKernelSize = 0x100000; // 1 MB for the kernel
|
static const size_t kMaxKernelSize = 0x100000; // 1 MB for the kernel
|
||||||
|
static const uint32 kPageTableRegionEnd = 0xa0000;
|
||||||
|
|
||||||
// working page directory and page table
|
// working page directory and page table
|
||||||
static uint32 *sPageDirectory = 0;
|
static uint32 *sPageDirectory = 0;
|
||||||
@ -69,6 +70,8 @@ static addr_t sNextPhysicalAddress = 0x100000;
|
|||||||
static addr_t sNextVirtualAddress = KERNEL_BASE + kMaxKernelSize;
|
static addr_t sNextVirtualAddress = KERNEL_BASE + kMaxKernelSize;
|
||||||
static addr_t sMaxVirtualAddress = KERNEL_BASE + 0x400000;
|
static addr_t sMaxVirtualAddress = KERNEL_BASE + 0x400000;
|
||||||
|
|
||||||
|
static addr_t sNextPageTableAddress = 0x90000;
|
||||||
|
|
||||||
|
|
||||||
static addr_t
|
static addr_t
|
||||||
get_next_virtual_address(size_t size)
|
get_next_virtual_address(size_t size)
|
||||||
@ -104,6 +107,18 @@ get_next_physical_page()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static uint32 *
|
||||||
|
get_next_page_table()
|
||||||
|
{
|
||||||
|
addr_t address = sNextPageTableAddress;
|
||||||
|
if (address >= kPageTableRegionEnd)
|
||||||
|
return (uint32 *)get_next_physical_page();
|
||||||
|
|
||||||
|
sNextPageTableAddress += B_PAGE_SIZE;
|
||||||
|
return (uint32 *)address;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/** Adds a new page table for the specified base address and makes
|
/** Adds a new page table for the specified base address and makes
|
||||||
* it current, ie. sets sPageTable to point to the new table.
|
* it current, ie. sets sPageTable to point to the new table.
|
||||||
*/
|
*/
|
||||||
@ -111,8 +126,10 @@ get_next_physical_page()
|
|||||||
static void
|
static void
|
||||||
add_page_table(addr_t base)
|
add_page_table(addr_t base)
|
||||||
{
|
{
|
||||||
|
TRACE(("add_page_table(base = %p)\n", (void *)base));
|
||||||
|
|
||||||
// Get new page table and clear it out
|
// Get new page table and clear it out
|
||||||
sPageTable = (uint32 *)get_next_physical_page();
|
sPageTable = get_next_page_table();
|
||||||
if (sPageTable > (uint32 *)(8 * 1024 * 1024))
|
if (sPageTable > (uint32 *)(8 * 1024 * 1024))
|
||||||
panic("tried to add page table beyond the indentity mapped 8 MB region\n");
|
panic("tried to add page table beyond the indentity mapped 8 MB region\n");
|
||||||
|
|
||||||
@ -134,7 +151,10 @@ unmap_page(addr_t virtualAddress)
|
|||||||
if (virtualAddress < KERNEL_BASE)
|
if (virtualAddress < KERNEL_BASE)
|
||||||
panic("unmap_page: asked to unmap invalid page %p!\n", (void *)virtualAddress);
|
panic("unmap_page: asked to unmap invalid page %p!\n", (void *)virtualAddress);
|
||||||
|
|
||||||
sPageTable[(virtualAddress % (B_PAGE_SIZE * 1024)) / B_PAGE_SIZE] = 0;
|
// unmap the page from the correct page table
|
||||||
|
uint32 *pageTable = (uint32 *)(sPageDirectory[virtualAddress
|
||||||
|
/ (B_PAGE_SIZE * 1024)] & 0xfffff000);
|
||||||
|
pageTable[(virtualAddress % (B_PAGE_SIZE * 1024)) / B_PAGE_SIZE] = 0;
|
||||||
|
|
||||||
asm volatile("invlpg (%0)" : : "r" (virtualAddress));
|
asm volatile("invlpg (%0)" : : "r" (virtualAddress));
|
||||||
}
|
}
|
||||||
@ -166,9 +186,10 @@ map_page(addr_t virtualAddress, addr_t physicalAddress, uint32 flags)
|
|||||||
|
|
||||||
physicalAddress &= ~(B_PAGE_SIZE - 1);
|
physicalAddress &= ~(B_PAGE_SIZE - 1);
|
||||||
|
|
||||||
TRACE(("paddr 0x%lx @ index %ld\n", physicalAddress,
|
// map the page to the correct page table
|
||||||
(virtualAddress % (B_PAGE_SIZE * 1024)) / B_PAGE_SIZE));
|
uint32 *pageTable = (uint32 *)(sPageDirectory[virtualAddress
|
||||||
sPageTable[(virtualAddress % (B_PAGE_SIZE * 1024)) / B_PAGE_SIZE] = physicalAddress | flags;
|
/ (B_PAGE_SIZE * 1024)] & 0xfffff000);
|
||||||
|
pageTable[(virtualAddress % (B_PAGE_SIZE * 1024)) / B_PAGE_SIZE] = physicalAddress | flags;
|
||||||
|
|
||||||
asm volatile("invlpg (%0)" : : "r" (virtualAddress));
|
asm volatile("invlpg (%0)" : : "r" (virtualAddress));
|
||||||
}
|
}
|
||||||
@ -245,23 +266,23 @@ init_page_directory(void)
|
|||||||
// physical and virtual address are the same.
|
// physical and virtual address are the same.
|
||||||
// These page tables won't be taken over into the kernel.
|
// These page tables won't be taken over into the kernel.
|
||||||
|
|
||||||
// make a pagetable at this random spot
|
// make the first page table at the first free spot
|
||||||
uint32 *pgtable = (uint32 *)0x91000;
|
uint32 *pageTable = get_next_page_table();;
|
||||||
|
|
||||||
for (int32 i = 0; i < 1024; i++) {
|
for (int32 i = 0; i < 1024; i++) {
|
||||||
pgtable[i] = (i * 0x1000) | kDefaultPageFlags;
|
pageTable[i] = (i * 0x1000) | kDefaultPageFlags;
|
||||||
}
|
}
|
||||||
|
|
||||||
sPageDirectory[0] = (uint32)pgtable | kDefaultPageFlags;
|
sPageDirectory[0] = (uint32)pageTable | kDefaultPageFlags;
|
||||||
|
|
||||||
// make another pagetable at this random spot
|
// make the second page table
|
||||||
pgtable = (uint32 *)0x92000;
|
pageTable = get_next_page_table();;
|
||||||
|
|
||||||
for (int32 i = 0; i < 1024; i++) {
|
for (int32 i = 0; i < 1024; i++) {
|
||||||
pgtable[i] = (i * 0x1000 + 0x400000) | kDefaultPageFlags;
|
pageTable[i] = (i * 0x1000 + 0x400000) | kDefaultPageFlags;
|
||||||
}
|
}
|
||||||
|
|
||||||
sPageDirectory[1] = (uint32)pgtable | kDefaultPageFlags;
|
sPageDirectory[1] = (uint32)pageTable | kDefaultPageFlags;
|
||||||
|
|
||||||
gKernelArgs.arch_args.num_pgtables = 0;
|
gKernelArgs.arch_args.num_pgtables = 0;
|
||||||
add_page_table(KERNEL_BASE);
|
add_page_table(KERNEL_BASE);
|
||||||
@ -587,8 +608,8 @@ platform_allocate_region(void **_address, size_t size, uint8 protection)
|
|||||||
extern "C" status_t
|
extern "C" status_t
|
||||||
platform_free_region(void *address, size_t size)
|
platform_free_region(void *address, size_t size)
|
||||||
{
|
{
|
||||||
puts(__FUNCTION__);
|
mmu_free(address, size);
|
||||||
return B_ERROR;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user