/* * Copyright 2008-2011, Ingo Weinhold, ingo_weinhold@gmx.de. * Copyright 2002-2008, Axel Dörfler, axeld@pinc-software.de. * Distributed under the terms of the MIT License. * * Copyright 2001-2002, Travis Geiselbrecht. All rights reserved. * Distributed under the terms of the NewOS License. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include // needed for the factory only #include "VMAnonymousCache.h" #include "VMAnonymousNoSwapCache.h" #include "VMDeviceCache.h" #include "VMNullCache.h" #include "../cache/vnode_store.h" //#define TRACE_VM_CACHE #ifdef TRACE_VM_CACHE # define TRACE(x) dprintf x #else # define TRACE(x) ; #endif #if DEBUG_CACHE_LIST VMCache* gDebugCacheList; #endif static mutex sCacheListLock = MUTEX_INITIALIZER("global VMCache list"); // The lock is also needed when the debug feature is disabled. ObjectCache* gCacheRefObjectCache; #if ENABLE_SWAP_SUPPORT ObjectCache* gAnonymousCacheObjectCache; #endif ObjectCache* gAnonymousNoSwapCacheObjectCache; ObjectCache* gVnodeCacheObjectCache; ObjectCache* gDeviceCacheObjectCache; ObjectCache* gNullCacheObjectCache; struct VMCache::PageEventWaiter { Thread* thread; PageEventWaiter* next; vm_page* page; uint32 events; }; #if VM_CACHE_TRACING namespace VMCacheTracing { class VMCacheTraceEntry : public AbstractTraceEntry { public: VMCacheTraceEntry(VMCache* cache) : fCache(cache) { #if VM_CACHE_TRACING_STACK_TRACE fStackTrace = capture_tracing_stack_trace( VM_CACHE_TRACING_STACK_TRACE, 0, true); // Don't capture userland stack trace to avoid potential // deadlocks. #endif } #if VM_CACHE_TRACING_STACK_TRACE virtual void DumpStackTrace(TraceOutput& out) { out.PrintStackTrace(fStackTrace); } #endif VMCache* Cache() const { return fCache; } protected: VMCache* fCache; #if VM_CACHE_TRACING_STACK_TRACE tracing_stack_trace* fStackTrace; #endif }; class Create : public VMCacheTraceEntry { public: Create(VMCache* cache) : VMCacheTraceEntry(cache) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache create: -> cache: %p", fCache); } }; class Delete : public VMCacheTraceEntry { public: Delete(VMCache* cache) : VMCacheTraceEntry(cache) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache delete: cache: %p", fCache); } }; class SetMinimalCommitment : public VMCacheTraceEntry { public: SetMinimalCommitment(VMCache* cache, off_t commitment) : VMCacheTraceEntry(cache), fOldCommitment(cache->committed_size), fCommitment(commitment) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache set min commitment: cache: %p, " "commitment: %" B_PRIdOFF " -> %" B_PRIdOFF, fCache, fOldCommitment, fCommitment); } private: off_t fOldCommitment; off_t fCommitment; }; class Resize : public VMCacheTraceEntry { public: Resize(VMCache* cache, off_t size) : VMCacheTraceEntry(cache), fOldSize(cache->virtual_end), fSize(size) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache resize: cache: %p, size: %" B_PRIdOFF " -> %" B_PRIdOFF, fCache, fOldSize, fSize); } private: off_t fOldSize; off_t fSize; }; class Rebase : public VMCacheTraceEntry { public: Rebase(VMCache* cache, off_t base) : VMCacheTraceEntry(cache), fOldBase(cache->virtual_base), fBase(base) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache rebase: cache: %p, base: %lld -> %lld", fCache, fOldBase, fBase); } private: off_t fOldBase; off_t fBase; }; class AddConsumer : public VMCacheTraceEntry { public: AddConsumer(VMCache* cache, VMCache* consumer) : VMCacheTraceEntry(cache), fConsumer(consumer) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache add consumer: cache: %p, consumer: %p", fCache, fConsumer); } VMCache* Consumer() const { return fConsumer; } private: VMCache* fConsumer; }; class RemoveConsumer : public VMCacheTraceEntry { public: RemoveConsumer(VMCache* cache, VMCache* consumer) : VMCacheTraceEntry(cache), fConsumer(consumer) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache remove consumer: cache: %p, consumer: %p", fCache, fConsumer); } private: VMCache* fConsumer; }; class Merge : public VMCacheTraceEntry { public: Merge(VMCache* cache, VMCache* consumer) : VMCacheTraceEntry(cache), fConsumer(consumer) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache merge with consumer: cache: %p, consumer: %p", fCache, fConsumer); } private: VMCache* fConsumer; }; class InsertArea : public VMCacheTraceEntry { public: InsertArea(VMCache* cache, VMArea* area) : VMCacheTraceEntry(cache), fArea(area) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache insert area: cache: %p, area: %p", fCache, fArea); } VMArea* Area() const { return fArea; } private: VMArea* fArea; }; class RemoveArea : public VMCacheTraceEntry { public: RemoveArea(VMCache* cache, VMArea* area) : VMCacheTraceEntry(cache), fArea(area) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache remove area: cache: %p, area: %p", fCache, fArea); } private: VMArea* fArea; }; } // namespace VMCacheTracing # define T(x) new(std::nothrow) VMCacheTracing::x; # if VM_CACHE_TRACING >= 2 namespace VMCacheTracing { class InsertPage : public VMCacheTraceEntry { public: InsertPage(VMCache* cache, vm_page* page, off_t offset) : VMCacheTraceEntry(cache), fPage(page), fOffset(offset) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache insert page: cache: %p, page: %p, offset: %" B_PRIdOFF, fCache, fPage, fOffset); } private: vm_page* fPage; off_t fOffset; }; class RemovePage : public VMCacheTraceEntry { public: RemovePage(VMCache* cache, vm_page* page) : VMCacheTraceEntry(cache), fPage(page) { Initialized(); } virtual void AddDump(TraceOutput& out) { out.Print("vm cache remove page: cache: %p, page: %p", fCache, fPage); } private: vm_page* fPage; }; } // namespace VMCacheTracing # define T2(x) new(std::nothrow) VMCacheTracing::x; # else # define T2(x) ; # endif #else # define T(x) ; # define T2(x) ; #endif // #pragma mark - debugger commands #if VM_CACHE_TRACING static void* cache_stack_find_area_cache(const TraceEntryIterator& baseIterator, void* area) { using namespace VMCacheTracing; // find the previous "insert area" entry for the given area TraceEntryIterator iterator = baseIterator; TraceEntry* entry = iterator.Current(); while (entry != NULL) { if (InsertArea* insertAreaEntry = dynamic_cast(entry)) { if (insertAreaEntry->Area() == area) return insertAreaEntry->Cache(); } entry = iterator.Previous(); } return NULL; } static void* cache_stack_find_consumer(const TraceEntryIterator& baseIterator, void* cache) { using namespace VMCacheTracing; // find the previous "add consumer" or "create" entry for the given cache TraceEntryIterator iterator = baseIterator; TraceEntry* entry = iterator.Current(); while (entry != NULL) { if (Create* createEntry = dynamic_cast(entry)) { if (createEntry->Cache() == cache) return NULL; } else if (AddConsumer* addEntry = dynamic_cast(entry)) { if (addEntry->Consumer() == cache) return addEntry->Cache(); } entry = iterator.Previous(); } return NULL; } static int command_cache_stack(int argc, char** argv) { if (argc < 3 || argc > 4) { print_debugger_command_usage(argv[0]); return 0; } bool isArea = false; int argi = 1; if (argc == 4) { if (strcmp(argv[argi], "area") != 0) { print_debugger_command_usage(argv[0]); return 0; } argi++; isArea = true; } uint64 addressValue; uint64 debugEntryIndex; if (!evaluate_debug_expression(argv[argi++], &addressValue, false) || !evaluate_debug_expression(argv[argi++], &debugEntryIndex, false)) { return 0; } TraceEntryIterator baseIterator; if (baseIterator.MoveTo((int32)debugEntryIndex) == NULL) { kprintf("Invalid tracing entry index %" B_PRIu64 "\n", debugEntryIndex); return 0; } void* address = (void*)(addr_t)addressValue; kprintf("cache stack for %s %p at %" B_PRIu64 ":\n", isArea ? "area" : "cache", address, debugEntryIndex); if (isArea) { address = cache_stack_find_area_cache(baseIterator, address); if (address == NULL) { kprintf(" cache not found\n"); return 0; } } while (address != NULL) { kprintf(" %p\n", address); address = cache_stack_find_consumer(baseIterator, address); } return 0; } #endif // VM_CACHE_TRACING // #pragma mark - status_t vm_cache_init(kernel_args* args) { // Create object caches for the structures we allocate here. gCacheRefObjectCache = create_object_cache("cache refs", sizeof(VMCacheRef), 0, NULL, NULL, NULL); #if ENABLE_SWAP_SUPPORT gAnonymousCacheObjectCache = create_object_cache("anon caches", sizeof(VMAnonymousCache), 0, NULL, NULL, NULL); #endif gAnonymousNoSwapCacheObjectCache = create_object_cache( "anon no-swap caches", sizeof(VMAnonymousNoSwapCache), 0, NULL, NULL, NULL); gVnodeCacheObjectCache = create_object_cache("vnode caches", sizeof(VMVnodeCache), 0, NULL, NULL, NULL); gDeviceCacheObjectCache = create_object_cache("device caches", sizeof(VMDeviceCache), 0, NULL, NULL, NULL); gNullCacheObjectCache = create_object_cache("null caches", sizeof(VMNullCache), 0, NULL, NULL, NULL); if (gCacheRefObjectCache == NULL #if ENABLE_SWAP_SUPPORT || gAnonymousCacheObjectCache == NULL #endif || gAnonymousNoSwapCacheObjectCache == NULL || gVnodeCacheObjectCache == NULL || gDeviceCacheObjectCache == NULL || gNullCacheObjectCache == NULL) { panic("vm_cache_init(): Failed to create object caches!"); return B_NO_MEMORY; } return B_OK; } void vm_cache_init_post_heap() { #if VM_CACHE_TRACING add_debugger_command_etc("cache_stack", &command_cache_stack, "List the ancestors (sources) of a VMCache at the time given by " "tracing entry index", "[ \"area\" ]
\n" "All ancestors (sources) of a given VMCache at the time given by the\n" "tracing entry index are listed. If \"area\" is given the supplied\n" "address is an area instead of a cache address. The listing will\n" "start with the area's cache at that point.\n", 0); #endif // VM_CACHE_TRACING } VMCache* vm_cache_acquire_locked_page_cache(vm_page* page, bool dontWait) { mutex_lock(&sCacheListLock); while (dontWait) { VMCacheRef* cacheRef = page->CacheRef(); if (cacheRef == NULL) { mutex_unlock(&sCacheListLock); return NULL; } VMCache* cache = cacheRef->cache; if (!cache->TryLock()) { mutex_unlock(&sCacheListLock); return NULL; } if (cacheRef == page->CacheRef()) { mutex_unlock(&sCacheListLock); cache->AcquireRefLocked(); return cache; } // the cache changed in the meantime cache->Unlock(); } while (true) { VMCacheRef* cacheRef = page->CacheRef(); if (cacheRef == NULL) { mutex_unlock(&sCacheListLock); return NULL; } VMCache* cache = cacheRef->cache; if (!cache->SwitchLock(&sCacheListLock)) { // cache has been deleted mutex_lock(&sCacheListLock); continue; } mutex_lock(&sCacheListLock); if (cache == page->Cache()) { mutex_unlock(&sCacheListLock); cache->AcquireRefLocked(); return cache; } // the cache changed in the meantime cache->Unlock(); } } // #pragma mark - VMCache VMCacheRef::VMCacheRef(VMCache* cache) : cache(cache), ref_count(1) { } // #pragma mark - VMCache bool VMCache::_IsMergeable() const { return areas == NULL && temporary && !consumers.IsEmpty() && consumers.Head() == consumers.Tail(); } VMCache::VMCache() : fCacheRef(NULL) { } VMCache::~VMCache() { object_cache_delete(gCacheRefObjectCache, fCacheRef); } status_t VMCache::Init(uint32 cacheType, uint32 allocationFlags) { mutex_init(&fLock, "VMCache"); areas = NULL; fRefCount = 1; source = NULL; virtual_base = 0; virtual_end = 0; committed_size = 0; temporary = 0; page_count = 0; fWiredPagesCount = 0; type = cacheType; fPageEventWaiters = NULL; #if DEBUG_CACHE_LIST debug_previous = NULL; debug_next = NULL; // initialize in case the following fails #endif fCacheRef = new(gCacheRefObjectCache, allocationFlags) VMCacheRef(this); if (fCacheRef == NULL) return B_NO_MEMORY; #if DEBUG_CACHE_LIST mutex_lock(&sCacheListLock); if (gDebugCacheList != NULL) gDebugCacheList->debug_previous = this; debug_next = gDebugCacheList; gDebugCacheList = this; mutex_unlock(&sCacheListLock); #endif return B_OK; } void VMCache::Delete() { if (areas != NULL) panic("cache %p to be deleted still has areas", this); if (!consumers.IsEmpty()) panic("cache %p to be deleted still has consumers", this); T(Delete(this)); // free all of the pages in the cache while (vm_page* page = pages.Root()) { if (!page->mappings.IsEmpty() || page->WiredCount() != 0) { panic("remove page %p from cache %p: page still has mappings!\n" "@!page %p; cache %p", page, this, page, this); } // remove it pages.Remove(page); page->SetCacheRef(NULL); TRACE(("vm_cache_release_ref: freeing page 0x%lx\n", page->physical_page_number)); DEBUG_PAGE_ACCESS_START(page); vm_page_free(this, page); } // remove the ref to the source if (source) source->_RemoveConsumer(this); // We lock and unlock the sCacheListLock, even if the DEBUG_CACHE_LIST is // not enabled. This synchronization point is needed for // vm_cache_acquire_locked_page_cache(). mutex_lock(&sCacheListLock); #if DEBUG_CACHE_LIST if (debug_previous) debug_previous->debug_next = debug_next; if (debug_next) debug_next->debug_previous = debug_previous; if (this == gDebugCacheList) gDebugCacheList = debug_next; #endif mutex_destroy(&fLock); mutex_unlock(&sCacheListLock); DeleteObject(); } void VMCache::Unlock(bool consumerLocked) { while (fRefCount == 1 && _IsMergeable()) { VMCache* consumer = consumers.Head(); if (consumerLocked) { _MergeWithOnlyConsumer(); } else if (consumer->TryLock()) { _MergeWithOnlyConsumer(); consumer->Unlock(); } else { // Someone else has locked the consumer ATM. Unlock this cache and // wait for the consumer lock. Increment the cache's ref count // temporarily, so that no one else will try what we are doing or // delete the cache. fRefCount++; bool consumerLockedTemp = consumer->SwitchLock(&fLock); Lock(); fRefCount--; if (consumerLockedTemp) { if (fRefCount == 1 && _IsMergeable() && consumer == consumers.Head()) { // nothing has changed in the meantime -- merge _MergeWithOnlyConsumer(); } consumer->Unlock(); } } } if (fRefCount == 0) { // delete this cache Delete(); } else mutex_unlock(&fLock); } vm_page* VMCache::LookupPage(off_t offset) { AssertLocked(); vm_page* page = pages.Lookup((page_num_t)(offset >> PAGE_SHIFT)); #if KDEBUG if (page != NULL && page->Cache() != this) panic("page %p not in cache %p\n", page, this); #endif return page; } void VMCache::InsertPage(vm_page* page, off_t offset) { TRACE(("VMCache::InsertPage(): cache %p, page %p, offset %" B_PRIdOFF "\n", this, page, offset)); AssertLocked(); if (page->CacheRef() != NULL) { panic("insert page %p into cache %p: page cache is set to %p\n", page, this, page->Cache()); } T2(InsertPage(this, page, offset)); page->cache_offset = (page_num_t)(offset >> PAGE_SHIFT); page_count++; page->SetCacheRef(fCacheRef); #if KDEBUG vm_page* otherPage = pages.Lookup(page->cache_offset); if (otherPage != NULL) { panic("VMCache::InsertPage(): there's already page %p with cache " "offset %" B_PRIuPHYSADDR " in cache %p; inserting page %p", otherPage, page->cache_offset, this, page); } #endif // KDEBUG pages.Insert(page); if (page->WiredCount() > 0) IncrementWiredPagesCount(); } /*! Removes the vm_page from this cache. Of course, the page must really be in this cache or evil things will happen. The cache lock must be held. */ void VMCache::RemovePage(vm_page* page) { TRACE(("VMCache::RemovePage(): cache %p, page %p\n", this, page)); AssertLocked(); if (page->Cache() != this) { panic("remove page %p from cache %p: page cache is set to %p\n", page, this, page->Cache()); } T2(RemovePage(this, page)); pages.Remove(page); page_count--; page->SetCacheRef(NULL); if (page->WiredCount() > 0) DecrementWiredPagesCount(); } /*! Moves the given page from its current cache inserts it into this cache at the given offset. Both caches must be locked. */ void VMCache::MovePage(vm_page* page, off_t offset) { VMCache* oldCache = page->Cache(); AssertLocked(); oldCache->AssertLocked(); // remove from old cache oldCache->pages.Remove(page); oldCache->page_count--; T2(RemovePage(oldCache, page)); // change the offset page->cache_offset = offset >> PAGE_SHIFT; // insert here pages.Insert(page); page_count++; page->SetCacheRef(fCacheRef); if (page->WiredCount() > 0) { IncrementWiredPagesCount(); oldCache->DecrementWiredPagesCount(); } T2(InsertPage(this, page, page->cache_offset << PAGE_SHIFT)); } /*! Moves the given page from its current cache inserts it into this cache. Both caches must be locked. */ void VMCache::MovePage(vm_page* page) { MovePage(page, page->cache_offset << PAGE_SHIFT); } /*! Moves all pages from the given cache to this one. Both caches must be locked. This cache must be empty. */ void VMCache::MoveAllPages(VMCache* fromCache) { AssertLocked(); fromCache->AssertLocked(); ASSERT(page_count == 0); std::swap(fromCache->pages, pages); page_count = fromCache->page_count; fromCache->page_count = 0; fWiredPagesCount = fromCache->fWiredPagesCount; fromCache->fWiredPagesCount = 0; // swap the VMCacheRefs mutex_lock(&sCacheListLock); std::swap(fCacheRef, fromCache->fCacheRef); fCacheRef->cache = this; fromCache->fCacheRef->cache = fromCache; mutex_unlock(&sCacheListLock); #if VM_CACHE_TRACING >= 2 for (VMCachePagesTree::Iterator it = pages.GetIterator(); vm_page* page = it.Next();) { T2(RemovePage(fromCache, page)); T2(InsertPage(this, page, page->cache_offset << PAGE_SHIFT)); } #endif } /*! Waits until one or more events happened for a given page which belongs to this cache. The cache must be locked. It will be unlocked by the method. \a relock specifies whether the method shall re-lock the cache before returning. \param page The page for which to wait. \param events The mask of events the caller is interested in. \param relock If \c true, the cache will be locked when returning, otherwise it won't be locked. */ void VMCache::WaitForPageEvents(vm_page* page, uint32 events, bool relock) { PageEventWaiter waiter; waiter.thread = thread_get_current_thread(); waiter.next = fPageEventWaiters; waiter.page = page; waiter.events = events; fPageEventWaiters = &waiter; thread_prepare_to_block(waiter.thread, 0, THREAD_BLOCK_TYPE_OTHER, "cache page events"); Unlock(); thread_block(); if (relock) Lock(); } /*! Makes this case the source of the \a consumer cache, and adds the \a consumer to its list. This also grabs a reference to the source cache. Assumes you have the cache and the consumer's lock held. */ void VMCache::AddConsumer(VMCache* consumer) { TRACE(("add consumer vm cache %p to cache %p\n", consumer, this)); AssertLocked(); consumer->AssertLocked(); T(AddConsumer(this, consumer)); consumer->source = this; consumers.Add(consumer); AcquireRefLocked(); AcquireStoreRef(); } /*! Adds the \a area to this cache. Assumes you have the locked the cache. */ status_t VMCache::InsertAreaLocked(VMArea* area) { TRACE(("VMCache::InsertAreaLocked(cache %p, area %p)\n", this, area)); AssertLocked(); T(InsertArea(this, area)); area->cache_next = areas; if (area->cache_next) area->cache_next->cache_prev = area; area->cache_prev = NULL; areas = area; AcquireStoreRef(); return B_OK; } status_t VMCache::RemoveArea(VMArea* area) { TRACE(("VMCache::RemoveArea(cache %p, area %p)\n", this, area)); T(RemoveArea(this, area)); // We release the store reference first, since otherwise we would reverse // the locking order or even deadlock ourselves (... -> free_vnode() -> ... // -> bfs_remove_vnode() -> ... -> file_cache_set_size() -> mutex_lock()). // Also cf. _RemoveConsumer(). ReleaseStoreRef(); AutoLocker locker(this); if (area->cache_prev) area->cache_prev->cache_next = area->cache_next; if (area->cache_next) area->cache_next->cache_prev = area->cache_prev; if (areas == area) areas = area->cache_next; return B_OK; } /*! Transfers the areas from \a fromCache to this cache. This cache must not have areas yet. Both caches must be locked. */ void VMCache::TransferAreas(VMCache* fromCache) { AssertLocked(); fromCache->AssertLocked(); ASSERT(areas == NULL); areas = fromCache->areas; fromCache->areas = NULL; for (VMArea* area = areas; area != NULL; area = area->cache_next) { area->cache = this; AcquireRefLocked(); fromCache->ReleaseRefLocked(); T(RemoveArea(fromCache, area)); T(InsertArea(this, area)); } } uint32 VMCache::CountWritableAreas(VMArea* ignoreArea) const { uint32 count = 0; for (VMArea* area = areas; area != NULL; area = area->cache_next) { if (area != ignoreArea && (area->protection & (B_WRITE_AREA | B_KERNEL_WRITE_AREA)) != 0) { count++; } } return count; } status_t VMCache::WriteModified() { TRACE(("VMCache::WriteModified(cache = %p)\n", this)); if (temporary) return B_OK; Lock(); status_t status = vm_page_write_modified_pages(this); Unlock(); return status; } /*! Commits the memory to the store if the \a commitment is larger than what's committed already. Assumes you have the cache's lock held. */ status_t VMCache::SetMinimalCommitment(off_t commitment, int priority) { TRACE(("VMCache::SetMinimalCommitment(cache %p, commitment %" B_PRIdOFF ")\n", this, commitment)); AssertLocked(); T(SetMinimalCommitment(this, commitment)); status_t status = B_OK; // If we don't have enough committed space to cover through to the new end // of the area... if (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 status = Commit(commitment, priority); } return status; } bool VMCache::_FreePageRange(VMCachePagesTree::Iterator it, page_num_t* toPage = NULL) { for (vm_page* page = it.Next(); page != NULL && (toPage == NULL || page->cache_offset < *toPage); page = it.Next()) { if (page->busy) { if (page->busy_writing) { // We cannot wait for the page to become available // as we might cause a deadlock this way page->busy_writing = false; // this will notify the writer to free the page continue; } // wait for page to become unbusy WaitForPageEvents(page, PAGE_EVENT_NOT_BUSY, true); return true; } // remove the page and put it into the free queue DEBUG_PAGE_ACCESS_START(page); vm_remove_all_page_mappings(page); ASSERT(page->WiredCount() == 0); // TODO: Find a real solution! If the page is wired // temporarily (e.g. by lock_memory()), we actually must not // unmap it! RemovePage(page); // Note: When iterating through a IteratableSplayTree // removing the current node is safe. vm_page_free(this, page); } return false; } /*! This function updates the size field of the cache. If needed, it will free up all pages that don't belong to the cache anymore. The cache 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. Note, this function may temporarily release the cache lock in case it has to wait for busy pages. */ status_t VMCache::Resize(off_t newSize, int priority) { TRACE(("VMCache::Resize(cache %p, newSize %" B_PRIdOFF ") old size %" B_PRIdOFF "\n", this, newSize, this->virtual_end)); this->AssertLocked(); T(Resize(this, newSize)); status_t status = Commit(newSize - virtual_base, priority); if (status != B_OK) return status; page_num_t oldPageCount = (page_num_t)((virtual_end + B_PAGE_SIZE - 1) >> PAGE_SHIFT); page_num_t newPageCount = (page_num_t)((newSize + B_PAGE_SIZE - 1) >> PAGE_SHIFT); if (newPageCount < oldPageCount) { // we need to remove all pages in the cache outside of the new virtual // size while (_FreePageRange(pages.GetIterator(newPageCount, true, true))) ; } virtual_end = newSize; return B_OK; } /*! This function updates the virtual_base field of the cache. If needed, it will free up all pages that don't belong to the cache anymore. The cache 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. Note, this function may temporarily release the cache lock in case it has to wait for busy pages. */ status_t VMCache::Rebase(off_t newBase, int priority) { TRACE(("VMCache::Rebase(cache %p, newBase %Ld) old base %Ld\n", this, newBase, this->virtual_base)); this->AssertLocked(); T(Rebase(this, newBase)); status_t status = Commit(virtual_end - newBase, priority); if (status != B_OK) return status; page_num_t basePage = (page_num_t)(newBase >> PAGE_SHIFT); if (newBase > virtual_base) { // we need to remove all pages in the cache outside of the new virtual // base while (_FreePageRange(pages.GetIterator(), &basePage)) ; } virtual_base = newBase; return B_OK; } /*! Moves pages in the given range from the source cache into this cache. Both caches must be locked. */ status_t VMCache::Adopt(VMCache* source, off_t offset, off_t size, off_t newOffset) { page_num_t startPage = offset >> PAGE_SHIFT; page_num_t endPage = (offset + size + B_PAGE_SIZE - 1) >> PAGE_SHIFT; off_t offsetChange = newOffset - offset; VMCachePagesTree::Iterator it = source->pages.GetIterator(startPage, true, true); for (vm_page* page = it.Next(); page != NULL && page->cache_offset < endPage; page = it.Next()) { MovePage(page, (page->cache_offset << PAGE_SHIFT) + offsetChange); } return B_OK; } /*! Discards pages in the given range. */ status_t VMCache::Discard(off_t offset, off_t size) { page_num_t startPage = offset >> PAGE_SHIFT; page_num_t endPage = (offset + size + B_PAGE_SIZE - 1) >> PAGE_SHIFT; while (_FreePageRange(pages.GetIterator(startPage, true, true), &endPage)) ; return B_OK; } /*! You have to call this function with the VMCache lock held. */ status_t VMCache::FlushAndRemoveAllPages() { ASSERT_LOCKED_MUTEX(&fLock); while (page_count > 0) { // write back modified pages status_t status = vm_page_write_modified_pages(this); if (status != B_OK) return status; // remove pages for (VMCachePagesTree::Iterator it = pages.GetIterator(); vm_page* page = it.Next();) { if (page->busy) { // wait for page to become unbusy WaitForPageEvents(page, PAGE_EVENT_NOT_BUSY, true); // restart from the start of the list it = pages.GetIterator(); continue; } // skip modified pages -- they will be written back in the next // iteration if (page->State() == PAGE_STATE_MODIFIED) continue; // We can't remove mapped pages. if (page->IsMapped()) return B_BUSY; DEBUG_PAGE_ACCESS_START(page); RemovePage(page); vm_page_free(this, page); // Note: When iterating through a IteratableSplayTree // removing the current node is safe. } } return B_OK; } status_t VMCache::Commit(off_t size, int priority) { committed_size = size; return B_OK; } /*! Returns whether the cache's underlying backing store could deliver the page at the given offset. Basically it returns whether a Read() at \a offset would at least read a partial page (assuming that no unexpected errors occur or the situation changes in the meantime). */ bool VMCache::HasPage(off_t offset) { // In accordance with Fault() the default implementation doesn't have a // backing store and doesn't allow faults. return false; } status_t VMCache::Read(off_t offset, const generic_io_vec *vecs, size_t count, uint32 flags, generic_size_t *_numBytes) { return B_ERROR; } status_t VMCache::Write(off_t offset, const generic_io_vec *vecs, size_t count, uint32 flags, generic_size_t *_numBytes) { return B_ERROR; } status_t VMCache::WriteAsync(off_t offset, const generic_io_vec* vecs, size_t count, generic_size_t numBytes, uint32 flags, AsyncIOCallback* callback) { // Not supported, fall back to the synchronous hook. generic_size_t transferred = numBytes; status_t error = Write(offset, vecs, count, flags, &transferred); if (callback != NULL) callback->IOFinished(error, transferred != numBytes, transferred); return error; } /*! \brief Returns whether the cache can write the page at the given offset. The cache must be locked when this function is invoked. @param offset The page offset. @return \c true, if the page can be written, \c false otherwise. */ bool VMCache::CanWritePage(off_t offset) { return false; } status_t VMCache::Fault(struct VMAddressSpace *aspace, off_t offset) { return B_BAD_ADDRESS; } void VMCache::Merge(VMCache* source) { for (VMCachePagesTree::Iterator it = source->pages.GetIterator(); vm_page* page = it.Next();) { // Note: Removing the current node while iterating through a // IteratableSplayTree is safe. vm_page* consumerPage = LookupPage( (off_t)page->cache_offset << PAGE_SHIFT); if (consumerPage == NULL) { // the page is not yet in the consumer cache - move it upwards MovePage(page); } } } status_t VMCache::AcquireUnreferencedStoreRef() { return B_OK; } void VMCache::AcquireStoreRef() { } void VMCache::ReleaseStoreRef() { } /*! Kernel debugger version of HasPage(). Does not do any locking. */ bool VMCache::DebugHasPage(off_t offset) { // default that works for all subclasses that don't lock anyway return HasPage(offset); } /*! Kernel debugger version of LookupPage(). Does not do any locking. */ vm_page* VMCache::DebugLookupPage(off_t offset) { return pages.Lookup((page_num_t)(offset >> PAGE_SHIFT)); } void VMCache::Dump(bool showPages) const { kprintf("CACHE %p:\n", this); kprintf(" ref_count: %" B_PRId32 "\n", RefCount()); kprintf(" source: %p\n", source); kprintf(" type: %s\n", vm_cache_type_to_string(type)); kprintf(" virtual_base: 0x%" B_PRIx64 "\n", virtual_base); kprintf(" virtual_end: 0x%" B_PRIx64 "\n", virtual_end); kprintf(" temporary: %" B_PRIu32 "\n", uint32(temporary)); kprintf(" lock: %p\n", &fLock); #if KDEBUG kprintf(" lock.holder: %" B_PRId32 "\n", fLock.holder); #endif kprintf(" areas:\n"); for (VMArea* area = areas; area != NULL; area = area->cache_next) { kprintf(" area 0x%" B_PRIx32 ", %s\n", area->id, area->name); kprintf("\tbase_addr: 0x%lx, size: 0x%lx\n", area->Base(), area->Size()); kprintf("\tprotection: 0x%" B_PRIx32 "\n", area->protection); kprintf("\towner: 0x%" B_PRIx32 "\n", area->address_space->ID()); } kprintf(" consumers:\n"); for (ConsumerList::ConstIterator it = consumers.GetIterator(); VMCache* consumer = it.Next();) { kprintf("\t%p\n", consumer); } kprintf(" pages:\n"); if (showPages) { for (VMCachePagesTree::ConstIterator it = pages.GetIterator(); vm_page* page = it.Next();) { if (!vm_page_is_dummy(page)) { kprintf("\t%p ppn %#" B_PRIxPHYSADDR " offset %#" B_PRIxPHYSADDR " state %u (%s) wired_count %u\n", page, page->physical_page_number, page->cache_offset, page->State(), page_state_to_string(page->State()), page->WiredCount()); } else { kprintf("\t%p DUMMY PAGE state %u (%s)\n", page, page->State(), page_state_to_string(page->State())); } } } else kprintf("\t%" B_PRIu32 " in cache\n", page_count); } /*! Wakes up threads waiting for page events. \param page The page for which events occurred. \param events The mask of events that occurred. */ void VMCache::_NotifyPageEvents(vm_page* page, uint32 events) { PageEventWaiter** it = &fPageEventWaiters; while (PageEventWaiter* waiter = *it) { if (waiter->page == page && (waiter->events & events) != 0) { // remove from list and unblock *it = waiter->next; thread_unblock(waiter->thread, B_OK); } else it = &waiter->next; } } /*! Merges the given cache with its only consumer. The caller must hold both the cache's and the consumer's lock. The method does release neither lock. */ void VMCache::_MergeWithOnlyConsumer() { VMCache* consumer = consumers.RemoveHead(); TRACE(("merge vm cache %p (ref == %" B_PRId32 ") with vm cache %p\n", this, this->fRefCount, consumer)); T(Merge(this, consumer)); // merge the cache consumer->Merge(this); // The remaining consumer has got a new source. if (source != NULL) { VMCache* newSource = source; newSource->Lock(); newSource->consumers.Remove(this); newSource->consumers.Add(consumer); consumer->source = newSource; source = NULL; newSource->Unlock(); } else consumer->source = NULL; // Release the reference the cache's consumer owned. The consumer takes // over the cache's ref to its source (if any) instead. ReleaseRefLocked(); } /*! Removes the \a consumer from this cache. It will also release the reference to the cache owned by the consumer. Assumes you have the consumer's cache lock held. This cache must not be locked. */ void VMCache::_RemoveConsumer(VMCache* consumer) { TRACE(("remove consumer vm cache %p from cache %p\n", consumer, this)); consumer->AssertLocked(); T(RemoveConsumer(this, consumer)); // Remove the store ref before locking the cache. Otherwise we'd call into // the VFS while holding the cache lock, which would reverse the usual // locking order. ReleaseStoreRef(); // remove the consumer from the cache, but keep its reference until later Lock(); consumers.Remove(consumer); consumer->source = NULL; ReleaseRefAndUnlock(); } // #pragma mark - VMCacheFactory // TODO: Move to own source file! /*static*/ status_t VMCacheFactory::CreateAnonymousCache(VMCache*& _cache, bool canOvercommit, int32 numPrecommittedPages, int32 numGuardPages, bool swappable, int priority) { uint32 allocationFlags = HEAP_DONT_WAIT_FOR_MEMORY | HEAP_DONT_LOCK_KERNEL_SPACE; if (priority >= VM_PRIORITY_VIP) allocationFlags |= HEAP_PRIORITY_VIP; #if ENABLE_SWAP_SUPPORT if (swappable) { VMAnonymousCache* cache = new(gAnonymousCacheObjectCache, allocationFlags) VMAnonymousCache; if (cache == NULL) return B_NO_MEMORY; status_t error = cache->Init(canOvercommit, numPrecommittedPages, numGuardPages, allocationFlags); if (error != B_OK) { cache->Delete(); return error; } T(Create(cache)); _cache = cache; return B_OK; } #endif VMAnonymousNoSwapCache* cache = new(gAnonymousNoSwapCacheObjectCache, allocationFlags) VMAnonymousNoSwapCache; if (cache == NULL) return B_NO_MEMORY; status_t error = cache->Init(canOvercommit, numPrecommittedPages, numGuardPages, allocationFlags); if (error != B_OK) { cache->Delete(); return error; } T(Create(cache)); _cache = cache; return B_OK; } /*static*/ status_t VMCacheFactory::CreateVnodeCache(VMCache*& _cache, struct vnode* vnode) { const uint32 allocationFlags = HEAP_DONT_WAIT_FOR_MEMORY | HEAP_DONT_LOCK_KERNEL_SPACE; // Note: Vnode cache creation is never VIP. VMVnodeCache* cache = new(gVnodeCacheObjectCache, allocationFlags) VMVnodeCache; if (cache == NULL) return B_NO_MEMORY; status_t error = cache->Init(vnode, allocationFlags); if (error != B_OK) { cache->Delete(); return error; } T(Create(cache)); _cache = cache; return B_OK; } /*static*/ status_t VMCacheFactory::CreateDeviceCache(VMCache*& _cache, addr_t baseAddress) { const uint32 allocationFlags = HEAP_DONT_WAIT_FOR_MEMORY | HEAP_DONT_LOCK_KERNEL_SPACE; // Note: Device cache creation is never VIP. VMDeviceCache* cache = new(gDeviceCacheObjectCache, allocationFlags) VMDeviceCache; if (cache == NULL) return B_NO_MEMORY; status_t error = cache->Init(baseAddress, allocationFlags); if (error != B_OK) { cache->Delete(); return error; } T(Create(cache)); _cache = cache; return B_OK; } /*static*/ status_t VMCacheFactory::CreateNullCache(int priority, VMCache*& _cache) { uint32 allocationFlags = HEAP_DONT_WAIT_FOR_MEMORY | HEAP_DONT_LOCK_KERNEL_SPACE; if (priority >= VM_PRIORITY_VIP) allocationFlags |= HEAP_PRIORITY_VIP; VMNullCache* cache = new(gNullCacheObjectCache, allocationFlags) VMNullCache; if (cache == NULL) return B_NO_MEMORY; status_t error = cache->Init(allocationFlags); if (error != B_OK) { cache->Delete(); return error; } T(Create(cache)); _cache = cache; return B_OK; }