aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/md/md.c35
-rw-r--r--include/linux/raid/md_k.h2
2 files changed, 23 insertions, 14 deletions
diff --git a/drivers/md/md.c b/drivers/md/md.c
index 3d5a6beb333..bae42331182 100644
--- a/drivers/md/md.c
+++ b/drivers/md/md.c
@@ -5091,7 +5091,7 @@ static int is_mddev_idle(mddev_t *mddev)
5091 mdk_rdev_t * rdev; 5091 mdk_rdev_t * rdev;
5092 struct list_head *tmp; 5092 struct list_head *tmp;
5093 int idle; 5093 int idle;
5094 unsigned long curr_events; 5094 long curr_events;
5095 5095
5096 idle = 1; 5096 idle = 1;
5097 ITERATE_RDEV(mddev,rdev,tmp) { 5097 ITERATE_RDEV(mddev,rdev,tmp) {
@@ -5099,20 +5099,29 @@ static int is_mddev_idle(mddev_t *mddev)
5099 curr_events = disk_stat_read(disk, sectors[0]) + 5099 curr_events = disk_stat_read(disk, sectors[0]) +
5100 disk_stat_read(disk, sectors[1]) - 5100 disk_stat_read(disk, sectors[1]) -
5101 atomic_read(&disk->sync_io); 5101 atomic_read(&disk->sync_io);
5102 /* The difference between curr_events and last_events 5102 /* sync IO will cause sync_io to increase before the disk_stats
5103 * will be affected by any new non-sync IO (making 5103 * as sync_io is counted when a request starts, and
5104 * curr_events bigger) and any difference in the amount of 5104 * disk_stats is counted when it completes.
5105 * in-flight syncio (making current_events bigger or smaller) 5105 * So resync activity will cause curr_events to be smaller than
5106 * The amount in-flight is currently limited to 5106 * when there was no such activity.
5107 * 32*64K in raid1/10 and 256*PAGE_SIZE in raid5/6 5107 * non-sync IO will cause disk_stat to increase without
5108 * which is at most 4096 sectors. 5108 * increasing sync_io so curr_events will (eventually)
5109 * These numbers are fairly fragile and should be made 5109 * be larger than it was before. Once it becomes
5110 * more robust, probably by enforcing the 5110 * substantially larger, the test below will cause
5111 * 'window size' that md_do_sync sort-of uses. 5111 * the array to appear non-idle, and resync will slow
5112 * down.
5113 * If there is a lot of outstanding resync activity when
5114 * we set last_event to curr_events, then all that activity
5115 * completing might cause the array to appear non-idle
5116 * and resync will be slowed down even though there might
5117 * not have been non-resync activity. This will only
5118 * happen once though. 'last_events' will soon reflect
5119 * the state where there is little or no outstanding
5120 * resync requests, and further resync activity will
5121 * always make curr_events less than last_events.
5112 * 5122 *
5113 * Note: the following is an unsigned comparison.
5114 */ 5123 */
5115 if ((long)curr_events - (long)rdev->last_events > 4096) { 5124 if (curr_events - rdev->last_events > 4096) {
5116 rdev->last_events = curr_events; 5125 rdev->last_events = curr_events;
5117 idle = 0; 5126 idle = 0;
5118 } 5127 }
diff --git a/include/linux/raid/md_k.h b/include/linux/raid/md_k.h
index de72c49747c..28ac632b42d 100644
--- a/include/linux/raid/md_k.h
+++ b/include/linux/raid/md_k.h
@@ -51,7 +51,7 @@ struct mdk_rdev_s
51 51
52 sector_t size; /* Device size (in blocks) */ 52 sector_t size; /* Device size (in blocks) */
53 mddev_t *mddev; /* RAID array if running */ 53 mddev_t *mddev; /* RAID array if running */
54 unsigned long last_events; /* IO event timestamp */ 54 long last_events; /* IO event timestamp */
55 55
56 struct block_device *bdev; /* block device handle */ 56 struct block_device *bdev; /* block device handle */
57 57