* The "page_stats" command does now also print the longest contiguous runs of

free respective free and cached pages.
* Removed the unused vm_page_allocate_page_run_no_base().
* vm_page_allocate_page_run() (and allocate_page_run()):
  - Use vm_page_reserve_pages() instead of vm_page_try_reserve_pages(), i.e.
    wait until the reservation succeeds.
  - Now we iterates two times through the pages to find a suitable page run. In
    the first iteration it only looks for free/clear pages, in the second
    iteration it also considers cached pages. This increases the chance of the
    function to succeed, when a lot of caching is going on.
    This reduces the amount of memory required to use the IOCache when booting
    off the anyboot Live CD to around 160 MB in qemu. It also seems to work with
    128 MB, but the syslog indicates that some memory allocations fail, which
    is not exactly inspiring confidence.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@36489 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Ingo Weinhold 2010-04-26 13:56:09 +00:00
parent 01becc3026
commit b1e06d23ce
2 changed files with 218 additions and 89 deletions

View File

@ -60,9 +60,7 @@ bool vm_page_try_reserve_pages(vm_page_reservation* reservation, uint32 count,
struct vm_page *vm_page_allocate_page(vm_page_reservation* reservation,
uint32 flags);
struct vm_page *vm_page_allocate_page_run(uint32 flags, addr_t base,
addr_t length, int priority);
struct vm_page *vm_page_allocate_page_run_no_base(uint32 flags, addr_t count,
int priority);
size_t length, int priority);
struct vm_page *vm_page_at_index(int32 index);
struct vm_page *vm_lookup_page(addr_t pageNumber);
bool vm_page_is_dummy(struct vm_page *page);

View File

