diff options
-rw-r--r-- | drivers/scsi/scsi_scan.c | 27 | ||||
-rw-r--r-- | include/scsi/scsi_host.h | 18 |
2 files changed, 38 insertions, 7 deletions
diff --git a/drivers/scsi/scsi_scan.c b/drivers/scsi/scsi_scan.c index 3ccaa4be92d8..4d656148bd67 100644 --- a/drivers/scsi/scsi_scan.c +++ b/drivers/scsi/scsi_scan.c | |||
@@ -1642,7 +1642,7 @@ static void scsi_sysfs_add_devices(struct Scsi_Host *shost) | |||
1642 | * that other asynchronous scans started after this one won't affect the | 1642 | * that other asynchronous scans started after this one won't affect the |
1643 | * ordering of the discovered devices. | 1643 | * ordering of the discovered devices. |
1644 | */ | 1644 | */ |
1645 | struct async_scan_data *scsi_prep_async_scan(struct Scsi_Host *shost) | 1645 | static struct async_scan_data *scsi_prep_async_scan(struct Scsi_Host *shost) |
1646 | { | 1646 | { |
1647 | struct async_scan_data *data; | 1647 | struct async_scan_data *data; |
1648 | 1648 | ||
@@ -1686,7 +1686,7 @@ struct async_scan_data *scsi_prep_async_scan(struct Scsi_Host *shost) | |||
1686 | * This function announces all the devices it has found to the rest | 1686 | * This function announces all the devices it has found to the rest |
1687 | * of the system. | 1687 | * of the system. |
1688 | */ | 1688 | */ |
1689 | void scsi_finish_async_scan(struct async_scan_data *data) | 1689 | static void scsi_finish_async_scan(struct async_scan_data *data) |
1690 | { | 1690 | { |
1691 | struct Scsi_Host *shost; | 1691 | struct Scsi_Host *shost; |
1692 | 1692 | ||
@@ -1719,12 +1719,25 @@ void scsi_finish_async_scan(struct async_scan_data *data) | |||
1719 | kfree(data); | 1719 | kfree(data); |
1720 | } | 1720 | } |
1721 | 1721 | ||
1722 | static int do_scan_async(void *_data) | 1722 | static void do_scsi_scan_host(struct Scsi_Host *shost) |
1723 | { | 1723 | { |
1724 | struct async_scan_data *data = _data; | 1724 | if (shost->hostt->scan_finished) { |
1725 | scsi_scan_host_selected(data->shost, SCAN_WILD_CARD, SCAN_WILD_CARD, | 1725 | unsigned long start = jiffies; |
1726 | if (shost->hostt->scan_start) | ||
1727 | shost->hostt->scan_start(shost); | ||
1728 | |||
1729 | while (!shost->hostt->scan_finished(shost, jiffies - start)) | ||
1730 | msleep(10); | ||
1731 | } else { | ||
1732 | scsi_scan_host_selected(shost, SCAN_WILD_CARD, SCAN_WILD_CARD, | ||
1726 | SCAN_WILD_CARD, 0); | 1733 | SCAN_WILD_CARD, 0); |
1734 | } | ||
1735 | } | ||
1727 | 1736 | ||
1737 | static int do_scan_async(void *_data) | ||
1738 | { | ||
1739 | struct async_scan_data *data = _data; | ||
1740 | do_scsi_scan_host(data->shost); | ||
1728 | scsi_finish_async_scan(data); | 1741 | scsi_finish_async_scan(data); |
1729 | return 0; | 1742 | return 0; |
1730 | } | 1743 | } |
@@ -1742,10 +1755,10 @@ void scsi_scan_host(struct Scsi_Host *shost) | |||
1742 | 1755 | ||
1743 | data = scsi_prep_async_scan(shost); | 1756 | data = scsi_prep_async_scan(shost); |
1744 | if (!data) { | 1757 | if (!data) { |
1745 | scsi_scan_host_selected(shost, SCAN_WILD_CARD, SCAN_WILD_CARD, | 1758 | do_scsi_scan_host(shost); |
1746 | SCAN_WILD_CARD, 0); | ||
1747 | return; | 1759 | return; |
1748 | } | 1760 | } |
1761 | |||
1749 | kthread_run(do_scan_async, data, "scsi_scan_%d", shost->host_no); | 1762 | kthread_run(do_scan_async, data, "scsi_scan_%d", shost->host_no); |
1750 | } | 1763 | } |
1751 | EXPORT_SYMBOL(scsi_scan_host); | 1764 | EXPORT_SYMBOL(scsi_scan_host); |
diff --git a/include/scsi/scsi_host.h b/include/scsi/scsi_host.h index e618e711ea57..6d8945d71c65 100644 --- a/include/scsi/scsi_host.h +++ b/include/scsi/scsi_host.h | |||
@@ -241,6 +241,24 @@ struct scsi_host_template { | |||
241 | void (* target_destroy)(struct scsi_target *); | 241 | void (* target_destroy)(struct scsi_target *); |
242 | 242 | ||
243 | /* | 243 | /* |
244 | * If a host has the ability to discover targets on its own instead | ||
245 | * of scanning the entire bus, it can fill in this function and | ||
246 | * call scsi_scan_host(). This function will be called periodically | ||
247 | * until it returns 1 with the scsi_host and the elapsed time of | ||
248 | * the scan in jiffies. | ||
249 | * | ||
250 | * Status: OPTIONAL | ||
251 | */ | ||
252 | int (* scan_finished)(struct Scsi_Host *, unsigned long); | ||
253 | |||
254 | /* | ||
255 | * If the host wants to be called before the scan starts, but | ||
256 | * after the midlayer has set up ready for the scan, it can fill | ||
257 | * in this function. | ||
258 | */ | ||
259 | void (* scan_start)(struct Scsi_Host *); | ||
260 | |||
261 | /* | ||
244 | * fill in this function to allow the queue depth of this host | 262 | * fill in this function to allow the queue depth of this host |
245 | * to be changeable (on a per device basis). returns either | 263 | * to be changeable (on a per device basis). returns either |
246 | * the current queue depth setting (may be different from what | 264 | * the current queue depth setting (may be different from what |