2007-09-03 19:41:14 +04:00
|
|
|
/*
|
|
|
|
* Copyright 2007, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
|
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
#include <OS.h>
|
|
|
|
|
|
|
|
#include <vm.h>
|
|
|
|
#include <vm_priv.h>
|
|
|
|
#include <vm_cache.h>
|
|
|
|
#include <vm_low_memory.h>
|
|
|
|
#include <vm_page.h>
|
|
|
|
|
|
|
|
|
|
|
|
static sem_id sPageDaemonSem;
|
|
|
|
static uint32 sScanPagesCount;
|
|
|
|
static uint32 sNumPages;
|
|
|
|
static bigtime_t sScanWaitInterval;
|
|
|
|
|
|
|
|
|
|
|
|
class PageCacheLocker {
|
|
|
|
public:
|
|
|
|
PageCacheLocker(vm_page* page);
|
|
|
|
~PageCacheLocker();
|
|
|
|
|
|
|
|
bool IsLocked() { return fPage != NULL; }
|
|
|
|
|
|
|
|
bool Lock(vm_page* page);
|
|
|
|
void Unlock();
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool _IgnorePage(vm_page* page);
|
|
|
|
|
|
|
|
vm_page* fPage;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
PageCacheLocker::PageCacheLocker(vm_page* page)
|
|
|
|
:
|
|
|
|
fPage(NULL)
|
|
|
|
{
|
|
|
|
Lock(page);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
PageCacheLocker::~PageCacheLocker()
|
|
|
|
{
|
|
|
|
Unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
PageCacheLocker::_IgnorePage(vm_page* page)
|
|
|
|
{
|
|
|
|
if (page->state == PAGE_STATE_WIRED || page->state == PAGE_STATE_BUSY
|
|
|
|
|| page->state == PAGE_STATE_FREE || page->state == PAGE_STATE_CLEAR
|
|
|
|
|| page->state == PAGE_STATE_UNUSED || page->cache == NULL)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
PageCacheLocker::Lock(vm_page* page)
|
|
|
|
{
|
|
|
|
if (_IgnorePage(page))
|
|
|
|
return false;
|
|
|
|
|
2007-09-09 18:36:10 +04:00
|
|
|
// Grab a reference to this cache.
|
|
|
|
vm_cache* cache = vm_cache_acquire_page_cache_ref(page);
|
2007-09-03 19:41:14 +04:00
|
|
|
if (cache == NULL)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
mutex_lock(&cache->lock);
|
|
|
|
|
|
|
|
if (cache != page->cache || _IgnorePage(page)) {
|
|
|
|
mutex_unlock(&cache->lock);
|
|
|
|
vm_cache_release_ref(cache);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
fPage = page;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
PageCacheLocker::Unlock()
|
|
|
|
{
|
|
|
|
if (fPage == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
vm_cache* cache = fPage->cache;
|
|
|
|
mutex_unlock(&cache->lock);
|
|
|
|
vm_cache_release_ref(cache);
|
|
|
|
|
|
|
|
fPage = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
clear_page_activation(int32 index)
|
|
|
|
{
|
|
|
|
vm_page *page = vm_page_at_index(index);
|
|
|
|
PageCacheLocker locker(page);
|
|
|
|
if (!locker.IsLocked())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (page->state == PAGE_STATE_ACTIVE)
|
|
|
|
vm_clear_map_activation(page);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool
|
|
|
|
check_page_activation(int32 index)
|
|
|
|
{
|
|
|
|
vm_page *page = vm_page_at_index(index);
|
|
|
|
PageCacheLocker locker(page);
|
|
|
|
if (!locker.IsLocked())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool modified;
|
|
|
|
int32 activation = vm_test_map_activation(page, &modified);
|
|
|
|
if (modified && page->state != PAGE_STATE_MODIFIED) {
|
|
|
|
//dprintf("page %p -> move to modified\n", page);
|
|
|
|
vm_page_set_state(page, PAGE_STATE_MODIFIED);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (activation > 0) {
|
|
|
|
// page is still in active use
|
|
|
|
if (page->usage_count < 0) {
|
|
|
|
if (page->state != PAGE_STATE_MODIFIED)
|
|
|
|
vm_page_set_state(page, PAGE_STATE_ACTIVE);
|
|
|
|
page->usage_count = 1;
|
|
|
|
//dprintf("page %p -> move to active\n", page);
|
|
|
|
} else if (page->usage_count < 127)
|
|
|
|
page->usage_count++;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (page->usage_count > -128)
|
|
|
|
page->usage_count--;
|
|
|
|
|
|
|
|
if (page->usage_count < 0) {
|
|
|
|
vm_remove_all_page_mappings(page);
|
|
|
|
if (page->state == PAGE_STATE_MODIFIED) {
|
|
|
|
// TODO: schedule to write back!
|
|
|
|
} else
|
|
|
|
vm_page_set_state(page, PAGE_STATE_INACTIVE);
|
|
|
|
//dprintf("page %p -> move to inactive\n", page);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static status_t
|
|
|
|
page_daemon(void *unused)
|
|
|
|
{
|
|
|
|
uint32 clearPage = 0;
|
|
|
|
uint32 checkPage = sNumPages / 2;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
acquire_sem_etc(sPageDaemonSem, 1, B_RELATIVE_TIMEOUT,
|
|
|
|
sScanWaitInterval);
|
|
|
|
|
|
|
|
if (vm_low_memory_state() < B_LOW_MEMORY_NOTE) {
|
|
|
|
// don't do anything if we have enough free memory left
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 leftToFree = 32;
|
|
|
|
// TODO: make this depending on the low memory state
|
|
|
|
|
|
|
|
for (uint32 i = 0; i < sScanPagesCount && leftToFree > 0; i++) {
|
|
|
|
if (clearPage == 0)
|
|
|
|
dprintf("clear through\n");
|
|
|
|
if (checkPage == 0)
|
|
|
|
dprintf("check through\n");
|
|
|
|
clear_page_activation(clearPage);
|
|
|
|
|
|
|
|
if (check_page_activation(checkPage))
|
|
|
|
leftToFree--;
|
|
|
|
|
|
|
|
if (++clearPage == sNumPages)
|
|
|
|
clearPage = 0;
|
|
|
|
if (++checkPage == sNumPages)
|
|
|
|
checkPage = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
status_t
|
|
|
|
vm_daemon_init()
|
|
|
|
{
|
|
|
|
sPageDaemonSem = create_sem(0, "page daemon");
|
|
|
|
|
|
|
|
sNumPages = vm_page_num_pages();
|
|
|
|
// TODO: compute those depending on sNumPages and memory pressure!
|
|
|
|
sScanPagesCount = 512;
|
|
|
|
sScanWaitInterval = 1000000;
|
|
|
|
|
|
|
|
// create a kernel thread to select pages for pageout
|
|
|
|
thread_id thread = spawn_kernel_thread(&page_daemon, "page daemon",
|
|
|
|
B_LOW_PRIORITY, NULL);
|
|
|
|
send_signal_etc(thread, SIGCONT, B_DO_NOT_RESCHEDULE);
|
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|