@ -822,28 +822,59 @@ dump_page_stats(int argc, char **argv)
{
page_num_t swappableModified = 0;
page_num_t swappableModifiedInactive = 0;
size_t counter[8];
size_t busyCounter[8];
addr_t i;
memset(counter, 0, sizeof(counter));
memset(busyCounter, 0, sizeof(busyCounter));
for (i = 0; i < sNumPages; i++) {
struct page_run {
page_num_t start;
page_num_t end;
page_num_t Length() const { return end - start; }
};
page_run currentFreeRun = { 0, 0 };
page_run currentCachedRun = { 0, 0 };
page_run longestFreeRun = { 0, 0 };
page_run longestCachedRun = { 0, 0 };
for (addr_t i = 0; i < sNumPages; i++) {
if (sPages[i].State() > 7)
panic("page %li at %p has invalid state!\n", i, &sPages[i]);
counter[sPages[i].State()]++;
if (sPages[i].busy)
busyCounter[sPages[i].State()]++;
uint32 pageState = sPages[i].State();
if (sPages[i].State() == PAGE_STATE_MODIFIED
counter[pageState]++;
if (sPages[i].busy)
busyCounter[pageState]++;
if (pageState == PAGE_STATE_MODIFIED
&& sPages[i].Cache() != NULL
&& sPages[i].Cache()->temporary && sPages[i].wired_count == 0) {
swappableModified++;
if (sPages[i].usage_count == 0)
swappableModifiedInactive++;
}
// track free and cached pages runs
if (pageState == PAGE_STATE_FREE || pageState == PAGE_STATE_CLEAR) {
currentFreeRun.end = i + 1;
currentCachedRun.end = i + 1;
} else {
if (currentFreeRun.Length() > longestFreeRun.Length())
longestFreeRun = currentFreeRun;
currentFreeRun.start = currentFreeRun.end = i + 1;
if (pageState == PAGE_STATE_CACHED) {
currentCachedRun.end = i + 1;
} else {
if (currentCachedRun.Length() > longestCachedRun.Length())
longestCachedRun = currentCachedRun;
currentCachedRun.start = currentCachedRun.end = i + 1;
}
}
}
kprintf("page stats:\n");
@ -868,6 +899,12 @@ dump_page_stats(int argc, char **argv)
kprintf("unsatisfied page reservations: %" B_PRId32 "\n",
sUnsatisfiedPageReservations);
kprintf("mapped pages: %lu\n", gMappedPagesCount);
kprintf("longest free pages run: %" B_PRIuSIZE " pages (at %" B_PRIuSIZE
")\n", longestFreeRun.Length(),
sPages[longestFreeRun.start].physical_page_number);
kprintf("longest free/cached pages run: %" B_PRIuSIZE " pages (at %"
B_PRIuSIZE ")\n", longestCachedRun.Length(),
sPages[longestCachedRun.start].physical_page_number);
kprintf("waiting threads:\n");
for (PageReservationWaiterList::Iterator it
@ -2098,6 +2135,9 @@ free_cached_page(vm_page *page, bool dontWait)
// we can now steal this page
cache->RemovePage(page);
// Now the page doesn't have cache anymore, so no one else (e.g.
// vm_page_allocate_page_run() can pick it up), since they would be
// required to lock the cache first, which would fail.
sCachedPageQueue.RemoveUnlocked(page);
return true;
@ -3089,7 +3129,48 @@ vm_page_allocate_page(vm_page_reservation* reservation, uint32 flags)
}
static vm_page*
static void
allocate_page_run_cleanup(VMPageQueue::PageList& freePages,
VMPageQueue::PageList& clearPages)
{
while (vm_page* page = freePages.RemoveHead()) {
page->busy = false;
page->SetState(PAGE_STATE_FREE);
DEBUG_PAGE_ACCESS_END(page);
sFreePageQueue.PrependUnlocked(page);
}
while (vm_page* page = clearPages.RemoveHead()) {
page->busy = false;
page->SetState(PAGE_STATE_CLEAR);
DEBUG_PAGE_ACCESS_END(page);
sClearPageQueue.PrependUnlocked(page);
}
}
/*! Tries to allocate the a contiguous run of \a length pages starting at
index \a start.
The must have write-locked the free/clear page queues. The function will
unlock regardless of whether it succeeds or fails.
If the function fails, it cleans up after itself, i.e. it will free all
pages it managed to allocate.
\param start The start index (into \c sPages) of the run.
\param length The number of pages to allocate.
\param flags Page allocation flags. Encodes the state the function shall
set the allocated pages to, whether the pages shall be marked busy
(VM_PAGE_ALLOC_BUSY), and whether the pages shall be cleared
(VM_PAGE_ALLOC_CLEAR).
\param freeClearQueueLocker Locked WriteLocker for the free/clear page
queues in locked state. Will be unlocked by the function.
\return The index of the first page that could not be allocated. \a length
is returned when the function was successful.
*/
static page_num_t
allocate_page_run(page_num_t start, page_num_t length, uint32 flags,
WriteLocker& freeClearQueueLocker)
{
@ -3099,29 +3180,110 @@ allocate_page_run(page_num_t start, page_num_t length, uint32 flags,
TA(AllocatePageRun(length));
// pull the pages out of the appropriate queues
// Pull the free/clear pages out of their respective queues. Cached pages
// are allocated later.
page_num_t cachedPages = 0;
VMPageQueue::PageList freePages;
VMPageQueue::PageList clearPages;
for (page_num_t i = 0; i < length; i++) {
page_num_t i = 0;
for (; i < length; i++) {
bool pageAllocated = true;
bool noPage = false;
vm_page& page = sPages[start + i];
switch (page.State()) {
case PAGE_STATE_CLEAR:
DEBUG_PAGE_ACCESS_START(&page);
if (page.State() == PAGE_STATE_CLEAR) {
sClearPageQueue.Remove(&page);
clearPages.Add(&page);
} else {
break;
case PAGE_STATE_FREE:
DEBUG_PAGE_ACCESS_START(&page);
sFreePageQueue.Remove(&page);
freePages.Add(&page);
break;
case PAGE_STATE_CACHED:
// We allocate cached pages later.
cachedPages++;
pageAllocated = false;
break;
default:
// Probably a page was cached when our caller checked. Now it's
// gone and we have to abort.
noPage = true;
break;
}
if (noPage)
break;
if (pageAllocated) {
page.SetState(flags & VM_PAGE_ALLOC_STATE);
page.busy = flags & VM_PAGE_ALLOC_BUSY;
page.busy = (flags & VM_PAGE_ALLOC_BUSY) != 0;
page.usage_count = 0;
page.accessed = false;
page.modified = false;
}
}
if (i < length) {
// failed to allocate a page -- free all that we've got
allocate_page_run_cleanup(freePages, clearPages);
return i;
}
freeClearQueueLocker.Unlock();
if (cachedPages > 0) {
// allocate the pages that weren't free but cached
page_num_t freedCachedPages = 0;
page_num_t nextIndex = start;
vm_page* freePage = freePages.Head();
vm_page* clearPage = clearPages.Head();
while (cachedPages > 0) {
// skip, if we've already got the page
if (freePage != NULL && size_t(freePage - sPages) == nextIndex) {
freePage = freePages.GetNext(freePage);
nextIndex++;
continue;
}
if (clearPage != NULL && size_t(clearPage - sPages) == nextIndex) {
clearPage = clearPages.GetNext(clearPage);
nextIndex++;
continue;
}
// free the page, if it is still cached
vm_page& page = sPages[nextIndex];
if (!free_cached_page(&page, false))
break;
page.SetState(flags & VM_PAGE_ALLOC_STATE);
page.busy = (flags & VM_PAGE_ALLOC_BUSY) != 0;
page.usage_count = 0;
page.accessed = false;
page.modified = false;
freePages.InsertBefore(freePage, &page);
freedCachedPages++;
cachedPages--;
nextIndex++;
}
// If we have freed cached pages, we need to balance things.
if (freedCachedPages > 0)
unreserve_pages(freedCachedPages);
if (nextIndex - start < length) {
// failed to allocate all cached pages -- free all that we've got
freeClearQueueLocker.Lock();
allocate_page_run_cleanup(freePages, clearPages);
freeClearQueueLocker.Unlock();
return nextIndex - start;
}
}
// clear pages, if requested
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0) {
for (VMPageQueue::PageList::Iterator it = freePages.GetIterator();
@ -3139,27 +3301,42 @@ allocate_page_run(page_num_t start, page_num_t length, uint32 flags,
// Note: We don't unreserve the pages since we pulled them out of the
// free/clear queues without adjusting sUnreservedFreePages.
return &sPages[start];
return length;
}
vm_page *
vm_page_allocate_page_run(uint32 flags, addr_t base, addr_t length,
vm_page_allocate_page_run(uint32 flags, addr_t base, size_t length,
int priority)
{
uint32 start = base >> PAGE_SHIFT;
vm_page_reservation reservation;
if (!vm_page_try_reserve_pages(&reservation, length, priority))
return NULL;
// TODO: add more tries, ie. free some inactive, ...
// no free space
vm_page_reserve_pages(&reservation, length, priority);
WriteLocker freeClearQueueLocker(sFreePageQueuesLock);
// First we try to get a run with free pages only. If that fails, we also
// consider cached pages. If there are only few free pages and many cached
// ones, the odds are that we won't find enough contiguous ones, so we skip
// the first iteration in this case.
int32 freePages = sUnreservedFreePages;
int useCached = freePages > 0 && (page_num_t)freePages > 2 * length ? 0 : 1;
for (;;) {
bool foundRun = true;
if (start + length > sNumPages) {
if (useCached == 0) {
// The first iteration with free pages only was unsuccessful.
// Try again also considering cached pages.
useCached = 1;
start = base >> PAGE_SHIFT;
continue;
}
dprintf("vm_page_allocate_page_run(): Failed to allocate run of "
"length %" B_PRIuSIZE " in second iteration!", length);
freeClearQueueLocker.Unlock();
vm_page_unreserve_pages(&reservation);
return NULL;
@ -3167,72 +3344,26 @@ vm_page_allocate_page_run(uint32 flags, addr_t base, addr_t length,
uint32 i;
for (i = 0; i < length; i++) {
if (sPages[start + i].State() != PAGE_STATE_FREE
&& sPages[start + i].State() != PAGE_STATE_CLEAR) {
foundRun = false;
i++;
break;
}
}
if (foundRun)
return allocate_page_run(start, length, flags,
freeClearQueueLocker);
start += i;
}
}
vm_page *
vm_page_allocate_page_run_no_base(uint32 flags, addr_t count, int priority)
{
VMPageQueue* queue;
VMPageQueue* otherQueue;
if ((flags & VM_PAGE_ALLOC_CLEAR) != 0) {
queue = &sClearPageQueue;
otherQueue = &sFreePageQueue;
} else {
queue = &sFreePageQueue;
otherQueue = &sClearPageQueue;
}
vm_page_reservation reservation;
if (!vm_page_try_reserve_pages(&reservation, count, priority))
return NULL;
// TODO: add more tries, ie. free some inactive, ...
// no free space
WriteLocker freeClearQueueLocker(sFreePageQueuesLock);
for (;;) {
VMPageQueue::Iterator it = queue->GetIterator();
while (vm_page *page = it.Next()) {
vm_page *current = page;
if (current >= &sPages[sNumPages - count])
continue;
bool foundRun = true;
for (uint32 i = 0; i < count; i++, current++) {
if (current->State() != PAGE_STATE_FREE
&& current->State() != PAGE_STATE_CLEAR) {
uint32 pageState = sPages[start + i].State();
if (pageState != PAGE_STATE_FREE
&& pageState != PAGE_STATE_CLEAR
&& (pageState != PAGE_STATE_CACHED || useCached == 0)) {
foundRun = false;
break;
}
}
if (foundRun) {
return allocate_page_run(page - sPages, count, flags,
freeClearQueueLocker);
}
i = allocate_page_run(start, length, flags, freeClearQueueLocker);
if (i == length)
return &sPages[start];
// apparently a cached page couldn't be allocated -- skip it and
// continue
freeClearQueueLocker.Lock();
}
if (queue == otherQueue) {
freeClearQueueLocker.Unlock();
vm_page_unreserve_pages(&reservation);
}
queue = otherQueue;
start += i + 1;
}
}