diff options
author | Chandra Seetharaman <sekharan@us.ibm.com> | 2008-05-01 17:50:11 -0400 |
---|---|---|
committer | James Bottomley <James.Bottomley@HansenPartnership.com> | 2008-06-05 10:23:41 -0400 |
commit | cfae5c9bb66325cd32d5f2ee41f14749f062a53c (patch) | |
tree | abceb01b7053f77366b37331e9b4f8408c89df60 /drivers | |
parent | 5e7dccad3621f6e2b572f309cf830a2c902cae80 (diff) |
[SCSI] scsi_dh: Use SCSI device handler in dm-multipath
This patch converts dm-mpath to use scsi device handlers instead of
dm's hardware handlers.
This patch does not add any new functionality. Old behaviors remain and
userspace tools work as is except that arguments supplied with hardware
handler are ignored.
One behavioral exception is: Activation of a path is synchronous in this
patch, opposed to the older behavior of being asynchronous (changed in
patch 07: scsi_dh: Add a single threaded workqueue for initializing a path)
Note: There is no need to get a reference for the device handler module
(as it was done in the dm hardware handler case) here as the reference
is held when the device was first found. Instead we check and make sure
that support for the specified device is present at table load time.
Signed-off-by: Chandra Seetharaman <sekharan@us.ibm.com>
Signed-off-by: Mike Christie <michaelc@cs.wisc.edu>
Acked-by: Alasdair G Kergon <agk@redhat.com>
Signed-off-by: James Bottomley <James.Bottomley@HansenPartnership.com>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/md/Kconfig | 1 | ||||
-rw-r--r-- | drivers/md/dm-mpath.c | 131 |
2 files changed, 81 insertions, 51 deletions
diff --git a/drivers/md/Kconfig b/drivers/md/Kconfig index 610af916891e..5303af55d2c7 100644 --- a/drivers/md/Kconfig +++ b/drivers/md/Kconfig | |||
@@ -252,6 +252,7 @@ config DM_ZERO | |||
252 | config DM_MULTIPATH | 252 | config DM_MULTIPATH |
253 | tristate "Multipath target" | 253 | tristate "Multipath target" |
254 | depends on BLK_DEV_DM | 254 | depends on BLK_DEV_DM |
255 | select SCSI_DH | ||
255 | ---help--- | 256 | ---help--- |
256 | Allow volume managers to support multipath hardware. | 257 | Allow volume managers to support multipath hardware. |
257 | 258 | ||
diff --git a/drivers/md/dm-mpath.c b/drivers/md/dm-mpath.c index e7ee59e655d5..e54ff372d711 100644 --- a/drivers/md/dm-mpath.c +++ b/drivers/md/dm-mpath.c | |||
@@ -20,6 +20,7 @@ | |||
20 | #include <linux/slab.h> | 20 | #include <linux/slab.h> |
21 | #include <linux/time.h> | 21 | #include <linux/time.h> |
22 | #include <linux/workqueue.h> | 22 | #include <linux/workqueue.h> |
23 | #include <scsi/scsi_dh.h> | ||
23 | #include <asm/atomic.h> | 24 | #include <asm/atomic.h> |
24 | 25 | ||
25 | #define DM_MSG_PREFIX "multipath" | 26 | #define DM_MSG_PREFIX "multipath" |
@@ -61,7 +62,7 @@ struct multipath { | |||
61 | 62 | ||
62 | spinlock_t lock; | 63 | spinlock_t lock; |
63 | 64 | ||
64 | struct hw_handler hw_handler; | 65 | const char *hw_handler_name; |
65 | unsigned nr_priority_groups; | 66 | unsigned nr_priority_groups; |
66 | struct list_head priority_groups; | 67 | struct list_head priority_groups; |
67 | unsigned pg_init_required; /* pg_init needs calling? */ | 68 | unsigned pg_init_required; /* pg_init needs calling? */ |
@@ -109,6 +110,7 @@ static struct kmem_cache *_mpio_cache; | |||
109 | static struct workqueue_struct *kmultipathd; | 110 | static struct workqueue_struct *kmultipathd; |
110 | static void process_queued_ios(struct work_struct *work); | 111 | static void process_queued_ios(struct work_struct *work); |
111 | static void trigger_event(struct work_struct *work); | 112 | static void trigger_event(struct work_struct *work); |
113 | static void pg_init_done(struct dm_path *, int); | ||
112 | 114 | ||
113 | 115 | ||
114 | /*----------------------------------------------- | 116 | /*----------------------------------------------- |
@@ -193,18 +195,13 @@ static struct multipath *alloc_multipath(struct dm_target *ti) | |||
193 | static void free_multipath(struct multipath *m) | 195 | static void free_multipath(struct multipath *m) |
194 | { | 196 | { |
195 | struct priority_group *pg, *tmp; | 197 | struct priority_group *pg, *tmp; |
196 | struct hw_handler *hwh = &m->hw_handler; | ||
197 | 198 | ||
198 | list_for_each_entry_safe(pg, tmp, &m->priority_groups, list) { | 199 | list_for_each_entry_safe(pg, tmp, &m->priority_groups, list) { |
199 | list_del(&pg->list); | 200 | list_del(&pg->list); |
200 | free_priority_group(pg, m->ti); | 201 | free_priority_group(pg, m->ti); |
201 | } | 202 | } |
202 | 203 | ||
203 | if (hwh->type) { | 204 | kfree(m->hw_handler_name); |
204 | hwh->type->destroy(hwh); | ||
205 | dm_put_hw_handler(hwh->type); | ||
206 | } | ||
207 | |||
208 | mempool_destroy(m->mpio_pool); | 205 | mempool_destroy(m->mpio_pool); |
209 | kfree(m); | 206 | kfree(m); |
210 | } | 207 | } |
@@ -216,12 +213,10 @@ static void free_multipath(struct multipath *m) | |||
216 | 213 | ||
217 | static void __switch_pg(struct multipath *m, struct pgpath *pgpath) | 214 | static void __switch_pg(struct multipath *m, struct pgpath *pgpath) |
218 | { | 215 | { |
219 | struct hw_handler *hwh = &m->hw_handler; | ||
220 | |||
221 | m->current_pg = pgpath->pg; | 216 | m->current_pg = pgpath->pg; |
222 | 217 | ||
223 | /* Must we initialise the PG first, and queue I/O till it's ready? */ | 218 | /* Must we initialise the PG first, and queue I/O till it's ready? */ |
224 | if (hwh->type && hwh->type->pg_init) { | 219 | if (m->hw_handler_name) { |
225 | m->pg_init_required = 1; | 220 | m->pg_init_required = 1; |
226 | m->queue_io = 1; | 221 | m->queue_io = 1; |
227 | } else { | 222 | } else { |
@@ -409,7 +404,6 @@ static void process_queued_ios(struct work_struct *work) | |||
409 | { | 404 | { |
410 | struct multipath *m = | 405 | struct multipath *m = |
411 | container_of(work, struct multipath, process_queued_ios); | 406 | container_of(work, struct multipath, process_queued_ios); |
412 | struct hw_handler *hwh = &m->hw_handler; | ||
413 | struct pgpath *pgpath = NULL; | 407 | struct pgpath *pgpath = NULL; |
414 | unsigned init_required = 0, must_queue = 1; | 408 | unsigned init_required = 0, must_queue = 1; |
415 | unsigned long flags; | 409 | unsigned long flags; |
@@ -438,8 +432,11 @@ static void process_queued_ios(struct work_struct *work) | |||
438 | out: | 432 | out: |
439 | spin_unlock_irqrestore(&m->lock, flags); | 433 | spin_unlock_irqrestore(&m->lock, flags); |
440 | 434 | ||
441 | if (init_required) | 435 | if (init_required) { |
442 | hwh->type->pg_init(hwh, pgpath->pg->bypassed, &pgpath->path); | 436 | struct dm_path *path = &pgpath->path; |
437 | int ret = scsi_dh_activate(bdev_get_queue(path->dev->bdev)); | ||
438 | pg_init_done(path, ret); | ||
439 | } | ||
443 | 440 | ||
444 | if (!must_queue) | 441 | if (!must_queue) |
445 | dispatch_queued_ios(m); | 442 | dispatch_queued_ios(m); |
@@ -652,8 +649,6 @@ static struct priority_group *parse_priority_group(struct arg_set *as, | |||
652 | 649 | ||
653 | static int parse_hw_handler(struct arg_set *as, struct multipath *m) | 650 | static int parse_hw_handler(struct arg_set *as, struct multipath *m) |
654 | { | 651 | { |
655 | int r; | ||
656 | struct hw_handler_type *hwht; | ||
657 | unsigned hw_argc; | 652 | unsigned hw_argc; |
658 | struct dm_target *ti = m->ti; | 653 | struct dm_target *ti = m->ti; |
659 | 654 | ||
@@ -661,30 +656,18 @@ static int parse_hw_handler(struct arg_set *as, struct multipath *m) | |||
661 | {0, 1024, "invalid number of hardware handler args"}, | 656 | {0, 1024, "invalid number of hardware handler args"}, |
662 | }; | 657 | }; |
663 | 658 | ||
664 | r = read_param(_params, shift(as), &hw_argc, &ti->error); | 659 | if (read_param(_params, shift(as), &hw_argc, &ti->error)) |
665 | if (r) | ||
666 | return -EINVAL; | 660 | return -EINVAL; |
667 | 661 | ||
668 | if (!hw_argc) | 662 | if (!hw_argc) |
669 | return 0; | 663 | return 0; |
670 | 664 | ||
671 | hwht = dm_get_hw_handler(shift(as)); | 665 | m->hw_handler_name = kstrdup(shift(as), GFP_KERNEL); |
672 | if (!hwht) { | 666 | request_module("scsi_dh_%s", m->hw_handler_name); |
667 | if (scsi_dh_handler_exist(m->hw_handler_name) == 0) { | ||
673 | ti->error = "unknown hardware handler type"; | 668 | ti->error = "unknown hardware handler type"; |
674 | return -EINVAL; | 669 | return -EINVAL; |
675 | } | 670 | } |
676 | |||
677 | m->hw_handler.md = dm_table_get_md(ti->table); | ||
678 | dm_put(m->hw_handler.md); | ||
679 | |||
680 | r = hwht->create(&m->hw_handler, hw_argc - 1, as->argv); | ||
681 | if (r) { | ||
682 | dm_put_hw_handler(hwht); | ||
683 | ti->error = "hardware handler constructor failed"; | ||
684 | return r; | ||
685 | } | ||
686 | |||
687 | m->hw_handler.type = hwht; | ||
688 | consume(as, hw_argc - 1); | 671 | consume(as, hw_argc - 1); |
689 | 672 | ||
690 | return 0; | 673 | return 0; |
@@ -1063,14 +1046,74 @@ void dm_pg_init_complete(struct dm_path *path, unsigned err_flags) | |||
1063 | spin_unlock_irqrestore(&m->lock, flags); | 1046 | spin_unlock_irqrestore(&m->lock, flags); |
1064 | } | 1047 | } |
1065 | 1048 | ||
1049 | static void pg_init_done(struct dm_path *path, int errors) | ||
1050 | { | ||
1051 | struct pgpath *pgpath = path_to_pgpath(path); | ||
1052 | struct priority_group *pg = pgpath->pg; | ||
1053 | struct multipath *m = pg->m; | ||
1054 | unsigned long flags; | ||
1055 | |||
1056 | /* device or driver problems */ | ||
1057 | switch (errors) { | ||
1058 | case SCSI_DH_OK: | ||
1059 | break; | ||
1060 | case SCSI_DH_NOSYS: | ||
1061 | if (!m->hw_handler_name) { | ||
1062 | errors = 0; | ||
1063 | break; | ||
1064 | } | ||
1065 | DMERR("Cannot failover device because scsi_dh_%s was not " | ||
1066 | "loaded.", m->hw_handler_name); | ||
1067 | /* | ||
1068 | * Fail path for now, so we do not ping pong | ||
1069 | */ | ||
1070 | fail_path(pgpath); | ||
1071 | break; | ||
1072 | case SCSI_DH_DEV_TEMP_BUSY: | ||
1073 | /* | ||
1074 | * Probably doing something like FW upgrade on the | ||
1075 | * controller so try the other pg. | ||
1076 | */ | ||
1077 | bypass_pg(m, pg, 1); | ||
1078 | break; | ||
1079 | /* TODO: For SCSI_DH_RETRY we should wait a couple seconds */ | ||
1080 | case SCSI_DH_RETRY: | ||
1081 | case SCSI_DH_IMM_RETRY: | ||
1082 | case SCSI_DH_RES_TEMP_UNAVAIL: | ||
1083 | if (pg_init_limit_reached(m, pgpath)) | ||
1084 | fail_path(pgpath); | ||
1085 | errors = 0; | ||
1086 | break; | ||
1087 | default: | ||
1088 | /* | ||
1089 | * We probably do not want to fail the path for a device | ||
1090 | * error, but this is what the old dm did. In future | ||
1091 | * patches we can do more advanced handling. | ||
1092 | */ | ||
1093 | fail_path(pgpath); | ||
1094 | } | ||
1095 | |||
1096 | spin_lock_irqsave(&m->lock, flags); | ||
1097 | if (errors) { | ||
1098 | DMERR("Could not failover device. Error %d.", errors); | ||
1099 | m->current_pgpath = NULL; | ||
1100 | m->current_pg = NULL; | ||
1101 | } else if (!m->pg_init_required) { | ||
1102 | m->queue_io = 0; | ||
1103 | pg->bypassed = 0; | ||
1104 | } | ||
1105 | |||
1106 | m->pg_init_in_progress = 0; | ||
1107 | queue_work(kmultipathd, &m->process_queued_ios); | ||
1108 | spin_unlock_irqrestore(&m->lock, flags); | ||
1109 | } | ||
1110 | |||
1066 | /* | 1111 | /* |
1067 | * end_io handling | 1112 | * end_io handling |
1068 | */ | 1113 | */ |
1069 | static int do_end_io(struct multipath *m, struct bio *bio, | 1114 | static int do_end_io(struct multipath *m, struct bio *bio, |
1070 | int error, struct dm_mpath_io *mpio) | 1115 | int error, struct dm_mpath_io *mpio) |
1071 | { | 1116 | { |
1072 | struct hw_handler *hwh = &m->hw_handler; | ||
1073 | unsigned err_flags = MP_FAIL_PATH; /* Default behavior */ | ||
1074 | unsigned long flags; | 1117 | unsigned long flags; |
1075 | 1118 | ||
1076 | if (!error) | 1119 | if (!error) |
@@ -1097,19 +1140,8 @@ static int do_end_io(struct multipath *m, struct bio *bio, | |||
1097 | } | 1140 | } |
1098 | spin_unlock_irqrestore(&m->lock, flags); | 1141 | spin_unlock_irqrestore(&m->lock, flags); |
1099 | 1142 | ||
1100 | if (hwh->type && hwh->type->error) | 1143 | if (mpio->pgpath) |
1101 | err_flags = hwh->type->error(hwh, bio); | 1144 | fail_path(mpio->pgpath); |
1102 | |||
1103 | if (mpio->pgpath) { | ||
1104 | if (err_flags & MP_FAIL_PATH) | ||
1105 | fail_path(mpio->pgpath); | ||
1106 | |||
1107 | if (err_flags & MP_BYPASS_PG) | ||
1108 | bypass_pg(m, mpio->pgpath->pg, 1); | ||
1109 | } | ||
1110 | |||
1111 | if (err_flags & MP_ERROR_IO) | ||
1112 | return -EIO; | ||
1113 | 1145 | ||
1114 | requeue: | 1146 | requeue: |
1115 | dm_bio_restore(&mpio->details, bio); | 1147 | dm_bio_restore(&mpio->details, bio); |
@@ -1194,7 +1226,6 @@ static int multipath_status(struct dm_target *ti, status_type_t type, | |||
1194 | int sz = 0; | 1226 | int sz = 0; |
1195 | unsigned long flags; | 1227 | unsigned long flags; |
1196 | struct multipath *m = (struct multipath *) ti->private; | 1228 | struct multipath *m = (struct multipath *) ti->private; |
1197 | struct hw_handler *hwh = &m->hw_handler; | ||
1198 | struct priority_group *pg; | 1229 | struct priority_group *pg; |
1199 | struct pgpath *p; | 1230 | struct pgpath *p; |
1200 | unsigned pg_num; | 1231 | unsigned pg_num; |
@@ -1214,12 +1245,10 @@ static int multipath_status(struct dm_target *ti, status_type_t type, | |||
1214 | DMEMIT("pg_init_retries %u ", m->pg_init_retries); | 1245 | DMEMIT("pg_init_retries %u ", m->pg_init_retries); |
1215 | } | 1246 | } |
1216 | 1247 | ||
1217 | if (hwh->type && hwh->type->status) | 1248 | if (!m->hw_handler_name || type == STATUSTYPE_INFO) |
1218 | sz += hwh->type->status(hwh, type, result + sz, maxlen - sz); | ||
1219 | else if (!hwh->type || type == STATUSTYPE_INFO) | ||
1220 | DMEMIT("0 "); | 1249 | DMEMIT("0 "); |
1221 | else | 1250 | else |
1222 | DMEMIT("1 %s ", hwh->type->name); | 1251 | DMEMIT("1 %s ", m->hw_handler_name); |
1223 | 1252 | ||
1224 | DMEMIT("%u ", m->nr_priority_groups); | 1253 | DMEMIT("%u ", m->nr_priority_groups); |
1225 | 1254 | ||