Commit 88d37bdd authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] reduced locking in buffer.c

Replace the buffer lru spinlock protection with local_irq_disable and a
cross-CPU call to invalidate them.
parent 8e4f2fd3
...@@ -1274,15 +1274,32 @@ __bread_slow(struct block_device *bdev, sector_t block, int size) ...@@ -1274,15 +1274,32 @@ __bread_slow(struct block_device *bdev, sector_t block, int size)
* *
* This is a transparent caching front-end to sb_bread(), sb_getblk() and * This is a transparent caching front-end to sb_bread(), sb_getblk() and
* sb_find_get_block(). * sb_find_get_block().
*
* The LRUs themselves only need locking against invalidate_bh_lrus. We use
* a local interrupt disable for that.
*/ */
#define BH_LRU_SIZE 7 #define BH_LRU_SIZE 8
static struct bh_lru { static struct bh_lru {
spinlock_t lock;
struct buffer_head *bhs[BH_LRU_SIZE]; struct buffer_head *bhs[BH_LRU_SIZE];
} ____cacheline_aligned_in_smp bh_lrus[NR_CPUS]; } ____cacheline_aligned_in_smp bh_lrus[NR_CPUS];
#ifdef CONFIG_SMP
#define bh_lru_lock() local_irq_disable()
#define bh_lru_unlock() local_irq_enable()
#else
#define bh_lru_lock() preempt_disable()
#define bh_lru_unlock() preempt_enable()
#endif
static inline void check_irqs_on(void)
{
#ifdef irqs_disabled
BUG_ON(irqs_disabled());
#endif
}
/* /*
* The LRU management algorithm is dopey-but-simple. Sorry. * The LRU management algorithm is dopey-but-simple. Sorry.
*/ */
...@@ -1294,8 +1311,9 @@ static void bh_lru_install(struct buffer_head *bh) ...@@ -1294,8 +1311,9 @@ static void bh_lru_install(struct buffer_head *bh)
if (bh == NULL) if (bh == NULL)
return; return;
lru = &bh_lrus[get_cpu()]; check_irqs_on();
spin_lock(&lru->lock); bh_lru_lock();
lru = &bh_lrus[smp_processor_id()];
if (lru->bhs[0] != bh) { if (lru->bhs[0] != bh) {
struct buffer_head *bhs[BH_LRU_SIZE]; struct buffer_head *bhs[BH_LRU_SIZE];
int in; int in;
...@@ -1321,8 +1339,7 @@ static void bh_lru_install(struct buffer_head *bh) ...@@ -1321,8 +1339,7 @@ static void bh_lru_install(struct buffer_head *bh)
bhs[out++] = NULL; bhs[out++] = NULL;
memcpy(lru->bhs, bhs, sizeof(bhs)); memcpy(lru->bhs, bhs, sizeof(bhs));
} }
spin_unlock(&lru->lock); bh_lru_unlock();
put_cpu();
if (evictee) { if (evictee) {
touch_buffer(evictee); touch_buffer(evictee);
...@@ -1337,8 +1354,9 @@ lookup_bh(struct block_device *bdev, sector_t block, int size) ...@@ -1337,8 +1354,9 @@ lookup_bh(struct block_device *bdev, sector_t block, int size)
struct bh_lru *lru; struct bh_lru *lru;
int i; int i;
lru = &bh_lrus[get_cpu()]; check_irqs_on();
spin_lock(&lru->lock); bh_lru_lock();
lru = &bh_lrus[smp_processor_id()];
for (i = 0; i < BH_LRU_SIZE; i++) { for (i = 0; i < BH_LRU_SIZE; i++) {
struct buffer_head *bh = lru->bhs[i]; struct buffer_head *bh = lru->bhs[i];
...@@ -1356,8 +1374,7 @@ lookup_bh(struct block_device *bdev, sector_t block, int size) ...@@ -1356,8 +1374,7 @@ lookup_bh(struct block_device *bdev, sector_t block, int size)
break; break;
} }
} }
spin_unlock(&lru->lock); bh_lru_unlock();
put_cpu();
return ret; return ret;
} }
...@@ -1404,26 +1421,33 @@ __bread(struct block_device *bdev, sector_t block, int size) ...@@ -1404,26 +1421,33 @@ __bread(struct block_device *bdev, sector_t block, int size)
EXPORT_SYMBOL(__bread); EXPORT_SYMBOL(__bread);
/* /*
* This is called rarely - at unmount. * invalidate_bh_lrus() is called rarely - at unmount. Because it is only for
* unmount it only needs to ensure that all buffers from the target device are
* invalidated on return and it doesn't need to worry about new buffers from
* that device being added - the unmount code has to prevent that.
*/ */
static void invalidate_bh_lrus(void) static void invalidate_bh_lru(void *arg)
{ {
int cpu_idx; const int cpu = get_cpu();
for (cpu_idx = 0; cpu_idx < NR_CPUS; cpu_idx++)
spin_lock(&bh_lrus[cpu_idx].lock);
for (cpu_idx = 0; cpu_idx < NR_CPUS; cpu_idx++) {
int i; int i;
for (i = 0; i < BH_LRU_SIZE; i++) { for (i = 0; i < BH_LRU_SIZE; i++) {
brelse(bh_lrus[cpu_idx].bhs[i]); brelse(bh_lrus[cpu].bhs[i]);
bh_lrus[cpu_idx].bhs[i] = NULL; bh_lrus[cpu].bhs[i] = NULL;
}
} }
for (cpu_idx = 0; cpu_idx < NR_CPUS; cpu_idx++) put_cpu();
spin_unlock(&bh_lrus[cpu_idx].lock); }
static void invalidate_bh_lrus(void)
{
preempt_disable();
invalidate_bh_lru(NULL);
smp_call_function(invalidate_bh_lru, NULL, 1, 1);
preempt_enable();
} }
void set_bh_page(struct buffer_head *bh, void set_bh_page(struct buffer_head *bh,
struct page *page, unsigned long offset) struct page *page, unsigned long offset)
{ {
...@@ -2545,9 +2569,6 @@ void __init buffer_init(void) ...@@ -2545,9 +2569,6 @@ void __init buffer_init(void)
{ {
int i; int i;
for (i = 0; i < NR_CPUS; i++)
spin_lock_init(&bh_lrus[i].lock);
bh_cachep = kmem_cache_create("buffer_head", bh_cachep = kmem_cache_create("buffer_head",
sizeof(struct buffer_head), 0, sizeof(struct buffer_head), 0,
SLAB_HWCACHE_ALIGN, init_buffer_head, NULL); SLAB_HWCACHE_ALIGN, init_buffer_head, NULL);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment