Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
a5f55d73
Commit
a5f55d73
authored
Apr 12, 2003
by
James Bottomley
Browse files
Options
Browse Files
Download
Plain Diff
More axboe/patmans conflicts
parents
99d60d5d
f64f6b58
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
94 additions
and
68 deletions
+94
-68
drivers/scsi/scsi.h
drivers/scsi/scsi.h
+3
-2
drivers/scsi/scsi_lib.c
drivers/scsi/scsi_lib.c
+86
-62
drivers/scsi/scsi_scan.c
drivers/scsi/scsi_scan.c
+5
-4
No files found.
drivers/scsi/scsi.h
View file @
a5f55d73
...
@@ -533,10 +533,11 @@ extern int scsi_dev_info_list_add_str(char *);
...
@@ -533,10 +533,11 @@ extern int scsi_dev_info_list_add_str(char *);
/*
/*
* scsi_target: representation of a scsi target, for now, this is only
* scsi_target: representation of a scsi target, for now, this is only
* used for single_lun devices.
* used for single_lun devices. If no one has active IO to the target,
* starget_sdev_user is NULL, else it points to the active sdev.
*/
*/
struct
scsi_target
{
struct
scsi_target
{
unsigned
int
starget_busy
;
struct
scsi_device
*
starget_sdev_user
;
unsigned
int
starget_refcnt
;
unsigned
int
starget_refcnt
;
};
};
...
...
drivers/scsi/scsi_lib.c
View file @
a5f55d73
...
@@ -327,9 +327,9 @@ void scsi_setup_cmd_retry(struct scsi_cmnd *cmd)
...
@@ -327,9 +327,9 @@ void scsi_setup_cmd_retry(struct scsi_cmnd *cmd)
}
}
/*
/*
* Called for single_lun devices on IO completion. Clear starget_
busy, and
* Called for single_lun devices on IO completion. Clear starget_
sdev_user,
*
Call __blk_run_queue for all the scsi_devices on the target - including
*
and call __blk_run_queue for all the scsi_devices on the target -
* current_sdev first.
*
including
current_sdev first.
*
*
* Called with *no* scsi locks held.
* Called with *no* scsi locks held.
*/
*/
...
@@ -338,19 +338,33 @@ static void scsi_single_lun_run(struct scsi_device *current_sdev)
...
@@ -338,19 +338,33 @@ static void scsi_single_lun_run(struct scsi_device *current_sdev)
struct
scsi_device
*
sdev
;
struct
scsi_device
*
sdev
;
unsigned
int
flags
,
flags2
;
unsigned
int
flags
,
flags2
;
spin_lock_irqsave
(
current_sdev
->
request_queue
->
queue_lock
,
flags2
);
spin_lock_irqsave
(
current_sdev
->
host
->
host_lock
,
flags
);
spin_lock_irqsave
(
current_sdev
->
host
->
host_lock
,
flags
);
WARN_ON
(
!
current_sdev
->
sdev_target
->
starget_busy
);
WARN_ON
(
!
current_sdev
->
sdev_target
->
starget_sdev_user
);
if
(
current_sdev
->
device_busy
==
0
)
current_sdev
->
sdev_target
->
starget_sdev_user
=
NULL
;
current_sdev
->
sdev_target
->
starget_busy
=
0
;
spin_unlock_irqrestore
(
current_sdev
->
host
->
host_lock
,
flags
);
spin_unlock_irqrestore
(
current_sdev
->
host
->
host_lock
,
flags
);
/*
/*
* Call __blk_run_queue for all LUNs on the target, starting with
* Call __blk_run_queue for all LUNs on the target, starting with
* current_sdev.
* current_sdev. We race with others (to set starget_sdev_user),
* but in most cases, we will be first. Ideally, each LU on the
* target would get some limited time or requests on the target.
*/
*/
spin_lock_irqsave
(
current_sdev
->
request_queue
->
queue_lock
,
flags2
);
__blk_run_queue
(
current_sdev
->
request_queue
);
__blk_run_queue
(
current_sdev
->
request_queue
);
spin_unlock_irqrestore
(
current_sdev
->
request_queue
->
queue_lock
,
flags2
);
spin_unlock_irqrestore
(
current_sdev
->
request_queue
->
queue_lock
,
flags2
);
spin_lock_irqsave
(
current_sdev
->
host
->
host_lock
,
flags
);
if
(
current_sdev
->
sdev_target
->
starget_sdev_user
)
{
/*
* After unlock, this races with anyone clearing
* starget_sdev_user, but we (should) always enter this
* function again, avoiding any problems.
*/
spin_unlock_irqrestore
(
current_sdev
->
host
->
host_lock
,
flags
);
return
;
}
spin_unlock_irqrestore
(
current_sdev
->
host
->
host_lock
,
flags
);
list_for_each_entry
(
sdev
,
&
current_sdev
->
same_target_siblings
,
list_for_each_entry
(
sdev
,
&
current_sdev
->
same_target_siblings
,
same_target_siblings
)
{
same_target_siblings
)
{
spin_lock_irqsave
(
sdev
->
request_queue
->
queue_lock
,
flags2
);
spin_lock_irqsave
(
sdev
->
request_queue
->
queue_lock
,
flags2
);
...
@@ -397,7 +411,7 @@ static void scsi_single_lun_run(struct scsi_device *current_sdev)
...
@@ -397,7 +411,7 @@ static void scsi_single_lun_run(struct scsi_device *current_sdev)
*/
*/
void
scsi_queue_next_request
(
request_queue_t
*
q
,
struct
scsi_cmnd
*
cmd
)
void
scsi_queue_next_request
(
request_queue_t
*
q
,
struct
scsi_cmnd
*
cmd
)
{
{
struct
scsi_device
*
sdev
,
*
sdev2
;
struct
scsi_device
*
sdev
;
struct
Scsi_Host
*
shost
;
struct
Scsi_Host
*
shost
;
unsigned
long
flags
;
unsigned
long
flags
;
...
@@ -446,16 +460,23 @@ void scsi_queue_next_request(request_queue_t *q, struct scsi_cmnd *cmd)
...
@@ -446,16 +460,23 @@ void scsi_queue_next_request(request_queue_t *q, struct scsi_cmnd *cmd)
* scsi_request_fn must get the host_lock before checking
* scsi_request_fn must get the host_lock before checking
* or modifying starved_list or starved_entry.
* or modifying starved_list or starved_entry.
*/
*/
sdev
2
=
list_entry
(
shost
->
starved_list
.
next
,
sdev
=
list_entry
(
shost
->
starved_list
.
next
,
struct
scsi_device
,
starved_entry
);
struct
scsi_device
,
starved_entry
);
list_del_init
(
&
sdev
2
->
starved_entry
);
list_del_init
(
&
sdev
->
starved_entry
);
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
spin_lock_irqsave
(
sdev
2
->
request_queue
->
queue_lock
,
flags
);
spin_lock_irqsave
(
sdev
->
request_queue
->
queue_lock
,
flags
);
__blk_run_queue
(
sdev
2
->
request_queue
);
__blk_run_queue
(
sdev
->
request_queue
);
spin_unlock_irqrestore
(
sdev
2
->
request_queue
->
queue_lock
,
flags
);
spin_unlock_irqrestore
(
sdev
->
request_queue
->
queue_lock
,
flags
);
spin_lock_irqsave
(
shost
->
host_lock
,
flags
);
spin_lock_irqsave
(
shost
->
host_lock
,
flags
);
if
(
unlikely
(
!
list_empty
(
&
sdev
->
starved_entry
)))
/*
* sdev lost a race, and was put back on the
* starved list. This is unlikely but without this
* in theory we could loop forever.
*/
break
;
}
}
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
...
@@ -1074,9 +1095,10 @@ static inline int scsi_dev_queue_ready(struct request_queue *q,
...
@@ -1074,9 +1095,10 @@ static inline int scsi_dev_queue_ready(struct request_queue *q,
/*
/*
* scsi_host_queue_ready: if we can send requests to shost, return 1 else
* scsi_host_queue_ready: if we can send requests to shost, return 1 else
* return 0.
* return 0. We must end up running the queue again whenever 0 is
* returned, else IO can hang.
*
*
* Called with
queue_lock and
host_lock held.
* Called with host_lock held.
*/
*/
static
inline
int
scsi_host_queue_ready
(
struct
request_queue
*
q
,
static
inline
int
scsi_host_queue_ready
(
struct
request_queue
*
q
,
struct
Scsi_Host
*
shost
,
struct
Scsi_Host
*
shost
,
...
@@ -1101,15 +1123,7 @@ static inline int scsi_host_queue_ready(struct request_queue *q,
...
@@ -1101,15 +1123,7 @@ static inline int scsi_host_queue_ready(struct request_queue *q,
return
0
;
return
0
;
if
((
shost
->
can_queue
>
0
&&
shost
->
host_busy
>=
shost
->
can_queue
)
||
if
((
shost
->
can_queue
>
0
&&
shost
->
host_busy
>=
shost
->
can_queue
)
||
shost
->
host_blocked
||
shost
->
host_self_blocked
)
{
shost
->
host_blocked
||
shost
->
host_self_blocked
)
{
SCSI_LOG_MLQUEUE
(
3
,
list_add_tail
(
&
sdev
->
starved_entry
,
&
shost
->
starved_list
);
printk
(
"add starved dev <%d,%d,%d,%d>; host "
"limit %d, busy %d, blocked %d selfblocked %d
\n
"
,
sdev
->
host
->
host_no
,
sdev
->
channel
,
sdev
->
id
,
sdev
->
lun
,
shost
->
can_queue
,
shost
->
host_busy
,
shost
->
host_blocked
,
shost
->
host_self_blocked
));
list_add_tail
(
&
sdev
->
starved_entry
,
&
shost
->
starved_list
);
return
0
;
return
0
;
}
}
...
@@ -1143,61 +1157,58 @@ static void scsi_request_fn(request_queue_t *q)
...
@@ -1143,61 +1157,58 @@ static void scsi_request_fn(request_queue_t *q)
if
(
blk_queue_plugged
(
q
))
if
(
blk_queue_plugged
(
q
))
goto
completed
;
goto
completed
;
if
(
!
scsi_dev_queue_ready
(
q
,
sdev
))
goto
completed
;
spin_lock_irqsave
(
shost
->
host_lock
,
flags
);
if
(
!
scsi_host_queue_ready
(
q
,
shost
,
sdev
))
goto
after_host_lock
;
if
(
sdev
->
single_lun
&&
!
sdev
->
device_busy
&&
sdev
->
sdev_target
->
starget_busy
)
goto
after_host_lock
;
/*
/*
* get next queueable request. We do this early to make sure
* get next queueable request. We do this early to make sure
* that the request is fully prepared even if we cannot
* that the request is fully prepared even if we cannot
* accept it. If there is no request, we'll detect this
* accept it.
* lower down.
*/
*/
req
=
elv_next_request
(
q
);
req
=
elv_next_request
(
q
);
if
(
!
req
)
{
if
(
!
req
)
{
/* If the device is busy, a returning I/O
/*
* will restart the queue. Otherwise, we have
* If the device is busy, a returning I/O will
* to plug the queue */
* restart the queue. Otherwise, we have to plug
if
(
sdev
->
device_busy
==
1
)
* the queue
*/
if
(
sdev
->
device_busy
==
0
)
blk_plug_device
(
q
);
blk_plug_device
(
q
);
goto
after_host_lock
;
goto
completed
;
}
}
cmd
=
req
->
special
;
if
(
!
scsi_dev_queue_ready
(
q
,
sdev
))
goto
completed
;
/*
* Should be impossible for a correctly prepared request
* please mail the stack trace to linux-scsi@vger.kernel.org
*/
BUG_ON
(
!
cmd
);
/*
/*
* Finally, before we release the lock, we copy the
* Remove the request from the request list.
* request to the command block, and remove the
* request from the request list. Note that we always
* operate on the queue head - there is absolutely no
* reason to search the list, because all of the
* commands in this queue are for the same device.
*/
*/
if
(
!
(
blk_queue_tagged
(
q
)
&&
(
blk_queue_start_tag
(
q
,
req
)
==
0
)))
if
(
!
(
blk_queue_tagged
(
q
)
&&
(
blk_queue_start_tag
(
q
,
req
)
==
0
)))
blkdev_dequeue_request
(
req
);
blkdev_dequeue_request
(
req
);
if
(
sdev
->
single_lun
)
sdev
->
device_busy
++
;
sdev
->
sdev_target
->
starget_busy
=
1
;
spin_unlock_irq
(
q
->
queue_lock
);
spin_lock_irqsave
(
shost
->
host_lock
,
flags
);
if
(
!
scsi_host_queue_ready
(
q
,
shost
,
sdev
))
goto
host_lock_held
;
if
(
sdev
->
single_lun
)
{
if
(
sdev
->
sdev_target
->
starget_sdev_user
&&
(
sdev
->
sdev_target
->
starget_sdev_user
!=
sdev
))
goto
host_lock_held
;
else
sdev
->
sdev_target
->
starget_sdev_user
=
sdev
;
}
shost
->
host_busy
++
;
shost
->
host_busy
++
;
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
sdev
->
device_busy
++
;
cmd
=
req
->
special
;
spin_unlock_irq
(
q
->
queue_lock
);
/*
* Should be impossible for a correctly prepared request
* please mail the stack trace to linux-scsi@vger.kernel.org
*/
BUG_ON
(
!
cmd
);
/*
/*
* Finally, initialize any error handling parameters, and set up
* Finally, initialize any error handling parameters, and set up
...
@@ -1219,8 +1230,21 @@ static void scsi_request_fn(request_queue_t *q)
...
@@ -1219,8 +1230,21 @@ static void scsi_request_fn(request_queue_t *q)
completed:
completed:
return
;
return
;
after_host_lock
:
host_lock_held
:
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
spin_unlock_irqrestore
(
shost
->
host_lock
,
flags
);
/*
* lock q, handle tag, requeue req, and decrement device_busy. We
* must return with queue_lock held.
*
* Decrementing device_busy without checking it is OK, as all such
* cases (host limits or settings) should run the queue at some
* later time.
*/
spin_lock_irq
(
q
->
queue_lock
);
if
(
blk_rq_tagged
(
req
))
blk_queue_end_tag
(
q
,
req
);
__elv_add_request
(
q
,
req
,
0
,
0
);
sdev
->
device_busy
--
;
}
}
u64
scsi_calculate_bounce_limit
(
struct
Scsi_Host
*
shost
)
u64
scsi_calculate_bounce_limit
(
struct
Scsi_Host
*
shost
)
...
...
drivers/scsi/scsi_scan.c
View file @
a5f55d73
...
@@ -496,13 +496,14 @@ static void scsi_free_sdev(struct scsi_device *sdev)
...
@@ -496,13 +496,14 @@ static void scsi_free_sdev(struct scsi_device *sdev)
sdev
->
host
->
hostt
->
slave_destroy
(
sdev
);
sdev
->
host
->
hostt
->
slave_destroy
(
sdev
);
if
(
sdev
->
inquiry
)
if
(
sdev
->
inquiry
)
kfree
(
sdev
->
inquiry
);
kfree
(
sdev
->
inquiry
);
if
(
sdev
->
single_lun
)
{
spin_lock_irqsave
(
sdev
->
host
->
host_lock
,
flags
);
spin_lock_irqsave
(
sdev
->
host
->
host_lock
,
flags
);
list_del
(
&
sdev
->
starved_entry
);
if
(
sdev
->
single_lun
)
{
sdev
->
sdev_target
->
starget_refcnt
--
;
sdev
->
sdev_target
->
starget_refcnt
--
;
if
(
sdev
->
sdev_target
->
starget_refcnt
==
0
)
if
(
sdev
->
sdev_target
->
starget_refcnt
==
0
)
kfree
(
sdev
->
sdev_target
);
kfree
(
sdev
->
sdev_target
);
spin_unlock_irqrestore
(
sdev
->
host
->
host_lock
,
flags
);
}
}
spin_unlock_irqrestore
(
sdev
->
host
->
host_lock
,
flags
);
kfree
(
sdev
);
kfree
(
sdev
);
}
}
...
@@ -1282,7 +1283,7 @@ static int scsi_add_lun(Scsi_Device *sdev, Scsi_Request *sreq,
...
@@ -1282,7 +1283,7 @@ static int scsi_add_lun(Scsi_Device *sdev, Scsi_Request *sreq,
}
}
}
}
if
(
!
starget
)
{
if
(
!
starget
)
{
starget
=
kmalloc
(
sizeof
(
*
starget
),
GFP_
KERNEL
);
starget
=
kmalloc
(
sizeof
(
*
starget
),
GFP_
ATOMIC
);
if
(
!
starget
)
{
if
(
!
starget
)
{
printk
(
ALLOC_FAILURE_MSG
,
__FUNCTION__
);
printk
(
ALLOC_FAILURE_MSG
,
__FUNCTION__
);
spin_unlock_irqrestore
(
sdev
->
host
->
host_lock
,
spin_unlock_irqrestore
(
sdev
->
host
->
host_lock
,
...
@@ -1290,7 +1291,7 @@ static int scsi_add_lun(Scsi_Device *sdev, Scsi_Request *sreq,
...
@@ -1290,7 +1291,7 @@ static int scsi_add_lun(Scsi_Device *sdev, Scsi_Request *sreq,
return
SCSI_SCAN_NO_RESPONSE
;
return
SCSI_SCAN_NO_RESPONSE
;
}
}
starget
->
starget_refcnt
=
0
;
starget
->
starget_refcnt
=
0
;
starget
->
starget_
busy
=
0
;
starget
->
starget_
sdev_user
=
NULL
;
}
}
starget
->
starget_refcnt
++
;
starget
->
starget_refcnt
++
;
sdev
->
sdev_target
=
starget
;
sdev
->
sdev_target
=
starget
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment