* Mixed consumer with consumerRef to identify the cache in the consumer list;

this fixes bug #227 again (which I recently opened again accidently).
* We actually switched the last consumer's source without having acquired its
  lock! This fixes some rare random app crashes as well as potential kernel
  crash ("cache to be deleted still has consumers").
* Some more comments to explain why things are done and can be done the way they
  are done :-)


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@19878 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2007-01-20 12:49:44 +00:00
parent a9ab9d2c21
commit 6a50382249
2 changed files with 31 additions and 22 deletions

View File

@ -92,7 +92,7 @@ typedef struct vm_area {
uint32 protection; uint32 protection;
uint16 wiring; uint16 wiring;
uint16 memory_type; uint16 memory_type;
int32 ref_count; vint32 ref_count;
struct vm_cache_ref *cache_ref; struct vm_cache_ref *cache_ref;
off_t cache_offset; off_t cache_offset;

View File

@ -407,12 +407,12 @@ vm_cache_resize(vm_cache_ref *cacheRef, off_t newSize)
void void
vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer) vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
{ {
vm_cache *newSource = NULL;
vm_cache *cache; vm_cache *cache;
TRACE(("remove consumer vm cache %p from cache %p\n", consumer, cacheRef->cache)); TRACE(("remove consumer vm cache %p from cache %p\n", consumer, cacheRef->cache));
ASSERT_LOCKED_MUTEX(&consumer->ref->lock); ASSERT_LOCKED_MUTEX(&consumer->ref->lock);
// remove the consumer from the cache, but keep its reference until later
mutex_lock(&cacheRef->lock); mutex_lock(&cacheRef->lock);
cache = cacheRef->cache; cache = cacheRef->cache;
list_remove_item(&cache->consumers, consumer); list_remove_item(&cache->consumers, consumer);
@ -427,17 +427,22 @@ vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
bool merge = false; bool merge = false;
consumer = list_get_first_item(&cache->consumers); consumer = list_get_first_item(&cache->consumers);
consumerRef = consumer->ref;
// Our cache doesn't have a ref to its consumer (only the other way around), // 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 // so we cannot just acquire it here; it might be deleted right now
while (true) { while (true) {
int32 count = consumerRef->ref_count; int32 count;
consumerRef = consumer->ref;
count = consumerRef->ref_count;
if (count == 0) if (count == 0)
break; break;
if (atomic_test_and_set(&consumerRef->ref_count, count + 1, count) == count) { if (atomic_test_and_set(&consumerRef->ref_count, count + 1, count) == count) {
// we managed to grab a reference to the consumer // 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.
merge = true; merge = true;
break; break;
} }
@ -453,10 +458,12 @@ vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
// the cache and the situation might have changed // the cache and the situation might have changed
cache = cacheRef->cache; cache = cacheRef->cache;
consumer = consumerRef->cache;
if (cacheRef->areas != NULL || cache->source == NULL if (cacheRef->areas != NULL || cache->source == NULL
|| list_is_empty(&cache->consumers) || list_is_empty(&cache->consumers)
|| cache->consumers.link.next != cache->consumers.link.prev || cache->consumers.link.next != cache->consumers.link.prev
|| consumerRef != list_get_first_item(&cache->consumers)) { || consumer != list_get_first_item(&cache->consumers)) {
merge = false; merge = false;
mutex_unlock(&consumerRef->lock); mutex_unlock(&consumerRef->lock);
} }
@ -464,6 +471,7 @@ vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
if (merge) { if (merge) {
vm_page *page, *nextPage; vm_page *page, *nextPage;
vm_cache *newSource;
consumer = list_remove_head_item(&cache->consumers); consumer = list_remove_head_item(&cache->consumers);
@ -488,27 +496,28 @@ vm_cache_remove_consumer(vm_cache_ref *cacheRef, vm_cache *consumer)
} }
newSource = cache->source; newSource = cache->source;
if (newSource != NULL) {
// 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
vm_cache_release_ref(cacheRef);
}
mutex_unlock(&consumerRef->lock); mutex_unlock(&consumerRef->lock);
} }
vm_cache_release_ref(consumerRef); vm_cache_release_ref(consumerRef);
} }
if (newSource != NULL) {
// 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
vm_cache_release_ref(cacheRef);
}
mutex_unlock(&cacheRef->lock); mutex_unlock(&cacheRef->lock);
vm_cache_release_ref(cacheRef); vm_cache_release_ref(cacheRef);
} }