mirror of
https://github.com/adulau/aha.git
synced 2024-12-28 11:46:19 +00:00
[SCSI] Add missing completion to scsi_complete_async_scans()
If either scsi_complete_async_scans() is called a second time before the first call has finished, or a host scan is started while scsi_complete_async_scans() is still sleeping, it would fail to wake up the other task, which would sleep forever. I've changed the kernel-doc to make it clear that scsi_complete_async_scans() only guarantees that scans which started before it was called are guaranteed to have finished when it returns. I considered making it wait until all scans are completed, but it can't guarantee that no more scans will start before it returns anyway, and it runs the risk of confusing other callers of scsi_complete_async_scans() for hosts actually scanning. Signed-off-by: Matthew Wilcox <matthew@wil.cx> Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
This commit is contained in:
parent
23be331d9e
commit
8bcc24127b
1 changed files with 9 additions and 6 deletions
|
@ -133,12 +133,10 @@ struct async_scan_data {
|
|||
/**
|
||||
* scsi_complete_async_scans - Wait for asynchronous scans to complete
|
||||
*
|
||||
* Asynchronous scans add themselves to the scanning_hosts list. Once
|
||||
* that list is empty, we know that the scans are complete. Rather than
|
||||
* waking up periodically to check the state of the list, we pretend to be
|
||||
* a scanning task by adding ourselves at the end of the list and going to
|
||||
* sleep. When the task before us wakes us up, we take ourselves off the
|
||||
* list and return.
|
||||
* When this function returns, any host which started scanning before
|
||||
* this function was called will have finished its scan. Hosts which
|
||||
* started scanning after this function was called may or may not have
|
||||
* finished.
|
||||
*/
|
||||
int scsi_complete_async_scans(void)
|
||||
{
|
||||
|
@ -171,6 +169,11 @@ int scsi_complete_async_scans(void)
|
|||
|
||||
spin_lock(&async_scan_lock);
|
||||
list_del(&data->list);
|
||||
if (!list_empty(&scanning_hosts)) {
|
||||
struct async_scan_data *next = list_entry(scanning_hosts.next,
|
||||
struct async_scan_data, list);
|
||||
complete(&next->prev_finished);
|
||||
}
|
||||
done:
|
||||
spin_unlock(&async_scan_lock);
|
||||
|
||||
|
|
Loading…
Reference in a new issue