diff options
Diffstat (limited to 'fs/fscache/cache.c')
| -rw-r--r-- | fs/fscache/cache.c | 34 | 
1 files changed, 18 insertions, 16 deletions
| diff --git a/fs/fscache/cache.c b/fs/fscache/cache.c index b52aed1dca97..f7cff367db7f 100644 --- a/fs/fscache/cache.c +++ b/fs/fscache/cache.c @@ -115,7 +115,7 @@ struct fscache_cache *fscache_select_cache_for_object(  				     struct fscache_object, cookie_link);  		cache = object->cache; -		if (object->state >= FSCACHE_OBJECT_DYING || +		if (fscache_object_is_dying(object) ||  		    test_bit(FSCACHE_IOERROR, &cache->flags))  			cache = NULL; @@ -224,8 +224,10 @@ int fscache_add_cache(struct fscache_cache *cache,  	BUG_ON(!ifsdef);  	cache->flags = 0; -	ifsdef->event_mask = ULONG_MAX & ~(1 << FSCACHE_OBJECT_EV_CLEARED); -	ifsdef->state = FSCACHE_OBJECT_ACTIVE; +	ifsdef->event_mask = +		((1 << NR_FSCACHE_OBJECT_EVENTS) - 1) & +		~(1 << FSCACHE_OBJECT_EV_CLEARED); +	__set_bit(FSCACHE_OBJECT_IS_AVAILABLE, &ifsdef->flags);  	if (!tagname)  		tagname = cache->identifier; @@ -330,25 +332,25 @@ static void fscache_withdraw_all_objects(struct fscache_cache *cache,  {  	struct fscache_object *object; -	spin_lock(&cache->object_list_lock); -  	while (!list_empty(&cache->object_list)) { -		object = list_entry(cache->object_list.next, -				    struct fscache_object, cache_link); -		list_move_tail(&object->cache_link, dying_objects); +		spin_lock(&cache->object_list_lock); -		_debug("withdraw %p", object->cookie); +		if (!list_empty(&cache->object_list)) { +			object = list_entry(cache->object_list.next, +					    struct fscache_object, cache_link); +			list_move_tail(&object->cache_link, dying_objects); -		spin_lock(&object->lock); -		spin_unlock(&cache->object_list_lock); -		fscache_raise_event(object, FSCACHE_OBJECT_EV_WITHDRAW); -		spin_unlock(&object->lock); +			_debug("withdraw %p", object->cookie); + +			/* This must be done under object_list_lock to prevent +			 * a race with fscache_drop_object(). +			 */ +			fscache_raise_event(object, FSCACHE_OBJECT_EV_KILL); +		} +		spin_unlock(&cache->object_list_lock);  		cond_resched(); -		spin_lock(&cache->object_list_lock);  	} - -	spin_unlock(&cache->object_list_lock);  }  /** | 
