aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorNeilBrown <neilb@suse.de>2012-05-21 23:55:15 -0400
committerNeilBrown <neilb@suse.de>2012-05-21 23:55:15 -0400
commitb405fe91e50c60c80e72d798025aea4917096421 (patch)
treebc0b8dc67e7bc7897c6299d4bc80e42a7842e7fe
parent84e923453e19d6427c6aa884d9561f01e1425d09 (diff)
md/bitmap: use set_bit, test_bit, etc for operation on bitmap->flags.
We currently use '&' and '|' which isn't the norm in the kernel and doesn't allow easy atomicity. So change to bit numbers and {set,clear,test}_bit. This allows us to remove a spinlock/unlock (which was dubious anyway) and some other simplifications. Signed-off-by: NeilBrown <neilb@suse.de>
-rw-r--r--drivers/md/bitmap.c46
-rw-r--r--drivers/md/bitmap.h6
2 files changed, 24 insertions, 28 deletions
diff --git a/drivers/md/bitmap.c b/drivers/md/bitmap.c
index 09d360ff77dc..f3a9dffc3fc8 100644
--- a/drivers/md/bitmap.c
+++ b/drivers/md/bitmap.c
@@ -271,7 +271,7 @@ static void write_page(struct bitmap *bitmap, struct page *page, int wait)
271 if (bitmap->storage.file == NULL) { 271 if (bitmap->storage.file == NULL) {
272 switch (write_sb_page(bitmap, page, wait)) { 272 switch (write_sb_page(bitmap, page, wait)) {
273 case -EINVAL: 273 case -EINVAL:
274 bitmap->flags |= BITMAP_WRITE_ERROR; 274 set_bit(BITMAP_WRITE_ERROR, &bitmap->flags);
275 } 275 }
276 } else { 276 } else {
277 277
@@ -289,20 +289,16 @@ static void write_page(struct bitmap *bitmap, struct page *page, int wait)
289 wait_event(bitmap->write_wait, 289 wait_event(bitmap->write_wait,
290 atomic_read(&bitmap->pending_writes)==0); 290 atomic_read(&bitmap->pending_writes)==0);
291 } 291 }
292 if (bitmap->flags & BITMAP_WRITE_ERROR) 292 if (test_bit(BITMAP_WRITE_ERROR, &bitmap->flags))
293 bitmap_file_kick(bitmap); 293 bitmap_file_kick(bitmap);
294} 294}
295 295
296static void end_bitmap_write(struct buffer_head *bh, int uptodate) 296static void end_bitmap_write(struct buffer_head *bh, int uptodate)
297{ 297{
298 struct bitmap *bitmap = bh->b_private; 298 struct bitmap *bitmap = bh->b_private;
299 unsigned long flags;
300 299
301 if (!uptodate) { 300 if (!uptodate)
302 spin_lock_irqsave(&bitmap->lock, flags); 301 set_bit(BITMAP_WRITE_ERROR, &bitmap->flags);
303 bitmap->flags |= BITMAP_WRITE_ERROR;
304 spin_unlock_irqrestore(&bitmap->lock, flags);
305 }
306 if (atomic_dec_and_test(&bitmap->pending_writes)) 302 if (atomic_dec_and_test(&bitmap->pending_writes))
307 wake_up(&bitmap->write_wait); 303 wake_up(&bitmap->write_wait);
308} 304}
@@ -389,7 +385,7 @@ static int read_page(struct file *file, unsigned long index,
389 385
390 wait_event(bitmap->write_wait, 386 wait_event(bitmap->write_wait,
391 atomic_read(&bitmap->pending_writes)==0); 387 atomic_read(&bitmap->pending_writes)==0);
392 if (bitmap->flags & BITMAP_WRITE_ERROR) 388 if (test_bit(BITMAP_WRITE_ERROR, &bitmap->flags))
393 ret = -EIO; 389 ret = -EIO;
394out: 390out:
395 if (ret) 391 if (ret)
@@ -521,7 +517,7 @@ static int bitmap_new_disk_sb(struct bitmap *bitmap)
521 517
522 memcpy(sb->uuid, bitmap->mddev->uuid, 16); 518 memcpy(sb->uuid, bitmap->mddev->uuid, 16);
523 519
524 bitmap->flags |= BITMAP_STALE; 520 set_bit(BITMAP_STALE, &bitmap->flags);
525 sb->state = cpu_to_le32(bitmap->flags); 521 sb->state = cpu_to_le32(bitmap->flags);
526 bitmap->events_cleared = bitmap->mddev->events; 522 bitmap->events_cleared = bitmap->mddev->events;
527 sb->events_cleared = cpu_to_le64(bitmap->mddev->events); 523 sb->events_cleared = cpu_to_le64(bitmap->mddev->events);
@@ -545,7 +541,7 @@ static int bitmap_read_sb(struct bitmap *bitmap)
545 chunksize = 128 * 1024 * 1024; 541 chunksize = 128 * 1024 * 1024;
546 daemon_sleep = 5 * HZ; 542 daemon_sleep = 5 * HZ;
547 write_behind = 0; 543 write_behind = 0;
548 bitmap->flags = BITMAP_STALE; 544 set_bit(BITMAP_STALE, &bitmap->flags);
549 err = 0; 545 err = 0;
550 goto out_no_sb; 546 goto out_no_sb;
551 } 547 }
@@ -617,20 +613,20 @@ static int bitmap_read_sb(struct bitmap *bitmap)
617 "-- forcing full recovery\n", 613 "-- forcing full recovery\n",
618 bmname(bitmap), events, 614 bmname(bitmap), events,
619 (unsigned long long) bitmap->mddev->events); 615 (unsigned long long) bitmap->mddev->events);
620 bitmap->flags |= BITMAP_STALE; 616 set_bit(BITMAP_STALE, &bitmap->flags);
621 } 617 }
622 } 618 }
623 619
624 /* assign fields using values from superblock */ 620 /* assign fields using values from superblock */
625 bitmap->flags |= le32_to_cpu(sb->state); 621 bitmap->flags |= le32_to_cpu(sb->state);
626 if (le32_to_cpu(sb->version) == BITMAP_MAJOR_HOSTENDIAN) 622 if (le32_to_cpu(sb->version) == BITMAP_MAJOR_HOSTENDIAN)
627 bitmap->flags |= BITMAP_HOSTENDIAN; 623 set_bit(BITMAP_HOSTENDIAN, &bitmap->flags);
628 bitmap->events_cleared = le64_to_cpu(sb->events_cleared); 624 bitmap->events_cleared = le64_to_cpu(sb->events_cleared);
629 err = 0; 625 err = 0;
630out: 626out:
631 kunmap_atomic(sb); 627 kunmap_atomic(sb);
632out_no_sb: 628out_no_sb:
633 if (bitmap->flags & BITMAP_STALE) 629 if (test_bit(BITMAP_STALE, &bitmap->flags))
634 bitmap->events_cleared = bitmap->mddev->events; 630 bitmap->events_cleared = bitmap->mddev->events;
635 bitmap->mddev->bitmap_info.chunksize = chunksize; 631 bitmap->mddev->bitmap_info.chunksize = chunksize;
636 bitmap->mddev->bitmap_info.daemon_sleep = daemon_sleep; 632 bitmap->mddev->bitmap_info.daemon_sleep = daemon_sleep;
@@ -796,8 +792,7 @@ static void bitmap_file_kick(struct bitmap *bitmap)
796{ 792{
797 char *path, *ptr = NULL; 793 char *path, *ptr = NULL;
798 794
799 if (!(bitmap->flags & BITMAP_STALE)) { 795 if (!test_and_set_bit(BITMAP_STALE, &bitmap->flags)) {
800 bitmap->flags |= BITMAP_STALE;
801 bitmap_update_sb(bitmap); 796 bitmap_update_sb(bitmap);
802 797
803 if (bitmap->storage.file) { 798 if (bitmap->storage.file) {
@@ -868,7 +863,7 @@ static void bitmap_file_set_bit(struct bitmap *bitmap, sector_t block)
868 863
869 /* set the bit */ 864 /* set the bit */
870 kaddr = kmap_atomic(page); 865 kaddr = kmap_atomic(page);
871 if (bitmap->flags & BITMAP_HOSTENDIAN) 866 if (test_bit(BITMAP_HOSTENDIAN, &bitmap->flags))
872 set_bit(bit, kaddr); 867 set_bit(bit, kaddr);
873 else 868 else
874 __set_bit_le(bit, kaddr); 869 __set_bit_le(bit, kaddr);
@@ -890,7 +885,7 @@ static void bitmap_file_clear_bit(struct bitmap *bitmap, sector_t block)
890 return; 885 return;
891 bit = file_page_offset(&bitmap->storage, chunk); 886 bit = file_page_offset(&bitmap->storage, chunk);
892 paddr = kmap_atomic(page); 887 paddr = kmap_atomic(page);
893 if (bitmap->flags & BITMAP_HOSTENDIAN) 888 if (test_bit(BITMAP_HOSTENDIAN, &bitmap->flags))
894 clear_bit(bit, paddr); 889 clear_bit(bit, paddr);
895 else 890 else
896 __clear_bit_le(bit, paddr); 891 __clear_bit_le(bit, paddr);
@@ -941,7 +936,7 @@ void bitmap_unplug(struct bitmap *bitmap)
941 else 936 else
942 md_super_wait(bitmap->mddev); 937 md_super_wait(bitmap->mddev);
943 } 938 }
944 if (bitmap->flags & BITMAP_WRITE_ERROR) 939 if (test_bit(BITMAP_WRITE_ERROR, &bitmap->flags))
945 bitmap_file_kick(bitmap); 940 bitmap_file_kick(bitmap);
946} 941}
947EXPORT_SYMBOL(bitmap_unplug); 942EXPORT_SYMBOL(bitmap_unplug);
@@ -988,7 +983,7 @@ static int bitmap_init_from_disk(struct bitmap *bitmap, sector_t start)
988 return 0; 983 return 0;
989 } 984 }
990 985
991 outofdate = bitmap->flags & BITMAP_STALE; 986 outofdate = test_bit(BITMAP_STALE, &bitmap->flags);
992 if (outofdate) 987 if (outofdate)
993 printk(KERN_INFO "%s: bitmap file is out of date, doing full " 988 printk(KERN_INFO "%s: bitmap file is out of date, doing full "
994 "recovery\n", bmname(bitmap)); 989 "recovery\n", bmname(bitmap));
@@ -1045,12 +1040,13 @@ static int bitmap_init_from_disk(struct bitmap *bitmap, sector_t start)
1045 write_page(bitmap, page, 1); 1040 write_page(bitmap, page, 1);
1046 1041
1047 ret = -EIO; 1042 ret = -EIO;
1048 if (bitmap->flags & BITMAP_WRITE_ERROR) 1043 if (test_bit(BITMAP_WRITE_ERROR,
1044 &bitmap->flags))
1049 goto err; 1045 goto err;
1050 } 1046 }
1051 } 1047 }
1052 paddr = kmap_atomic(page); 1048 paddr = kmap_atomic(page);
1053 if (bitmap->flags & BITMAP_HOSTENDIAN) 1049 if (test_bit(BITMAP_HOSTENDIAN, &bitmap->flags))
1054 b = test_bit(bit, paddr); 1050 b = test_bit(bit, paddr);
1055 else 1051 else
1056 b = test_bit_le(bit, paddr); 1052 b = test_bit_le(bit, paddr);
@@ -1758,7 +1754,7 @@ int bitmap_create(struct mddev *mddev)
1758 mddev->bitmap = bitmap; 1754 mddev->bitmap = bitmap;
1759 1755
1760 1756
1761 return (bitmap->flags & BITMAP_WRITE_ERROR) ? -EIO : 0; 1757 return test_bit(BITMAP_WRITE_ERROR, &bitmap->flags) ? -EIO : 0;
1762 1758
1763 error: 1759 error:
1764 bitmap_free(bitmap); 1760 bitmap_free(bitmap);
@@ -1799,7 +1795,7 @@ int bitmap_load(struct mddev *mddev)
1799 1795
1800 if (err) 1796 if (err)
1801 goto out; 1797 goto out;
1802 bitmap->flags &= ~BITMAP_STALE; 1798 clear_bit(BITMAP_STALE, &bitmap->flags);
1803 1799
1804 /* Kick recovery in case any bits were set */ 1800 /* Kick recovery in case any bits were set */
1805 set_bit(MD_RECOVERY_NEEDED, &bitmap->mddev->recovery); 1801 set_bit(MD_RECOVERY_NEEDED, &bitmap->mddev->recovery);
@@ -1809,7 +1805,7 @@ int bitmap_load(struct mddev *mddev)
1809 1805
1810 bitmap_update_sb(bitmap); 1806 bitmap_update_sb(bitmap);
1811 1807
1812 if (bitmap->flags & BITMAP_WRITE_ERROR) 1808 if (test_bit(BITMAP_WRITE_ERROR, &bitmap->flags))
1813 err = -EIO; 1809 err = -EIO;
1814out: 1810out:
1815 return err; 1811 return err;
diff --git a/drivers/md/bitmap.h b/drivers/md/bitmap.h
index 26689260e179..e2d999a62ba9 100644
--- a/drivers/md/bitmap.h
+++ b/drivers/md/bitmap.h
@@ -111,9 +111,9 @@ typedef __u16 bitmap_counter_t;
111 111
112/* use these for bitmap->flags and bitmap->sb->state bit-fields */ 112/* use these for bitmap->flags and bitmap->sb->state bit-fields */
113enum bitmap_state { 113enum bitmap_state {
114 BITMAP_STALE = 0x002, /* the bitmap file is out of date or had -EIO */ 114 BITMAP_STALE = 1, /* the bitmap file is out of date or had -EIO */
115 BITMAP_WRITE_ERROR = 0x004, /* A write error has occurred */ 115 BITMAP_WRITE_ERROR = 2, /* A write error has occurred */
116 BITMAP_HOSTENDIAN = 0x8000, 116 BITMAP_HOSTENDIAN =15,
117}; 117};
118 118
119/* the superblock at the front of the bitmap file -- little endian */ 119/* the superblock at the front of the bitmap file -- little endian */