The file cache code was completely ignoring the vm_page::modified flag. In

particular it wouldn't set the flag when writing something to a page, but
only move it to the modified queue. Since mapping the page would move it to
another queue, the information that the page was modified would be lost and
it would never be written to disk. Was well reproducible with a Haiku image
build and limited amount of memory.
Fixes the hopefully last remaining cause for #5374.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@35500 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Ingo Weinhold 2010-02-16 17:22:56 +00:00
parent 6949076049
commit 1f611c018b

View File

@ -309,7 +309,8 @@ reserve_pages(file_cache_ref* ref, vm_page_reservation* reservation,
vm_page* page; vm_page* page;
for (VMCachePagesTree::Iterator it = cache->pages.GetIterator(); for (VMCachePagesTree::Iterator it = cache->pages.GetIterator();
(page = it.Next()) != NULL && left > 0;) { (page = it.Next()) != NULL && left > 0;) {
if (page->state != PAGE_STATE_MODIFIED && !page->busy) { if (page->state != PAGE_STATE_MODIFIED && !page->modified
&& !page->busy) {
DEBUG_PAGE_ACCESS_START(page); DEBUG_PAGE_ACCESS_START(page);
cache->RemovePage(page); cache->RemovePage(page);
vm_page_set_state(page, PAGE_STATE_FREE); vm_page_set_state(page, PAGE_STATE_FREE);
@ -514,6 +515,8 @@ write_to_cache(file_cache_ref* ref, void* cookie, off_t offset,
(writeThrough ? PAGE_STATE_CACHED : PAGE_STATE_MODIFIED) (writeThrough ? PAGE_STATE_CACHED : PAGE_STATE_MODIFIED)
| VM_PAGE_ALLOC_BUSY); | VM_PAGE_ALLOC_BUSY);
page->modified = !writeThrough;
ref->cache->InsertPage(page, offset + pos); ref->cache->InsertPage(page, offset + pos);
add_to_iovec(vecs, vecCount, MAX_IO_VECS, add_to_iovec(vecs, vecCount, MAX_IO_VECS,
@ -807,9 +810,14 @@ cache_io(void* _cacheRef, void* cookie, off_t offset, addr_t buffer,
locker.Lock(); locker.Lock();
if (doWrite && page->state != PAGE_STATE_MODIFIED) { if (doWrite) {
DEBUG_PAGE_ACCESS_START(page); DEBUG_PAGE_ACCESS_START(page);
vm_page_set_state(page, PAGE_STATE_MODIFIED);
page->modified = true;
if (page->state != PAGE_STATE_MODIFIED)
vm_page_set_state(page, PAGE_STATE_MODIFIED);
DEBUG_PAGE_ACCESS_END(page); DEBUG_PAGE_ACCESS_END(page);
} }