2002-07-09 16:24:59 +04:00
|
|
|
/*
|
2007-01-12 18:07:18 +03:00
|
|
|
* Copyright 2002-2007, Axel Dörfler, axeld@pinc-software.de.
|
2004-11-23 06:34:04 +03:00
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*
|
|
|
|
* Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
|
|
|
|
* Distributed under the terms of the NewOS License.
|
|
|
|
*/
|
2002-10-30 02:07:06 +03:00
|
|
|
|
2004-11-08 17:25:09 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <kernel.h>
|
|
|
|
#include <vm.h>
|
|
|
|
#include <vm_priv.h>
|
|
|
|
#include <vm_cache.h>
|
|
|
|
#include <vm_page.h>
|
|
|
|
#include <int.h>
|
2005-03-19 04:58:05 +03:00
|
|
|
#include <util/khash.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <lock.h>
|
|
|
|
#include <debug.h>
|
|
|
|
#include <lock.h>
|
|
|
|
#include <smp.h>
|
|
|
|
#include <arch/cpu.h>
|
2005-03-19 04:58:05 +03:00
|
|
|
|
2006-10-10 21:16:06 +04:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
//#define TRACE_VM_CACHE
|
|
|
|
#ifdef TRACE_VM_CACHE
|
|
|
|
# define TRACE(x) dprintf x
|
|
|
|
#else
|
|
|
|
# define TRACE(x) ;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
/* hash table of pages keyed by cache they're in and offset */
|
2005-12-21 20:05:50 +03:00
|
|
|
#define PAGE_TABLE_SIZE 1024 /* TODO: make this dynamic */
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
static void *sPageCacheTable;
|
|
|
|
static spinlock sPageCacheTableLock;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
struct page_lookup_key {
|
2005-12-21 15:38:31 +03:00
|
|
|
uint32 offset;
|
2004-10-09 02:56:51 +04:00
|
|
|
vm_cache *cache;
|
2002-07-09 16:24:59 +04:00
|
|
|
};
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
static int
|
|
|
|
page_compare_func(void *_p, const void *_key)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-10-09 02:56:51 +04:00
|
|
|
vm_page *page = _p;
|
2002-07-09 16:24:59 +04:00
|
|
|
const struct page_lookup_key *key = _key;
|
|
|
|
|
2004-11-23 06:34:04 +03:00
|
|
|
TRACE(("page_compare_func: page %p, key %p\n", page, key));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2005-12-21 15:38:31 +03:00
|
|
|
if (page->cache == key->cache && page->cache_offset == key->offset)
|
2002-07-09 16:24:59 +04:00
|
|
|
return 0;
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
return -1;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-29 11:38:52 +03:00
|
|
|
static uint32
|
|
|
|
page_hash_func(void *_p, const void *_key, uint32 range)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-10-09 02:56:51 +04:00
|
|
|
vm_page *page = _p;
|
2002-07-09 16:24:59 +04:00
|
|
|
const struct page_lookup_key *key = _key;
|
2005-12-21 15:38:31 +03:00
|
|
|
|
|
|
|
#define HASH(offset, ref) ((offset) ^ ((uint32)(ref) >> 4))
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2004-10-09 02:56:51 +04:00
|
|
|
if (page)
|
2005-12-21 15:38:31 +03:00
|
|
|
return HASH(page->cache_offset, page->cache) % range;
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2004-10-09 02:56:51 +04:00
|
|
|
return HASH(key->offset, key->cache) % range;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2004-10-08 19:10:50 +04:00
|
|
|
status_t
|
2006-10-10 21:16:06 +04:00
|
|
|
vm_cache_init(kernel_args *args)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2006-10-10 21:25:38 +04:00
|
|
|
sPageCacheTable = hash_init(PAGE_TABLE_SIZE, offsetof(vm_page, hash_next),
|
2004-09-11 03:43:15 +04:00
|
|
|
&page_compare_func, &page_hash_func);
|
2006-10-10 21:25:38 +04:00
|
|
|
if (sPageCacheTable == NULL)
|
|
|
|
panic("vm_cache_init: no memory\n");
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-10-08 19:10:50 +04:00
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
vm_cache *
|
|
|
|
vm_cache_create(vm_store *store)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
vm_cache *cache;
|
|
|
|
|
2005-12-21 20:05:50 +03:00
|
|
|
if (store == NULL) {
|
|
|
|
panic("vm_cache created with NULL store!");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
cache = malloc(sizeof(vm_cache));
|
2004-09-11 03:43:15 +04:00
|
|
|
if (cache == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
return NULL;
|
|
|
|
|
2006-10-10 21:16:06 +04:00
|
|
|
list_init(&cache->consumers);
|
2002-07-09 16:24:59 +04:00
|
|
|
cache->page_list = NULL;
|
|
|
|
cache->ref = NULL;
|
|
|
|
cache->source = NULL;
|
2007-01-12 18:07:18 +03:00
|
|
|
cache->virtual_base = 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
cache->virtual_size = 0;
|
|
|
|
cache->temporary = 0;
|
|
|
|
cache->scan_skip = 0;
|
2006-03-18 23:17:31 +03:00
|
|
|
cache->page_count = 0;
|
2007-02-01 15:12:54 +03:00
|
|
|
cache->busy = false;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2005-12-21 20:05:50 +03:00
|
|
|
// connect the store to its cache
|
|
|
|
cache->store = store;
|
|
|
|
store->cache = cache;
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
return cache;
|
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-03-06 16:06:10 +03:00
|
|
|
status_t
|
2004-09-11 03:43:15 +04:00
|
|
|
vm_cache_ref_create(vm_cache *cache)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
vm_cache_ref *ref;
|
2005-12-21 20:58:49 +03:00
|
|
|
status_t status;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
ref = malloc(sizeof(vm_cache_ref));
|
2004-09-11 03:43:15 +04:00
|
|
|
if (ref == NULL)
|
2006-03-06 16:06:10 +03:00
|
|
|
return B_NO_MEMORY;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2005-12-21 20:58:49 +03:00
|
|
|
status = mutex_init(&ref->lock, "cache_ref_mutex");
|
|
|
|
if (status < B_OK && (!kernel_startup || status != B_NO_MORE_SEMS)) {
|
|
|
|
// During early boot, we cannot create semaphores - they are
|
|
|
|
// created later in vm_init_post_sem()
|
|
|
|
free(ref);
|
2006-03-06 16:06:10 +03:00
|
|
|
return status;
|
2005-12-21 20:05:50 +03:00
|
|
|
}
|
|
|
|
|
2004-11-08 17:25:09 +03:00
|
|
|
ref->areas = NULL;
|
2005-12-21 20:00:03 +03:00
|
|
|
ref->ref_count = 1;
|
2005-12-21 20:05:50 +03:00
|
|
|
|
|
|
|
// connect the cache to its ref
|
|
|
|
ref->cache = cache;
|
2002-07-09 16:24:59 +04:00
|
|
|
cache->ref = ref;
|
|
|
|
|
2006-03-06 16:06:10 +03:00
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
void
|
2006-10-10 21:25:38 +04:00
|
|
|
vm_cache_acquire_ref(vm_cache_ref *cacheRef)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2006-10-10 21:25:38 +04:00
|
|
|
TRACE(("vm_cache_acquire_ref: cacheRef %p, ref will be %ld\n",
|
|
|
|
cacheRef, cacheRef->ref_count + 1));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
panic("vm_cache_acquire_ref: passed NULL\n");
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef->cache->store->ops->acquire_ref != NULL)
|
|
|
|
cacheRef->cache->store->ops->acquire_ref(cacheRef->cache->store);
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
atomic_add(&cacheRef->ref_count, 1);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
void
|
2006-10-10 21:25:38 +04:00
|
|
|
vm_cache_release_ref(vm_cache_ref *cacheRef)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
vm_page *page;
|
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
TRACE(("vm_cache_release_ref: cacheRef %p, ref will be %ld\n",
|
|
|
|
cacheRef, cacheRef->ref_count - 1));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef == NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
panic("vm_cache_release_ref: passed NULL\n");
|
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
if (atomic_add(&cacheRef->ref_count, -1) != 1) {
|
2005-12-21 20:00:03 +03:00
|
|
|
// the store ref is only released on the "working" refs, not
|
|
|
|
// on the initial one (this is vnode specific)
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef->cache->store->ops->release_ref)
|
|
|
|
cacheRef->cache->store->ops->release_ref(cacheRef->cache->store);
|
2007-02-01 15:12:54 +03:00
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
// count min references to see if everything is okay
|
|
|
|
int32 min = 0;
|
|
|
|
vm_area *a;
|
|
|
|
vm_cache *c;
|
|
|
|
bool locked = false;
|
|
|
|
if (cacheRef->lock.holder != find_thread(NULL)) {
|
|
|
|
mutex_lock(&cacheRef->lock);
|
|
|
|
locked = true;
|
|
|
|
}
|
|
|
|
for (a = cacheRef->areas; a != NULL; a = a->cache_next)
|
|
|
|
min++;
|
|
|
|
for (c = NULL; (c = list_get_next_item(&cacheRef->cache->consumers, c)) != NULL; )
|
|
|
|
min++;
|
|
|
|
dprintf("! %ld release cache_ref %p, ref_count is now %ld (min %ld, called from %p)\n", find_thread(NULL), cacheRef, cacheRef->ref_count,
|
|
|
|
min, ((struct stack_frame *)x86_read_ebp())->return_address);
|
|
|
|
if (cacheRef->ref_count < min)
|
|
|
|
panic("cache_ref %p has too little ref_count!!!!", cacheRef);
|
|
|
|
if (locked)
|
|
|
|
mutex_unlock(&cacheRef->lock);
|
|
|
|
}
|
|
|
|
#endif
|
2004-09-11 03:43:15 +04:00
|
|
|
return;
|
|
|
|
}
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
// delete this cache
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-12-20 15:30:07 +03:00
|
|
|
if (cacheRef->areas != NULL)
|
2007-01-14 21:41:57 +03:00
|
|
|
panic("cache_ref %p to be deleted still has areas", cacheRef);
|
2006-12-20 15:30:07 +03:00
|
|
|
if (!list_is_empty(&cacheRef->cache->consumers))
|
|
|
|
panic("cache %p to be deleted still has consumers", cacheRef->cache);
|
|
|
|
|
2005-12-21 20:05:50 +03:00
|
|
|
// delete the cache's backing store
|
2006-10-10 21:25:38 +04:00
|
|
|
cacheRef->cache->store->ops->destroy(cacheRef->cache->store);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
// free all of the pages in the cache
|
2006-10-10 21:25:38 +04:00
|
|
|
page = cacheRef->cache->page_list;
|
2004-09-11 03:43:15 +04:00
|
|
|
while (page) {
|
|
|
|
vm_page *oldPage = page;
|
|
|
|
int state;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
page = page->cache_next;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
// remove it from the hash table
|
|
|
|
state = disable_interrupts();
|
2006-10-10 21:25:38 +04:00
|
|
|
acquire_spinlock(&sPageCacheTableLock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
hash_remove(sPageCacheTable, oldPage);
|
2007-02-01 15:12:54 +03:00
|
|
|
oldPage->cache = NULL;
|
|
|
|
// TODO: we also need to remove all of the page's mappings!
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
release_spinlock(&sPageCacheTableLock);
|
2004-09-11 03:43:15 +04:00
|
|
|
restore_interrupts(state);
|
|
|
|
|
2005-12-21 20:00:03 +03:00
|
|
|
TRACE(("vm_cache_release_ref: freeing page 0x%lx\n",
|
|
|
|
oldPage->physical_page_number));
|
2004-09-11 03:43:15 +04:00
|
|
|
vm_page_set_state(oldPage, PAGE_STATE_FREE);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
// remove the ref to the source
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef->cache->source)
|
|
|
|
vm_cache_remove_consumer(cacheRef->cache->source->ref, cacheRef->cache);
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
mutex_destroy(&cacheRef->lock);
|
|
|
|
free(cacheRef->cache);
|
|
|
|
free(cacheRef);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
vm_page *
|
2006-10-10 21:25:38 +04:00
|
|
|
vm_cache_lookup_page(vm_cache_ref *cacheRef, off_t offset)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct page_lookup_key key;
|
2004-12-14 01:05:47 +03:00
|
|
|
cpu_status state;
|
|
|
|
vm_page *page;
|
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2005-12-21 15:38:31 +03:00
|
|
|
key.offset = (uint32)(offset >> PAGE_SHIFT);
|
2006-10-10 21:25:38 +04:00
|
|
|
key.cache = cacheRef->cache;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2006-10-10 21:25:38 +04:00
|
|
|
acquire_spinlock(&sPageCacheTableLock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
page = hash_lookup(sPageCacheTable, &key);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
release_spinlock(&sPageCacheTableLock);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
|
|
|
void
|
2006-03-18 23:17:31 +03:00
|
|
|
vm_cache_insert_page(vm_cache_ref *cacheRef, vm_page *page, off_t offset)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-12-14 01:05:47 +03:00
|
|
|
cpu_status state;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
TRACE(("vm_cache_insert_page: cacheRef %p, page %p, offset %Ld\n", cacheRef, page, offset));
|
2006-03-18 23:17:31 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2005-12-21 15:38:31 +03:00
|
|
|
page->cache_offset = (uint32)(offset >> PAGE_SHIFT);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-03-18 23:17:31 +03:00
|
|
|
if (cacheRef->cache->page_list != NULL)
|
|
|
|
cacheRef->cache->page_list->cache_prev = page;
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-03-18 23:17:31 +03:00
|
|
|
page->cache_next = cacheRef->cache->page_list;
|
2002-07-09 16:24:59 +04:00
|
|
|
page->cache_prev = NULL;
|
2006-03-18 23:17:31 +03:00
|
|
|
cacheRef->cache->page_list = page;
|
|
|
|
cacheRef->cache->page_count++;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-03-18 23:17:31 +03:00
|
|
|
page->cache = cacheRef->cache;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2006-10-10 21:25:38 +04:00
|
|
|
acquire_spinlock(&sPageCacheTableLock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
hash_insert(sPageCacheTable, page);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
release_spinlock(&sPageCacheTableLock);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
/*!
|
|
|
|
Removes the vm_page from this cache. Of course, the page must
|
|
|
|
really be in this cache or evil things will happen.
|
|
|
|
The vm_cache_ref lock must be held.
|
|
|
|
*/
|
2004-09-11 03:43:15 +04:00
|
|
|
void
|
2006-03-18 23:17:31 +03:00
|
|
|
vm_cache_remove_page(vm_cache_ref *cacheRef, vm_page *page)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-12-14 01:05:47 +03:00
|
|
|
cpu_status state;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-03-18 23:17:31 +03:00
|
|
|
TRACE(("vm_cache_remove_page: cache %p, page %p\n", cacheRef, page));
|
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2006-10-10 21:25:38 +04:00
|
|
|
acquire_spinlock(&sPageCacheTableLock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
hash_remove(sPageCacheTable, page);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
release_spinlock(&sPageCacheTableLock);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-03-18 23:17:31 +03:00
|
|
|
if (cacheRef->cache->page_list == page) {
|
2004-09-11 03:43:15 +04:00
|
|
|
if (page->cache_next != NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
page->cache_next->cache_prev = NULL;
|
2006-03-18 23:17:31 +03:00
|
|
|
cacheRef->cache->page_list = page->cache_next;
|
2002-07-09 16:24:59 +04:00
|
|
|
} else {
|
2004-09-11 03:43:15 +04:00
|
|
|
if (page->cache_prev != NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
page->cache_prev->cache_next = page->cache_next;
|
2004-09-11 03:43:15 +04:00
|
|
|
if (page->cache_next != NULL)
|
2002-07-09 16:24:59 +04:00
|
|
|
page->cache_next->cache_prev = page->cache_prev;
|
|
|
|
}
|
2006-03-18 23:17:31 +03:00
|
|
|
cacheRef->cache->page_count--;
|
2004-10-09 02:56:51 +04:00
|
|
|
page->cache = NULL;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2004-11-23 06:34:04 +03:00
|
|
|
status_t
|
2006-04-12 17:34:04 +04:00
|
|
|
vm_cache_write_modified(vm_cache_ref *ref, bool fsReenter)
|
2004-11-23 06:34:04 +03:00
|
|
|
{
|
|
|
|
status_t status;
|
|
|
|
|
|
|
|
TRACE(("vm_cache_write_modified(ref = %p)\n", ref));
|
|
|
|
|
|
|
|
mutex_lock(&ref->lock);
|
2006-04-12 17:34:04 +04:00
|
|
|
status = vm_page_write_modified(ref->cache, fsReenter);
|
2004-11-23 06:34:04 +03:00
|
|
|
mutex_unlock(&ref->lock);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-01-14 21:41:57 +03:00
|
|
|
/*!
|
|
|
|
Commits the memory to the store if the \a commitment is larger than
|
|
|
|
what's committed already.
|
|
|
|
Assumes you have the \a ref's lock held.
|
|
|
|
*/
|
2004-11-03 20:24:41 +03:00
|
|
|
status_t
|
2007-01-14 21:41:57 +03:00
|
|
|
vm_cache_set_minimal_commitment_locked(vm_cache_ref *ref, off_t commitment)
|
2004-11-03 20:24:41 +03:00
|
|
|
{
|
|
|
|
status_t status = B_OK;
|
2007-01-14 21:41:57 +03:00
|
|
|
vm_store *store = ref->cache->store;
|
2004-11-03 20:24:41 +03:00
|
|
|
|
2007-01-14 21:41:57 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&ref->lock);
|
2004-11-03 20:24:41 +03:00
|
|
|
|
|
|
|
// If we don't have enough committed space to cover through to the new end of region...
|
|
|
|
if (store->committed_size < commitment) {
|
|
|
|
// ToDo: should we check if the cache's virtual size is large
|
|
|
|
// enough for a commitment of that size?
|
|
|
|
|
|
|
|
// try to commit more memory
|
2005-12-21 20:05:50 +03:00
|
|
|
status = store->ops->commit(store, commitment);
|
2004-11-03 20:24:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
/*!
|
|
|
|
This function updates the size field of the vm_cache structure.
|
|
|
|
If needed, it will free up all pages that don't belong to the cache anymore.
|
|
|
|
The vm_cache_ref lock must be held when you call it.
|
|
|
|
Since removed pages don't belong to the cache any longer, they are not
|
|
|
|
written back before they will be removed.
|
|
|
|
*/
|
2004-10-08 19:10:50 +04:00
|
|
|
status_t
|
2004-11-23 06:34:04 +03:00
|
|
|
vm_cache_resize(vm_cache_ref *cacheRef, off_t newSize)
|
2004-10-08 19:10:50 +04:00
|
|
|
{
|
|
|
|
vm_cache *cache = cacheRef->cache;
|
2004-11-03 20:24:41 +03:00
|
|
|
status_t status;
|
2006-03-07 01:28:40 +03:00
|
|
|
uint32 oldPageCount, newPageCount;
|
2004-10-08 19:10:50 +04:00
|
|
|
|
2004-11-03 20:24:41 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
|
|
|
|
|
|
|
status = cache->store->ops->commit(cache->store, newSize);
|
|
|
|
if (status != B_OK)
|
|
|
|
return status;
|
|
|
|
|
2006-03-07 01:28:40 +03:00
|
|
|
oldPageCount = (uint32)((cache->virtual_size + B_PAGE_SIZE - 1) >> PAGE_SHIFT);
|
|
|
|
newPageCount = (uint32)((newSize + B_PAGE_SIZE - 1) >> PAGE_SHIFT);
|
|
|
|
|
|
|
|
if (newPageCount < oldPageCount) {
|
2004-10-08 19:10:50 +04:00
|
|
|
// we need to remove all pages in the cache outside of the new virtual size
|
|
|
|
vm_page *page, *next;
|
|
|
|
|
|
|
|
for (page = cache->page_list; page; page = next) {
|
|
|
|
next = page->cache_next;
|
|
|
|
|
2006-03-07 01:28:40 +03:00
|
|
|
if (page->cache_offset >= newPageCount) {
|
2004-10-08 19:10:50 +04:00
|
|
|
// remove the page and put it into the free queue
|
|
|
|
vm_cache_remove_page(cacheRef, page);
|
|
|
|
vm_page_set_state(page, PAGE_STATE_FREE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cache->virtual_size = newSize;
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-10 21:16:06 +04:00
|
|
|
/*!
|
|
|
|
Removes the \a consumer from the \a cacheRef's cache.
|
|
|
|
It will also release the reference to the cacheRef owned by the consumer.
|
2007-01-14 21:41:57 +03:00
|
|
|
Assumes you have the consumer's cache_ref lock held.
|
2006-10-10 21:16:06 +04:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
|
|
|
|
{
|
2007-01-15 02:29:23 +03:00
|
|
|
vm_cache *cache;
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
TRACE(("remove consumer vm cache %p from cache %p\n", consumer, cacheRef->cache));
|
2007-01-14 21:41:57 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&consumer->ref->lock);
|
2006-10-10 21:16:06 +04:00
|
|
|
|
2007-01-20 15:49:44 +03:00
|
|
|
// remove the consumer from the cache, but keep its reference until later
|
2006-10-10 21:16:06 +04:00
|
|
|
mutex_lock(&cacheRef->lock);
|
2007-01-15 02:29:23 +03:00
|
|
|
cache = cacheRef->cache;
|
2006-10-10 21:16:06 +04:00
|
|
|
list_remove_item(&cache->consumers, consumer);
|
|
|
|
consumer->source = NULL;
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2006-10-10 21:16:06 +04:00
|
|
|
if (cacheRef->areas == NULL && cache->source != NULL
|
|
|
|
&& !list_is_empty(&cache->consumers)
|
|
|
|
&& cache->consumers.link.next == cache->consumers.link.prev) {
|
|
|
|
// The cache is not really needed anymore - it can be merged with its only
|
|
|
|
// consumer left.
|
2006-10-11 02:47:00 +04:00
|
|
|
vm_cache_ref *consumerRef;
|
2007-01-15 02:29:23 +03:00
|
|
|
bool merge = false;
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
consumer = list_get_first_item(&cache->consumers);
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
// Our cache doesn't have a ref to its consumer (only the other way around),
|
|
|
|
// so we cannot just acquire it here; it might be deleted right now
|
|
|
|
while (true) {
|
2007-01-20 15:49:44 +03:00
|
|
|
int32 count;
|
|
|
|
consumerRef = consumer->ref;
|
|
|
|
|
|
|
|
count = consumerRef->ref_count;
|
2007-01-15 02:29:23 +03:00
|
|
|
if (count == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (atomic_test_and_set(&consumerRef->ref_count, count + 1, count) == count) {
|
2007-01-20 15:49:44 +03:00
|
|
|
// We managed to grab a reference to the consumerRef.
|
|
|
|
// Since this doesn't guarantee that we get the cache we wanted
|
|
|
|
// to, we need to check if this cache is really the last
|
|
|
|
// consumer of the cache we want to merge it with.
|
2007-01-15 02:29:23 +03:00
|
|
|
merge = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (merge) {
|
|
|
|
// But since we need to keep the locking order upper->lower cache, we
|
|
|
|
// need to unlock our cache now
|
2007-02-01 15:12:54 +03:00
|
|
|
cache->busy = true;
|
2007-01-15 02:29:23 +03:00
|
|
|
mutex_unlock(&cacheRef->lock);
|
|
|
|
|
|
|
|
mutex_lock(&consumerRef->lock);
|
|
|
|
mutex_lock(&cacheRef->lock);
|
|
|
|
|
|
|
|
// the cache and the situation might have changed
|
|
|
|
cache = cacheRef->cache;
|
2007-01-20 15:49:44 +03:00
|
|
|
consumer = consumerRef->cache;
|
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
if (cacheRef->areas != NULL || cache->source == NULL
|
|
|
|
|| list_is_empty(&cache->consumers)
|
|
|
|
|| cache->consumers.link.next != cache->consumers.link.prev
|
2007-01-20 15:49:44 +03:00
|
|
|
|| consumer != list_get_first_item(&cache->consumers)) {
|
2007-01-15 02:29:23 +03:00
|
|
|
merge = false;
|
2007-02-01 15:12:54 +03:00
|
|
|
cache->busy = false;
|
2007-01-15 02:29:23 +03:00
|
|
|
mutex_unlock(&consumerRef->lock);
|
2007-02-01 15:12:54 +03:00
|
|
|
vm_cache_release_ref(consumerRef);
|
2007-01-15 02:29:23 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (merge) {
|
|
|
|
vm_page *page, *nextPage;
|
2007-01-20 15:49:44 +03:00
|
|
|
vm_cache *newSource;
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
consumer = list_remove_head_item(&cache->consumers);
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
TRACE(("merge vm cache %p (ref == %ld) with vm cache %p\n",
|
|
|
|
cache, cacheRef->ref_count, consumer));
|
2006-10-11 02:47:00 +04:00
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
for (page = cache->page_list; page != NULL; page = nextPage) {
|
|
|
|
nextPage = page->cache_next;
|
|
|
|
|
|
|
|
if (vm_cache_lookup_page(consumerRef,
|
2007-02-01 15:12:54 +03:00
|
|
|
(off_t)page->cache_offset << PAGE_SHIFT) == NULL) {
|
|
|
|
// the page already is not yet in the consumer cache - move it upwards
|
|
|
|
vm_cache_remove_page(cacheRef, page);
|
|
|
|
vm_cache_insert_page(consumerRef, page,
|
|
|
|
(off_t)page->cache_offset << PAGE_SHIFT);
|
2007-01-15 02:29:23 +03:00
|
|
|
}
|
|
|
|
|
2007-02-01 15:12:54 +03:00
|
|
|
// TODO: if we'd remove the pages in the cache, we'd also
|
|
|
|
// need to remove all of their mappings!
|
2006-10-11 02:47:00 +04:00
|
|
|
}
|
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
newSource = cache->source;
|
2007-02-01 15:12:54 +03:00
|
|
|
|
|
|
|
// The remaining consumer has gotten a new source
|
|
|
|
mutex_lock(&newSource->ref->lock);
|
|
|
|
|
|
|
|
list_remove_item(&newSource->consumers, cache);
|
|
|
|
list_add_item(&newSource->consumers, consumer);
|
|
|
|
consumer->source = newSource;
|
|
|
|
cache->source = NULL;
|
|
|
|
|
|
|
|
mutex_unlock(&newSource->ref->lock);
|
|
|
|
|
|
|
|
// Release the other reference to the cache - we take over
|
|
|
|
// its reference of its source cache; we can do this here
|
|
|
|
// (with the cacheRef locked) since we own another reference
|
|
|
|
// from the first consumer we removed
|
|
|
|
if (cacheRef->ref_count < 2)
|
|
|
|
panic("cacheRef %p ref count too low!\n", cacheRef);
|
|
|
|
vm_cache_release_ref(cacheRef);
|
|
|
|
|
2007-01-15 02:29:23 +03:00
|
|
|
mutex_unlock(&consumerRef->lock);
|
2007-02-01 15:12:54 +03:00
|
|
|
vm_cache_release_ref(consumerRef);
|
2006-10-11 02:47:00 +04:00
|
|
|
}
|
2006-10-10 21:16:06 +04:00
|
|
|
}
|
|
|
|
|
2007-01-14 21:41:57 +03:00
|
|
|
mutex_unlock(&cacheRef->lock);
|
2006-10-10 21:16:06 +04:00
|
|
|
vm_cache_release_ref(cacheRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
Marks the \a cacheRef's cache as source of the \a consumer cache,
|
|
|
|
and adds the \a consumer to its list.
|
|
|
|
This also grabs a reference to the source cache.
|
2007-01-14 21:41:57 +03:00
|
|
|
Assumes you have the cache_ref and the consumer's lock held.
|
2006-10-10 21:16:06 +04:00
|
|
|
*/
|
|
|
|
void
|
2007-01-14 21:41:57 +03:00
|
|
|
vm_cache_add_consumer_locked(vm_cache_ref *cacheRef, vm_cache *consumer)
|
2006-10-10 21:16:06 +04:00
|
|
|
{
|
|
|
|
TRACE(("add consumer vm cache %p to cache %p\n", consumer, cacheRef->cache));
|
2007-01-14 21:41:57 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
|
|
|
ASSERT_LOCKED_MUTEX(&consumer->ref->lock);
|
2006-10-10 21:16:06 +04:00
|
|
|
|
|
|
|
consumer->source = cacheRef->cache;
|
|
|
|
list_add_item(&cacheRef->cache->consumers, consumer);
|
|
|
|
|
|
|
|
vm_cache_acquire_ref(cacheRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-01-14 21:41:57 +03:00
|
|
|
/*!
|
|
|
|
Adds the \a area to the \a cacheRef.
|
|
|
|
Assumes you have the locked the cache_ref.
|
|
|
|
*/
|
2004-10-08 19:10:50 +04:00
|
|
|
status_t
|
2007-01-14 21:41:57 +03:00
|
|
|
vm_cache_insert_area_locked(vm_cache_ref *cacheRef, vm_area *area)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2007-01-14 21:41:57 +03:00
|
|
|
ASSERT_LOCKED_MUTEX(&cacheRef->lock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
area->cache_next = cacheRef->areas;
|
2004-11-08 17:25:09 +03:00
|
|
|
if (area->cache_next)
|
|
|
|
area->cache_next->cache_prev = area;
|
|
|
|
area->cache_prev = NULL;
|
2006-10-10 21:25:38 +04:00
|
|
|
cacheRef->areas = area;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-10-08 19:10:50 +04:00
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2004-09-11 03:43:15 +04:00
|
|
|
|
2004-10-08 19:10:50 +04:00
|
|
|
status_t
|
2006-10-10 21:25:38 +04:00
|
|
|
vm_cache_remove_area(vm_cache_ref *cacheRef, vm_area *area)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2006-10-10 21:25:38 +04:00
|
|
|
mutex_lock(&cacheRef->lock);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-11-08 17:25:09 +03:00
|
|
|
if (area->cache_prev)
|
|
|
|
area->cache_prev->cache_next = area->cache_next;
|
|
|
|
if (area->cache_next)
|
|
|
|
area->cache_next->cache_prev = area->cache_prev;
|
2006-10-10 21:25:38 +04:00
|
|
|
if (cacheRef->areas == area)
|
|
|
|
cacheRef->areas = area->cache_next;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2006-10-10 21:25:38 +04:00
|
|
|
mutex_unlock(&cacheRef->lock);
|
2004-10-08 19:10:50 +04:00
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|