* 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:
parent
a9ab9d2c21
commit
6a50382249
@ -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;
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user