* Replaced the vm_page_allocate_page*() "pageState" parameter by a more
general "flags" parameter. It encodes the target state of the page -- so that the page isn't unnecessarily put in the wrong page queue first -- a flag whether the page should be cleared, and one to indicate whether the page should be marked busy. * Added page state PAGE_STATE_CACHED. Not used yet. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@35333 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
00d70bc148
commit
e65c400299
@ -49,10 +49,10 @@ void vm_page_unreserve_pages(uint32 count);
|
|||||||
void vm_page_reserve_pages(uint32 count, int priority);
|
void vm_page_reserve_pages(uint32 count, int priority);
|
||||||
bool vm_page_try_reserve_pages(uint32 count, int priority);
|
bool vm_page_try_reserve_pages(uint32 count, int priority);
|
||||||
|
|
||||||
struct vm_page *vm_page_allocate_page(int pageState);
|
struct vm_page *vm_page_allocate_page(uint32 flags);
|
||||||
struct vm_page *vm_page_allocate_page_run(int state, addr_t base,
|
struct vm_page *vm_page_allocate_page_run(uint32 flags, addr_t base,
|
||||||
addr_t length, int priority);
|
addr_t length, int priority);
|
||||||
struct vm_page *vm_page_allocate_page_run_no_base(int state, addr_t count,
|
struct vm_page *vm_page_allocate_page_run_no_base(uint32 flags, addr_t count,
|
||||||
int priority);
|
int priority);
|
||||||
struct vm_page *vm_page_at_index(int32 index);
|
struct vm_page *vm_page_at_index(int32 index);
|
||||||
struct vm_page *vm_lookup_page(addr_t pageNumber);
|
struct vm_page *vm_lookup_page(addr_t pageNumber);
|
||||||
|
@ -128,13 +128,23 @@ enum {
|
|||||||
PAGE_STATE_ACTIVE = 0,
|
PAGE_STATE_ACTIVE = 0,
|
||||||
PAGE_STATE_INACTIVE,
|
PAGE_STATE_INACTIVE,
|
||||||
PAGE_STATE_MODIFIED,
|
PAGE_STATE_MODIFIED,
|
||||||
|
PAGE_STATE_CACHED,
|
||||||
PAGE_STATE_FREE,
|
PAGE_STATE_FREE,
|
||||||
PAGE_STATE_CLEAR,
|
PAGE_STATE_CLEAR,
|
||||||
PAGE_STATE_WIRED,
|
PAGE_STATE_WIRED,
|
||||||
PAGE_STATE_UNUSED
|
PAGE_STATE_UNUSED,
|
||||||
|
|
||||||
|
PAGE_STATE_COUNT,
|
||||||
|
|
||||||
|
PAGE_STATE_FIRST_UNQUEUED = PAGE_STATE_WIRED
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#define VM_PAGE_ALLOC_STATE 0x00000007
|
||||||
|
#define VM_PAGE_ALLOC_CLEAR 0x00000010
|
||||||
|
#define VM_PAGE_ALLOC_BUSY 0x00000020
|
||||||
|
|
||||||
|
|
||||||
#if DEBUG_PAGE_ACCESS
|
#if DEBUG_PAGE_ACCESS
|
||||||
# include <thread.h>
|
# include <thread.h>
|
||||||
|
|
||||||
|
@ -539,9 +539,9 @@ Aperture::AllocateMemory(aperture_memory *memory, uint32 flags)
|
|||||||
uint32 count = size / B_PAGE_SIZE;
|
uint32 count = size / B_PAGE_SIZE;
|
||||||
|
|
||||||
if ((flags & B_APERTURE_NEED_PHYSICAL) != 0) {
|
if ((flags & B_APERTURE_NEED_PHYSICAL) != 0) {
|
||||||
memory->page = vm_page_allocate_page_run(PAGE_STATE_CLEAR, 0, count,
|
memory->page = vm_page_allocate_page_run(
|
||||||
|
PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR, 0, count,
|
||||||
VM_PRIORITY_SYSTEM);
|
VM_PRIORITY_SYSTEM);
|
||||||
// TODO: Mark pages unbusy!
|
|
||||||
if (memory->page == NULL)
|
if (memory->page == NULL)
|
||||||
return B_NO_MEMORY;
|
return B_NO_MEMORY;
|
||||||
} else {
|
} else {
|
||||||
@ -552,8 +552,8 @@ Aperture::AllocateMemory(aperture_memory *memory, uint32 flags)
|
|||||||
|
|
||||||
vm_page_reserve_pages(count, VM_PRIORITY_SYSTEM);
|
vm_page_reserve_pages(count, VM_PRIORITY_SYSTEM);
|
||||||
for (uint32 i = 0; i < count; i++) {
|
for (uint32 i = 0; i < count; i++) {
|
||||||
memory->pages[i] = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
memory->pages[i] = vm_page_allocate_page(
|
||||||
memory->pages[i]->busy = false;
|
PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
|
||||||
}
|
}
|
||||||
vm_page_unreserve_pages(count);
|
vm_page_unreserve_pages(count);
|
||||||
}
|
}
|
||||||
|
@ -543,11 +543,7 @@ map_tmap(vm_translation_map *map, addr_t va, addr_t pa, uint32 attributes)
|
|||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
// we need to allocate a pgtable
|
// we need to allocate a pgtable
|
||||||
page = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
page = vm_page_allocate_page(PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
|
||||||
|
|
||||||
// mark the page WIRED
|
|
||||||
vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
DEBUG_PAGE_ACCESS_END(page);
|
DEBUG_PAGE_ACCESS_END(page);
|
||||||
|
|
||||||
@ -592,11 +588,7 @@ map_tmap(vm_translation_map *map, addr_t va, addr_t pa, uint32 attributes)
|
|||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
// we need to allocate a pgtable
|
// we need to allocate a pgtable
|
||||||
page = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
page = vm_page_allocate_page(PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
|
||||||
|
|
||||||
// mark the page WIRED
|
|
||||||
vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
DEBUG_PAGE_ACCESS_END(page);
|
DEBUG_PAGE_ACCESS_END(page);
|
||||||
|
|
||||||
|
@ -422,11 +422,7 @@ X86VMTranslationMap::Map(addr_t va, addr_t pa, uint32 attributes)
|
|||||||
vm_page *page;
|
vm_page *page;
|
||||||
|
|
||||||
// we need to allocate a pgtable
|
// we need to allocate a pgtable
|
||||||
page = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
page = vm_page_allocate_page(PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
|
||||||
|
|
||||||
// mark the page WIRED
|
|
||||||
vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
DEBUG_PAGE_ACCESS_END(page);
|
DEBUG_PAGE_ACCESS_END(page);
|
||||||
|
|
||||||
|
10
src/system/kernel/cache/file_cache.cpp
vendored
10
src/system/kernel/cache/file_cache.cpp
vendored
@ -157,7 +157,8 @@ PrecacheIO::Prepare()
|
|||||||
// allocate pages for the cache and mark them busy
|
// allocate pages for the cache and mark them busy
|
||||||
uint32 i = 0;
|
uint32 i = 0;
|
||||||
for (size_t pos = 0; pos < fSize; pos += B_PAGE_SIZE) {
|
for (size_t pos = 0; pos < fSize; pos += B_PAGE_SIZE) {
|
||||||
vm_page* page = vm_page_allocate_page(PAGE_STATE_FREE);
|
vm_page* page = vm_page_allocate_page(
|
||||||
|
PAGE_STATE_ACTIVE | VM_PAGE_ALLOC_BUSY);
|
||||||
|
|
||||||
fCache->InsertPage(page, fOffset + pos);
|
fCache->InsertPage(page, fOffset + pos);
|
||||||
|
|
||||||
@ -385,7 +386,7 @@ read_into_cache(file_cache_ref* ref, void* cookie, off_t offset,
|
|||||||
// allocate pages for the cache and mark them busy
|
// allocate pages for the cache and mark them busy
|
||||||
for (size_t pos = 0; pos < numBytes; pos += B_PAGE_SIZE) {
|
for (size_t pos = 0; pos < numBytes; pos += B_PAGE_SIZE) {
|
||||||
vm_page* page = pages[pageIndex++] = vm_page_allocate_page(
|
vm_page* page = pages[pageIndex++] = vm_page_allocate_page(
|
||||||
PAGE_STATE_FREE);
|
PAGE_STATE_ACTIVE | VM_PAGE_ALLOC_BUSY);
|
||||||
|
|
||||||
cache->InsertPage(page, offset + pos);
|
cache->InsertPage(page, offset + pos);
|
||||||
|
|
||||||
@ -508,7 +509,7 @@ write_to_cache(file_cache_ref* ref, void* cookie, off_t offset,
|
|||||||
// TODO: the pages we allocate here should have been reserved upfront
|
// TODO: the pages we allocate here should have been reserved upfront
|
||||||
// in cache_io()
|
// in cache_io()
|
||||||
vm_page* page = pages[pageIndex++] = vm_page_allocate_page(
|
vm_page* page = pages[pageIndex++] = vm_page_allocate_page(
|
||||||
PAGE_STATE_FREE);
|
PAGE_STATE_ACTIVE | VM_PAGE_ALLOC_BUSY);
|
||||||
|
|
||||||
ref->cache->InsertPage(page, offset + pos);
|
ref->cache->InsertPage(page, offset + pos);
|
||||||
|
|
||||||
@ -1074,7 +1075,8 @@ file_cache_init(void)
|
|||||||
{
|
{
|
||||||
// allocate a clean page we can use for writing zeroes
|
// allocate a clean page we can use for writing zeroes
|
||||||
vm_page_reserve_pages(1, VM_PRIORITY_SYSTEM);
|
vm_page_reserve_pages(1, VM_PRIORITY_SYSTEM);
|
||||||
vm_page* page = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
vm_page* page = vm_page_allocate_page(
|
||||||
|
PAGE_STATE_WIRED | VM_PAGE_ALLOC_CLEAR);
|
||||||
vm_page_unreserve_pages(1);
|
vm_page_unreserve_pages(1);
|
||||||
|
|
||||||
sZeroPage = (addr_t)page->physical_page_number * B_PAGE_SIZE;
|
sZeroPage = (addr_t)page->physical_page_number * B_PAGE_SIZE;
|
||||||
|
@ -1349,10 +1349,8 @@ MemoryManager::_MapChunk(VMArea* vmArea, addr_t address, size_t size,
|
|||||||
addr_t endAreaOffset = areaOffset + size;
|
addr_t endAreaOffset = areaOffset + size;
|
||||||
for (size_t offset = areaOffset; offset < endAreaOffset;
|
for (size_t offset = areaOffset; offset < endAreaOffset;
|
||||||
offset += B_PAGE_SIZE) {
|
offset += B_PAGE_SIZE) {
|
||||||
vm_page* page = vm_page_allocate_page(PAGE_STATE_FREE);
|
vm_page* page = vm_page_allocate_page(PAGE_STATE_WIRED);
|
||||||
cache->InsertPage(page, offset);
|
cache->InsertPage(page, offset);
|
||||||
vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
page->wired_count++;
|
page->wired_count++;
|
||||||
atomic_add(&gMappedPagesCount, 1);
|
atomic_add(&gMappedPagesCount, 1);
|
||||||
|
@ -920,8 +920,8 @@ vm_create_anonymous_area(team_id team, const char* name, void** address,
|
|||||||
bool isStack = (protection & B_STACK_AREA) != 0;
|
bool isStack = (protection & B_STACK_AREA) != 0;
|
||||||
page_num_t guardPages;
|
page_num_t guardPages;
|
||||||
bool canOvercommit = false;
|
bool canOvercommit = false;
|
||||||
uint32 newPageState = (flags & CREATE_AREA_DONT_CLEAR) != 0
|
uint32 pageAllocFlags = (flags & CREATE_AREA_DONT_CLEAR) == 0
|
||||||
? PAGE_STATE_FREE : PAGE_STATE_CLEAR;
|
? VM_PAGE_ALLOC_CLEAR : 0;
|
||||||
|
|
||||||
TRACE(("create_anonymous_area [%ld] %s: size 0x%lx\n", team, name, size));
|
TRACE(("create_anonymous_area [%ld] %s: size 0x%lx\n", team, name, size));
|
||||||
|
|
||||||
@ -1049,8 +1049,8 @@ vm_create_anonymous_area(team_id team, const char* name, void** address,
|
|||||||
if (wiring == B_CONTIGUOUS) {
|
if (wiring == B_CONTIGUOUS) {
|
||||||
// we try to allocate the page run here upfront as this may easily
|
// we try to allocate the page run here upfront as this may easily
|
||||||
// fail for obvious reasons
|
// fail for obvious reasons
|
||||||
page = vm_page_allocate_page_run(newPageState, physicalAddress,
|
page = vm_page_allocate_page_run(PAGE_STATE_WIRED | pageAllocFlags,
|
||||||
size / B_PAGE_SIZE, priority);
|
physicalAddress, size / B_PAGE_SIZE, priority);
|
||||||
if (page == NULL) {
|
if (page == NULL) {
|
||||||
status = B_NO_MEMORY;
|
status = B_NO_MEMORY;
|
||||||
goto err0;
|
goto err0;
|
||||||
@ -1122,10 +1122,9 @@ vm_create_anonymous_area(team_id team, const char* name, void** address,
|
|||||||
# endif
|
# endif
|
||||||
continue;
|
continue;
|
||||||
#endif
|
#endif
|
||||||
vm_page* page = vm_page_allocate_page(newPageState);
|
vm_page* page = vm_page_allocate_page(
|
||||||
cache->InsertPage(page, offset);
|
PAGE_STATE_ACTIVE | pageAllocFlags);
|
||||||
map_page(area, page, address, protection);
|
// PAGE_STATE_WIRED | pageAllocFlags);
|
||||||
// vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
// TODO: The pages should be PAGE_STATE_WIRED, since there's
|
// TODO: The pages should be PAGE_STATE_WIRED, since there's
|
||||||
// no need for the page daemon to play with them (the same
|
// no need for the page daemon to play with them (the same
|
||||||
// should be considered in vm_soft_fault()). ATM doing that
|
// should be considered in vm_soft_fault()). ATM doing that
|
||||||
@ -1134,7 +1133,8 @@ vm_create_anonymous_area(team_id team, const char* name, void** address,
|
|||||||
// will age pages way too fast (since it just skips
|
// will age pages way too fast (since it just skips
|
||||||
// PAGE_STATE_WIRED pages, while it processes
|
// PAGE_STATE_WIRED pages, while it processes
|
||||||
// PAGE_STATE_ACTIVE with wired_count > 0).
|
// PAGE_STATE_ACTIVE with wired_count > 0).
|
||||||
page->busy = false;
|
cache->InsertPage(page, offset);
|
||||||
|
map_page(area, page, address, protection);
|
||||||
|
|
||||||
DEBUG_PAGE_ACCESS_END(page);
|
DEBUG_PAGE_ACCESS_END(page);
|
||||||
|
|
||||||
@ -1217,8 +1217,6 @@ vm_create_anonymous_area(team_id team, const char* name, void** address,
|
|||||||
|
|
||||||
increment_page_wired_count(page);
|
increment_page_wired_count(page);
|
||||||
cache->InsertPage(page, offset);
|
cache->InsertPage(page, offset);
|
||||||
vm_page_set_state(page, PAGE_STATE_WIRED);
|
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
DEBUG_PAGE_ACCESS_END(page);
|
DEBUG_PAGE_ACCESS_END(page);
|
||||||
}
|
}
|
||||||
@ -3720,7 +3718,8 @@ fault_get_page(PageFaultContext& context)
|
|||||||
// see if the backing store has it
|
// see if the backing store has it
|
||||||
if (cache->HasPage(context.cacheOffset)) {
|
if (cache->HasPage(context.cacheOffset)) {
|
||||||
// insert a fresh page and mark it busy -- we're going to read it in
|
// insert a fresh page and mark it busy -- we're going to read it in
|
||||||
page = vm_page_allocate_page(PAGE_STATE_FREE);
|
page = vm_page_allocate_page(
|
||||||
|
PAGE_STATE_ACTIVE | VM_PAGE_ALLOC_BUSY);
|
||||||
cache->InsertPage(page, context.cacheOffset);
|
cache->InsertPage(page, context.cacheOffset);
|
||||||
|
|
||||||
// We need to unlock all caches and the address space while reading
|
// We need to unlock all caches and the address space while reading
|
||||||
@ -3774,8 +3773,7 @@ fault_get_page(PageFaultContext& context)
|
|||||||
cache = context.isWrite ? context.topCache : lastCache;
|
cache = context.isWrite ? context.topCache : lastCache;
|
||||||
|
|
||||||
// allocate a clean page
|
// allocate a clean page
|
||||||
page = vm_page_allocate_page(PAGE_STATE_CLEAR);
|
page = vm_page_allocate_page(PAGE_STATE_ACTIVE | VM_PAGE_ALLOC_CLEAR);
|
||||||
page->busy = false;
|
|
||||||
FTRACE(("vm_soft_fault: just allocated page 0x%lx\n",
|
FTRACE(("vm_soft_fault: just allocated page 0x%lx\n",
|
||||||
page->physical_page_number));
|
page->physical_page_number));
|
||||||
|
|
||||||
@ -3789,8 +3787,7 @@ fault_get_page(PageFaultContext& context)
|
|||||||
// TODO: If memory is low, it might be a good idea to steal the page
|
// TODO: If memory is low, it might be a good idea to steal the page
|
||||||
// from our source cache -- if possible, that is.
|
// from our source cache -- if possible, that is.
|
||||||
FTRACE(("get new page, copy it, and put it into the topmost cache\n"));
|
FTRACE(("get new page, copy it, and put it into the topmost cache\n"));
|
||||||
page = vm_page_allocate_page(PAGE_STATE_FREE);
|
page = vm_page_allocate_page(PAGE_STATE_ACTIVE);
|
||||||
page->busy = false;
|
|
||||||
|
|
||||||
// To not needlessly kill concurrency we unlock all caches but the top
|
// To not needlessly kill concurrency we unlock all caches but the top
|
||||||
// one while copying the page. Lacking another mechanism to ensure that
|
// one while copying the page. Lacking another mechanism to ensure that
|
||||||
|
@ -73,11 +73,14 @@ static const uint32 kMinimumSystemReserve = VM_PAGE_RESERVE_USER;
|
|||||||
|
|
||||||
int32 gMappedPagesCount;
|
int32 gMappedPagesCount;
|
||||||
|
|
||||||
static VMPageQueue sFreePageQueue;
|
static VMPageQueue sPageQueues[PAGE_STATE_COUNT];
|
||||||
static VMPageQueue sClearPageQueue;
|
|
||||||
static VMPageQueue sModifiedPageQueue;
|
static VMPageQueue& sFreePageQueue = sPageQueues[PAGE_STATE_FREE];
|
||||||
static VMPageQueue sInactivePageQueue;
|
static VMPageQueue& sClearPageQueue = sPageQueues[PAGE_STATE_CLEAR];
|
||||||
static VMPageQueue sActivePageQueue;
|
static VMPageQueue& sModifiedPageQueue = sPageQueues[PAGE_STATE_MODIFIED];
|
||||||
|
static VMPageQueue& sInactivePageQueue = sPageQueues[PAGE_STATE_INACTIVE];
|
||||||
|
static VMPageQueue& sActivePageQueue = sPageQueues[PAGE_STATE_ACTIVE];
|
||||||
|
static VMPageQueue& sCachedPageQueue = sPageQueues[PAGE_STATE_CACHED];
|
||||||
|
|
||||||
static vm_page *sPages;
|
static vm_page *sPages;
|
||||||
static addr_t sPhysicalPageOffset;
|
static addr_t sPhysicalPageOffset;
|
||||||
@ -295,6 +298,7 @@ find_page(int argc, char **argv)
|
|||||||
{ "modified", &sModifiedPageQueue },
|
{ "modified", &sModifiedPageQueue },
|
||||||
{ "active", &sActivePageQueue },
|
{ "active", &sActivePageQueue },
|
||||||
{ "inactive", &sInactivePageQueue },
|
{ "inactive", &sInactivePageQueue },
|
||||||
|
{ "cached", &sCachedPageQueue },
|
||||||
{ NULL, NULL }
|
{ NULL, NULL }
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -336,6 +340,8 @@ page_state_to_string(int state)
|
|||||||
return "inactive";
|
return "inactive";
|
||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
return "modified";
|
return "modified";
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
return "cached";
|
||||||
case PAGE_STATE_FREE:
|
case PAGE_STATE_FREE:
|
||||||
return "free";
|
return "free";
|
||||||
case PAGE_STATE_CLEAR:
|
case PAGE_STATE_CLEAR:
|
||||||
@ -494,6 +500,8 @@ dump_page_queue(int argc, char **argv)
|
|||||||
queue = &sActivePageQueue;
|
queue = &sActivePageQueue;
|
||||||
else if (!strcmp(argv[1], "inactive"))
|
else if (!strcmp(argv[1], "inactive"))
|
||||||
queue = &sInactivePageQueue;
|
queue = &sInactivePageQueue;
|
||||||
|
else if (!strcmp(argv[1], "cached"))
|
||||||
|
queue = &sCachedPageQueue;
|
||||||
else {
|
else {
|
||||||
kprintf("page_queue: unknown queue \"%s\".\n", argv[1]);
|
kprintf("page_queue: unknown queue \"%s\".\n", argv[1]);
|
||||||
return 0;
|
return 0;
|
||||||
@ -573,6 +581,8 @@ dump_page_stats(int argc, char **argv)
|
|||||||
counter[PAGE_STATE_ACTIVE], busyCounter[PAGE_STATE_ACTIVE]);
|
counter[PAGE_STATE_ACTIVE], busyCounter[PAGE_STATE_ACTIVE]);
|
||||||
kprintf("inactive: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
kprintf("inactive: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
||||||
counter[PAGE_STATE_INACTIVE], busyCounter[PAGE_STATE_INACTIVE]);
|
counter[PAGE_STATE_INACTIVE], busyCounter[PAGE_STATE_INACTIVE]);
|
||||||
|
kprintf("cached: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
||||||
|
counter[PAGE_STATE_CACHED], busyCounter[PAGE_STATE_CACHED]);
|
||||||
kprintf("unused: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
kprintf("unused: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
||||||
counter[PAGE_STATE_UNUSED], busyCounter[PAGE_STATE_UNUSED]);
|
counter[PAGE_STATE_UNUSED], busyCounter[PAGE_STATE_UNUSED]);
|
||||||
kprintf("wired: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
kprintf("wired: %" B_PRIuSIZE " (busy: %" B_PRIuSIZE ")\n",
|
||||||
@ -598,6 +608,8 @@ dump_page_stats(int argc, char **argv)
|
|||||||
sActivePageQueue.Count());
|
sActivePageQueue.Count());
|
||||||
kprintf("inactive queue: %p, count = %ld\n", &sInactivePageQueue,
|
kprintf("inactive queue: %p, count = %ld\n", &sInactivePageQueue,
|
||||||
sInactivePageQueue.Count());
|
sInactivePageQueue.Count());
|
||||||
|
kprintf("cached queue: %p, count = %ld\n", &sCachedPageQueue,
|
||||||
|
sCachedPageQueue.Count());
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -645,6 +657,9 @@ free_page(vm_page* page, bool clear)
|
|||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
fromQueue = &sModifiedPageQueue;
|
fromQueue = &sModifiedPageQueue;
|
||||||
break;
|
break;
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
fromQueue = &sCachedPageQueue;
|
||||||
|
break;
|
||||||
case PAGE_STATE_FREE:
|
case PAGE_STATE_FREE:
|
||||||
case PAGE_STATE_CLEAR:
|
case PAGE_STATE_CLEAR:
|
||||||
panic("free_page(): page %p already free", page);
|
panic("free_page(): page %p already free", page);
|
||||||
@ -717,6 +732,9 @@ set_page_state(vm_page *page, int pageState)
|
|||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
fromQueue = &sModifiedPageQueue;
|
fromQueue = &sModifiedPageQueue;
|
||||||
break;
|
break;
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
fromQueue = &sCachedPageQueue;
|
||||||
|
break;
|
||||||
case PAGE_STATE_FREE:
|
case PAGE_STATE_FREE:
|
||||||
case PAGE_STATE_CLEAR:
|
case PAGE_STATE_CLEAR:
|
||||||
panic("set_page_state(): page %p is free/clear", page);
|
panic("set_page_state(): page %p is free/clear", page);
|
||||||
@ -743,6 +761,9 @@ set_page_state(vm_page *page, int pageState)
|
|||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
toQueue = &sModifiedPageQueue;
|
toQueue = &sModifiedPageQueue;
|
||||||
break;
|
break;
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
toQueue = &sCachedPageQueue;
|
||||||
|
break;
|
||||||
case PAGE_STATE_FREE:
|
case PAGE_STATE_FREE:
|
||||||
case PAGE_STATE_CLEAR:
|
case PAGE_STATE_CLEAR:
|
||||||
panic("set_page_state(): target state is free/clear");
|
panic("set_page_state(): target state is free/clear");
|
||||||
@ -809,6 +830,7 @@ move_page_to_active_or_inactive_queue(vm_page *page, bool dequeued)
|
|||||||
state = PAGE_STATE_ACTIVE;
|
state = PAGE_STATE_ACTIVE;
|
||||||
else
|
else
|
||||||
state = PAGE_STATE_INACTIVE;
|
state = PAGE_STATE_INACTIVE;
|
||||||
|
// TODO: Cached queue!
|
||||||
|
|
||||||
if (dequeued) {
|
if (dequeued) {
|
||||||
page->state = state;
|
page->state = state;
|
||||||
@ -928,6 +950,9 @@ remove_page_marker(struct vm_page &marker)
|
|||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
queue = &sModifiedPageQueue;
|
queue = &sModifiedPageQueue;
|
||||||
break;
|
break;
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
queue = &sCachedPageQueue;
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return;
|
return;
|
||||||
@ -1919,6 +1944,7 @@ vm_page_init(kernel_args *args)
|
|||||||
sModifiedPageQueue.Init("modified pages queue");
|
sModifiedPageQueue.Init("modified pages queue");
|
||||||
sInactivePageQueue.Init("inactive pages queue");
|
sInactivePageQueue.Init("inactive pages queue");
|
||||||
sActivePageQueue.Init("active pages queue");
|
sActivePageQueue.Init("active pages queue");
|
||||||
|
sCachedPageQueue.Init("cached pages queue");
|
||||||
sFreePageQueue.Init("free pages queue");
|
sFreePageQueue.Init("free pages queue");
|
||||||
sClearPageQueue.Init("clear pages queue");
|
sClearPageQueue.Init("clear pages queue");
|
||||||
|
|
||||||
@ -2077,6 +2103,7 @@ vm_mark_page_range_inuse(addr_t startPage, addr_t length)
|
|||||||
case PAGE_STATE_ACTIVE:
|
case PAGE_STATE_ACTIVE:
|
||||||
case PAGE_STATE_INACTIVE:
|
case PAGE_STATE_INACTIVE:
|
||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
case PAGE_STATE_UNUSED:
|
case PAGE_STATE_UNUSED:
|
||||||
default:
|
default:
|
||||||
// uh
|
// uh
|
||||||
@ -2279,22 +2306,21 @@ vm_page_try_reserve_pages(uint32 count, int priority)
|
|||||||
|
|
||||||
|
|
||||||
vm_page *
|
vm_page *
|
||||||
vm_page_allocate_page(int pageState)
|
vm_page_allocate_page(uint32 flags)
|
||||||
{
|
{
|
||||||
|
uint32 pageState = flags & VM_PAGE_ALLOC_STATE;
|
||||||
|
ASSERT(pageState != PAGE_STATE_FREE);
|
||||||
|
ASSERT(pageState != PAGE_STATE_CLEAR);
|
||||||
|
|
||||||
VMPageQueue* queue;
|
VMPageQueue* queue;
|
||||||
VMPageQueue* otherQueue;
|
VMPageQueue* otherQueue;
|
||||||
|
|
||||||
switch (pageState) {
|
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0) {
|
||||||
case PAGE_STATE_FREE:
|
queue = &sClearPageQueue;
|
||||||
queue = &sFreePageQueue;
|
otherQueue = &sFreePageQueue;
|
||||||
otherQueue = &sClearPageQueue;
|
} else {
|
||||||
break;
|
queue = &sFreePageQueue;
|
||||||
case PAGE_STATE_CLEAR:
|
otherQueue = &sClearPageQueue;
|
||||||
queue = &sClearPageQueue;
|
|
||||||
otherQueue = &sFreePageQueue;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return NULL; // invalid
|
|
||||||
}
|
}
|
||||||
|
|
||||||
atomic_add(&sUnreservedFreePages, -1);
|
atomic_add(&sUnreservedFreePages, -1);
|
||||||
@ -2336,19 +2362,20 @@ vm_page_allocate_page(int pageState)
|
|||||||
DEBUG_PAGE_ACCESS_START(page);
|
DEBUG_PAGE_ACCESS_START(page);
|
||||||
|
|
||||||
int oldPageState = page->state;
|
int oldPageState = page->state;
|
||||||
page->state = PAGE_STATE_ACTIVE;
|
page->state = pageState;
|
||||||
page->busy = true;
|
page->busy = (flags & VM_PAGE_ALLOC_BUSY) != 0;
|
||||||
page->usage_count = 2;
|
page->usage_count = 2;
|
||||||
page->accessed = false;
|
page->accessed = false;
|
||||||
page->modified = false;
|
page->modified = false;
|
||||||
|
|
||||||
locker.Unlock();
|
locker.Unlock();
|
||||||
|
|
||||||
sActivePageQueue.AppendUnlocked(page);
|
if (pageState < PAGE_STATE_FIRST_UNQUEUED)
|
||||||
|
sPageQueues[pageState].AppendUnlocked(page);
|
||||||
|
|
||||||
// clear the page, if we had to take it from the free queue and a clear
|
// clear the page, if we had to take it from the free queue and a clear
|
||||||
// page was requested
|
// page was requested
|
||||||
if (pageState == PAGE_STATE_CLEAR && oldPageState != PAGE_STATE_CLEAR)
|
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0 && oldPageState != PAGE_STATE_CLEAR)
|
||||||
clear_page(page);
|
clear_page(page);
|
||||||
|
|
||||||
return page;
|
return page;
|
||||||
@ -2356,9 +2383,13 @@ vm_page_allocate_page(int pageState)
|
|||||||
|
|
||||||
|
|
||||||
static vm_page*
|
static vm_page*
|
||||||
allocate_page_run(page_num_t start, page_num_t length, int pageState,
|
allocate_page_run(page_num_t start, page_num_t length, uint32 flags,
|
||||||
WriteLocker& freeClearQueueLocker)
|
WriteLocker& freeClearQueueLocker)
|
||||||
{
|
{
|
||||||
|
uint32 pageState = flags & VM_PAGE_ALLOC_STATE;
|
||||||
|
ASSERT(pageState != PAGE_STATE_FREE);
|
||||||
|
ASSERT(pageState != PAGE_STATE_CLEAR);
|
||||||
|
|
||||||
T(AllocatePageRun(length));
|
T(AllocatePageRun(length));
|
||||||
|
|
||||||
// pull the pages out of the appropriate queues
|
// pull the pages out of the appropriate queues
|
||||||
@ -2375,8 +2406,8 @@ allocate_page_run(page_num_t start, page_num_t length, int pageState,
|
|||||||
freePages.Add(&page);
|
freePages.Add(&page);
|
||||||
}
|
}
|
||||||
|
|
||||||
page.state = PAGE_STATE_ACTIVE;
|
page.state = flags & VM_PAGE_ALLOC_STATE;
|
||||||
page.busy = true;
|
page.busy = flags & VM_PAGE_ALLOC_BUSY;
|
||||||
page.usage_count = 1;
|
page.usage_count = 1;
|
||||||
page.accessed = false;
|
page.accessed = false;
|
||||||
page.modified = false;
|
page.modified = false;
|
||||||
@ -2385,16 +2416,18 @@ allocate_page_run(page_num_t start, page_num_t length, int pageState,
|
|||||||
freeClearQueueLocker.Unlock();
|
freeClearQueueLocker.Unlock();
|
||||||
|
|
||||||
// clear pages, if requested
|
// clear pages, if requested
|
||||||
if (pageState == PAGE_STATE_CLEAR) {
|
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0) {
|
||||||
for (VMPageQueue::PageList::Iterator it = freePages.GetIterator();
|
for (VMPageQueue::PageList::Iterator it = freePages.GetIterator();
|
||||||
vm_page* page = it.Next();) {
|
vm_page* page = it.Next();) {
|
||||||
clear_page(page);
|
clear_page(page);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// add pages to active queue
|
// add pages to target queue
|
||||||
freePages.MoveFrom(&clearPages);
|
if (pageState < PAGE_STATE_FIRST_UNQUEUED) {
|
||||||
sActivePageQueue.AppendUnlocked(freePages, length);
|
freePages.MoveFrom(&clearPages);
|
||||||
|
sPageQueues[pageState].AppendUnlocked(freePages, length);
|
||||||
|
}
|
||||||
|
|
||||||
// Note: We don't unreserve the pages since we pulled them out of the
|
// Note: We don't unreserve the pages since we pulled them out of the
|
||||||
// free/clear queues without adjusting sUnreservedFreePages.
|
// free/clear queues without adjusting sUnreservedFreePages.
|
||||||
@ -2404,7 +2437,7 @@ allocate_page_run(page_num_t start, page_num_t length, int pageState,
|
|||||||
|
|
||||||
|
|
||||||
vm_page *
|
vm_page *
|
||||||
vm_page_allocate_page_run(int pageState, addr_t base, addr_t length,
|
vm_page_allocate_page_run(uint32 flags, addr_t base, addr_t length,
|
||||||
int priority)
|
int priority)
|
||||||
{
|
{
|
||||||
uint32 start = base >> PAGE_SHIFT;
|
uint32 start = base >> PAGE_SHIFT;
|
||||||
@ -2435,7 +2468,7 @@ vm_page_allocate_page_run(int pageState, addr_t base, addr_t length,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (foundRun)
|
if (foundRun)
|
||||||
return allocate_page_run(start, length, pageState,
|
return allocate_page_run(start, length, flags,
|
||||||
freeClearQueueLocker);
|
freeClearQueueLocker);
|
||||||
|
|
||||||
start += i;
|
start += i;
|
||||||
@ -2444,21 +2477,16 @@ vm_page_allocate_page_run(int pageState, addr_t base, addr_t length,
|
|||||||
|
|
||||||
|
|
||||||
vm_page *
|
vm_page *
|
||||||
vm_page_allocate_page_run_no_base(int pageState, addr_t count, int priority)
|
vm_page_allocate_page_run_no_base(uint32 flags, addr_t count, int priority)
|
||||||
{
|
{
|
||||||
VMPageQueue* queue;
|
VMPageQueue* queue;
|
||||||
VMPageQueue* otherQueue;
|
VMPageQueue* otherQueue;
|
||||||
switch (pageState) {
|
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0) {
|
||||||
case PAGE_STATE_FREE:
|
queue = &sClearPageQueue;
|
||||||
queue = &sFreePageQueue;
|
otherQueue = &sFreePageQueue;
|
||||||
otherQueue = &sClearPageQueue;
|
} else {
|
||||||
break;
|
queue = &sFreePageQueue;
|
||||||
case PAGE_STATE_CLEAR:
|
otherQueue = &sClearPageQueue;
|
||||||
queue = &sClearPageQueue;
|
|
||||||
otherQueue = &sFreePageQueue;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return NULL; // invalid
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!vm_page_try_reserve_pages(count, priority))
|
if (!vm_page_try_reserve_pages(count, priority))
|
||||||
@ -2485,7 +2513,7 @@ vm_page_allocate_page_run_no_base(int pageState, addr_t count, int priority)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (foundRun) {
|
if (foundRun) {
|
||||||
return allocate_page_run(page - sPages, count, pageState,
|
return allocate_page_run(page - sPages, count, flags,
|
||||||
freeClearQueueLocker);
|
freeClearQueueLocker);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2578,6 +2606,9 @@ vm_page_requeue(struct vm_page *page, bool tail)
|
|||||||
case PAGE_STATE_MODIFIED:
|
case PAGE_STATE_MODIFIED:
|
||||||
queue = &sModifiedPageQueue;
|
queue = &sModifiedPageQueue;
|
||||||
break;
|
break;
|
||||||
|
case PAGE_STATE_CACHED:
|
||||||
|
queue = &sCachedPageQueue;
|
||||||
|
break;
|
||||||
case PAGE_STATE_FREE:
|
case PAGE_STATE_FREE:
|
||||||
case PAGE_STATE_CLEAR:
|
case PAGE_STATE_CLEAR:
|
||||||
panic("vm_page_requeue() called for free/clear page %p", page);
|
panic("vm_page_requeue() called for free/clear page %p", page);
|
||||||
|
Loading…
Reference in New Issue
Block a user