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
60d2ad5f
Commit
60d2ad5f
authored
May 30, 2002
by
Dave Jones
Committed by
Linus Torvalds
May 30, 2002
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[PATCH] Fix deadlock in nbd
Variant of a patch in 2.4 by Steven Whitehouse.
parent
1b3c4f5a
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
79 additions
and
55 deletions
+79
-55
drivers/block/nbd.c
drivers/block/nbd.c
+70
-55
include/linux/nbd.h
include/linux/nbd.h
+9
-0
No files found.
drivers/block/nbd.c
View file @
60d2ad5f
...
@@ -24,6 +24,8 @@
...
@@ -24,6 +24,8 @@
* 01-3-11 Make nbd work with new Linux block layer code. It now supports
* 01-3-11 Make nbd work with new Linux block layer code. It now supports
* plugging like all the other block devices. Also added in MSG_MORE to
* plugging like all the other block devices. Also added in MSG_MORE to
* reduce number of partial TCP segments sent. <steve@chygwyn.com>
* reduce number of partial TCP segments sent. <steve@chygwyn.com>
* 01-12-6 Fix deadlock condition by making queue locks independant of
* the transmit lock. <steve@chygwyn.com>
*
*
* possible FIXME: make set_sock / set_blksize / set_size / do_it one syscall
* possible FIXME: make set_sock / set_blksize / set_size / do_it one syscall
* why not: would need verify_area and friends, would share yet another
* why not: would need verify_area and friends, would share yet another
...
@@ -148,27 +150,28 @@ static int nbd_xmit(int send, struct socket *sock, char *buf, int size, int msg_
...
@@ -148,27 +150,28 @@ static int nbd_xmit(int send, struct socket *sock, char *buf, int size, int msg_
#define FAIL( s ) { printk( KERN_ERR "NBD: " s "(result %d)\n", result ); goto error_out; }
#define FAIL( s ) { printk( KERN_ERR "NBD: " s "(result %d)\n", result ); goto error_out; }
void
nbd_send_req
(
struct
socket
*
sock
,
struct
request
*
req
)
void
nbd_send_req
(
struct
nbd_device
*
lo
,
struct
request
*
req
)
{
{
int
result
,
rw
,
i
,
flags
;
int
result
,
i
,
flags
;
struct
nbd_request
request
;
struct
nbd_request
request
;
unsigned
long
size
=
req
->
nr_sectors
<<
9
;
unsigned
long
size
=
req
->
nr_sectors
<<
9
;
struct
socket
*
sock
=
lo
->
sock
;
DEBUG
(
"NBD: sending control, "
);
DEBUG
(
"NBD: sending control, "
);
rw
=
rq_data_dir
(
req
);
request
.
magic
=
htonl
(
NBD_REQUEST_MAGIC
);
request
.
magic
=
htonl
(
NBD_REQUEST_MAGIC
);
request
.
type
=
htonl
(
(
rw
&
WRITE
)
?
1
:
0
);
request
.
type
=
htonl
(
nbd_cmd
(
req
)
);
request
.
from
=
cpu_to_be64
(
(
u64
)
req
->
sector
<<
9
);
request
.
from
=
cpu_to_be64
(
(
u64
)
req
->
sector
<<
9
);
request
.
len
=
htonl
(
size
);
request
.
len
=
htonl
(
size
);
memcpy
(
request
.
handle
,
&
req
,
sizeof
(
req
));
memcpy
(
request
.
handle
,
&
req
,
sizeof
(
req
));
result
=
nbd_xmit
(
1
,
sock
,
(
char
*
)
&
request
,
sizeof
(
request
),
rw
&
WRITE
?
MSG_MORE
:
0
);
down
(
&
lo
->
tx_lock
);
result
=
nbd_xmit
(
1
,
sock
,
(
char
*
)
&
request
,
sizeof
(
request
),
nbd_cmd
(
req
)
==
NBD_CMD_WRITE
?
MSG_MORE
:
0
);
if
(
result
<=
0
)
if
(
result
<=
0
)
FAIL
(
"Sendmsg failed for control."
);
FAIL
(
"Sendmsg failed for control."
);
if
(
rw
&
WRITE
)
{
if
(
nbd_cmd
(
req
)
==
NBD_CMD_
WRITE
)
{
struct
bio
*
bio
;
struct
bio
*
bio
;
/*
/*
* we are really probing at internals to determine
* we are really probing at internals to determine
...
@@ -187,37 +190,59 @@ void nbd_send_req(struct socket *sock, struct request *req)
...
@@ -187,37 +190,59 @@ void nbd_send_req(struct socket *sock, struct request *req)
}
}
}
}
}
}
up
(
&
lo
->
tx_lock
);
return
;
return
;
error_out:
error_out:
up
(
&
lo
->
tx_lock
);
req
->
errors
++
;
req
->
errors
++
;
}
}
static
struct
request
*
nbd_find_request
(
struct
nbd_device
*
lo
,
char
*
handle
)
{
struct
request
*
req
;
struct
list_head
*
tmp
;
struct
request
*
xreq
;
memcpy
(
&
xreq
,
handle
,
sizeof
(
xreq
));
spin_lock
(
&
lo
->
queue_lock
);
list_for_each
(
tmp
,
&
lo
->
queue_head
)
{
req
=
list_entry
(
tmp
,
struct
request
,
queuelist
);
if
(
req
!=
xreq
)
continue
;
list_del
(
&
req
->
queuelist
);
spin_unlock
(
&
lo
->
queue_lock
);
return
req
;
}
spin_unlock
(
&
lo
->
queue_lock
);
return
NULL
;
}
#define HARDFAIL( s ) { printk( KERN_ERR "NBD: " s "(result %d)\n", result ); lo->harderror = result; return NULL; }
#define HARDFAIL( s ) { printk( KERN_ERR "NBD: " s "(result %d)\n", result ); lo->harderror = result; return NULL; }
struct
request
*
nbd_read_stat
(
struct
nbd_device
*
lo
)
struct
request
*
nbd_read_stat
(
struct
nbd_device
*
lo
)
/* NULL returned = something went wrong, inform userspace */
/* NULL returned = something went wrong, inform userspace */
{
{
int
result
;
int
result
;
struct
nbd_reply
reply
;
struct
nbd_reply
reply
;
struct
request
*
xreq
,
*
req
;
struct
request
*
req
;
DEBUG
(
"reading control, "
);
DEBUG
(
"reading control, "
);
reply
.
magic
=
0
;
reply
.
magic
=
0
;
result
=
nbd_xmit
(
0
,
lo
->
sock
,
(
char
*
)
&
reply
,
sizeof
(
reply
),
MSG_WAITALL
);
result
=
nbd_xmit
(
0
,
lo
->
sock
,
(
char
*
)
&
reply
,
sizeof
(
reply
),
MSG_WAITALL
);
if
(
result
<=
0
)
if
(
result
<=
0
)
HARDFAIL
(
"Recv control failed."
);
HARDFAIL
(
"Recv control failed."
);
memcpy
(
&
xreq
,
reply
.
handle
,
sizeof
(
xreq
));
req
=
nbd_find_request
(
lo
,
reply
.
handle
);
req
=
blkdev_entry_to_request
(
lo
->
queue_head
.
prev
);
if
(
req
==
NULL
)
HARDFAIL
(
"Unexpected reply"
);
if
(
xreq
!=
req
)
FAIL
(
"Unexpected handle received.
\n
"
);
DEBUG
(
"ok, "
);
DEBUG
(
"ok, "
);
if
(
ntohl
(
reply
.
magic
)
!=
NBD_REPLY_MAGIC
)
if
(
ntohl
(
reply
.
magic
)
!=
NBD_REPLY_MAGIC
)
HARDFAIL
(
"Not enough magic."
);
HARDFAIL
(
"Not enough magic."
);
if
(
ntohl
(
reply
.
error
))
if
(
ntohl
(
reply
.
error
))
FAIL
(
"Other side returned error."
);
FAIL
(
"Other side returned error."
);
if
(
rq_data_dir
(
req
)
==
READ
)
{
if
(
nbd_cmd
(
req
)
==
NBD_CMD_READ
)
{
struct
bio
*
bio
=
req
->
bio
;
struct
bio
*
bio
=
req
->
bio
;
DEBUG
(
"data, "
);
DEBUG
(
"data, "
);
do
{
do
{
...
@@ -240,20 +265,14 @@ void nbd_do_it(struct nbd_device *lo)
...
@@ -240,20 +265,14 @@ void nbd_do_it(struct nbd_device *lo)
{
{
struct
request
*
req
;
struct
request
*
req
;
down
(
&
lo
->
tx_lock
);
while
(
1
)
{
while
(
1
)
{
up
(
&
lo
->
tx_lock
);
req
=
nbd_read_stat
(
lo
);
req
=
nbd_read_stat
(
lo
);
down
(
&
lo
->
tx_lock
);
if
(
!
req
)
{
if
(
!
req
)
{
printk
(
KERN_ALERT
"req should never be null
\n
"
);
printk
(
KERN_ALERT
"req should never be null
\n
"
);
goto
out
;
goto
out
;
}
}
#ifdef PARANOIA
#ifdef PARANOIA
if
(
req
!=
blkdev_entry_to_request
(
lo
->
queue_head
.
prev
))
{
printk
(
KERN_ALERT
"NBD: I have problem...
\n
"
);
}
if
(
lo
!=
&
nbd_dev
[
minor
(
req
->
rq_dev
)])
{
if
(
lo
!=
&
nbd_dev
[
minor
(
req
->
rq_dev
)])
{
printk
(
KERN_ALERT
"NBD: request corrupted!
\n
"
);
printk
(
KERN_ALERT
"NBD: request corrupted!
\n
"
);
continue
;
continue
;
...
@@ -263,15 +282,10 @@ void nbd_do_it(struct nbd_device *lo)
...
@@ -263,15 +282,10 @@ void nbd_do_it(struct nbd_device *lo)
goto
out
;
goto
out
;
}
}
#endif
#endif
blkdev_dequeue_request
(
req
);
up
(
&
lo
->
tx_lock
);
nbd_end_request
(
req
);
nbd_end_request
(
req
);
down
(
&
lo
->
tx_lock
);
}
}
out:
out:
up
(
&
lo
->
tx_lock
);
}
}
void
nbd_clear_que
(
struct
nbd_device
*
lo
)
void
nbd_clear_que
(
struct
nbd_device
*
lo
)
...
@@ -285,26 +299,19 @@ void nbd_clear_que(struct nbd_device *lo)
...
@@ -285,26 +299,19 @@ void nbd_clear_que(struct nbd_device *lo)
}
}
#endif
#endif
while
(
!
list_empty
(
&
lo
->
queue_head
))
{
do
{
req
=
blkdev_entry_to_request
(
lo
->
queue_head
.
prev
);
req
=
NULL
;
#ifdef PARANOIA
spin_lock
(
&
lo
->
queue_lock
);
if
(
!
req
)
{
if
(
!
list_empty
(
&
lo
->
queue_head
))
{
printk
(
KERN_ALERT
"NBD: panic, panic, panic
\n
"
);
req
=
list_entry
(
lo
->
queue_head
.
next
,
struct
request
,
queuelist
);
break
;
list_del
(
&
req
->
queuelist
);
}
if
(
lo
!=
&
nbd_dev
[
minor
(
req
->
rq_dev
)])
{
printk
(
KERN_ALERT
"NBD: request corrupted when clearing!
\n
"
);
continue
;
}
}
#endif
spin_unlock
(
&
lo
->
queue_lock
);
if
(
req
)
{
req
->
errors
++
;
req
->
errors
++
;
blkdev_dequeue_request
(
req
);
up
(
&
lo
->
tx_lock
);
nbd_end_request
(
req
);
nbd_end_request
(
req
);
down
(
&
lo
->
tx_lock
);
}
}
}
while
(
req
);
}
}
/*
/*
...
@@ -340,8 +347,12 @@ static void do_nbd_request(request_queue_t * q)
...
@@ -340,8 +347,12 @@ static void do_nbd_request(request_queue_t * q)
lo
=
&
nbd_dev
[
dev
];
lo
=
&
nbd_dev
[
dev
];
if
(
!
lo
->
file
)
if
(
!
lo
->
file
)
FAIL
(
"Request when not-ready."
);
FAIL
(
"Request when not-ready."
);
if
((
rq_data_dir
(
req
)
==
WRITE
)
&&
(
lo
->
flags
&
NBD_READ_ONLY
))
nbd_cmd
(
req
)
=
NBD_CMD_READ
;
if
(
rq_data_dir
(
req
)
==
WRITE
)
{
nbd_cmd
(
req
)
=
NBD_CMD_WRITE
;
if
(
lo
->
flags
&
NBD_READ_ONLY
)
FAIL
(
"Write on read-only"
);
FAIL
(
"Write on read-only"
);
}
#ifdef PARANOIA
#ifdef PARANOIA
if
(
lo
->
magic
!=
LO_MAGIC
)
if
(
lo
->
magic
!=
LO_MAGIC
)
FAIL
(
"nbd[] is not magical!"
);
FAIL
(
"nbd[] is not magical!"
);
...
@@ -351,10 +362,11 @@ static void do_nbd_request(request_queue_t * q)
...
@@ -351,10 +362,11 @@ static void do_nbd_request(request_queue_t * q)
blkdev_dequeue_request
(
req
);
blkdev_dequeue_request
(
req
);
spin_unlock_irq
(
q
->
queue_lock
);
spin_unlock_irq
(
q
->
queue_lock
);
down
(
&
lo
->
tx
_lock
);
spin_lock
(
&
lo
->
queue
_lock
);
list_add
(
&
req
->
queuelist
,
&
lo
->
queue_head
);
list_add
(
&
req
->
queuelist
,
&
lo
->
queue_head
);
nbd_send_req
(
lo
->
sock
,
req
);
/* Why does this block? */
spin_unlock
(
&
lo
->
queue_lock
);
up
(
&
lo
->
tx_lock
);
nbd_send_req
(
lo
,
req
);
spin_lock_irq
(
q
->
queue_lock
);
spin_lock_irq
(
q
->
queue_lock
);
continue
;
continue
;
...
@@ -389,21 +401,23 @@ static int nbd_ioctl(struct inode *inode, struct file *file,
...
@@ -389,21 +401,23 @@ static int nbd_ioctl(struct inode *inode, struct file *file,
lo
=
&
nbd_dev
[
dev
];
lo
=
&
nbd_dev
[
dev
];
switch
(
cmd
)
{
switch
(
cmd
)
{
case
NBD_DISCONNECT
:
case
NBD_DISCONNECT
:
printk
(
"NBD_DISCONNECT
\n
"
)
;
printk
(
KERN_INFO
"NBD_DISCONNECT
\n
"
);
sreq
.
flags
=
REQ_SPECIAL
;
/* FIXME: interpet as shutdown cmd */
sreq
.
flags
=
REQ_SPECIAL
;
if
(
!
lo
->
sock
)
return
-
EINVAL
;
nbd_cmd
(
&
sreq
)
=
NBD_CMD_DISC
;
nbd_send_req
(
lo
->
sock
,
&
sreq
)
;
if
(
!
lo
->
sock
)
return
-
EINVAL
;
nbd_send_req
(
lo
,
&
sreq
);
return
0
;
return
0
;
case
NBD_CLEAR_SOCK
:
case
NBD_CLEAR_SOCK
:
down
(
&
lo
->
tx_lock
);
nbd_clear_que
(
lo
);
nbd_clear_que
(
lo
);
spin_lock
(
&
lo
->
queue_lock
);
if
(
!
list_empty
(
&
lo
->
queue_head
))
{
if
(
!
list_empty
(
&
lo
->
queue_head
))
{
up
(
&
lo
->
tx
_lock
);
spin_unlock
(
&
lo
->
queue
_lock
);
printk
(
KERN_ERR
"nbd: Some requests are in progress -> can not turn off.
\n
"
);
printk
(
KERN_ERR
"nbd: Some requests are in progress -> can not turn off.
\n
"
);
return
-
EBUSY
;
return
-
EBUSY
;
}
}
up
(
&
lo
->
tx
_lock
);
spin_unlock
(
&
lo
->
queue
_lock
);
file
=
lo
->
file
;
file
=
lo
->
file
;
if
(
!
file
)
if
(
!
file
)
return
-
EINVAL
;
return
-
EINVAL
;
...
@@ -525,6 +539,7 @@ static int __init nbd_init(void)
...
@@ -525,6 +539,7 @@ static int __init nbd_init(void)
nbd_dev
[
i
].
file
=
NULL
;
nbd_dev
[
i
].
file
=
NULL
;
nbd_dev
[
i
].
magic
=
LO_MAGIC
;
nbd_dev
[
i
].
magic
=
LO_MAGIC
;
nbd_dev
[
i
].
flags
=
0
;
nbd_dev
[
i
].
flags
=
0
;
spin_lock_init
(
&
nbd_dev
[
i
].
queue_lock
);
INIT_LIST_HEAD
(
&
nbd_dev
[
i
].
queue_head
);
INIT_LIST_HEAD
(
&
nbd_dev
[
i
].
queue_head
);
init_MUTEX
(
&
nbd_dev
[
i
].
tx_lock
);
init_MUTEX
(
&
nbd_dev
[
i
].
tx_lock
);
nbd_blksizes
[
i
]
=
1024
;
nbd_blksizes
[
i
]
=
1024
;
...
...
include/linux/nbd.h
View file @
60d2ad5f
...
@@ -20,6 +20,12 @@
...
@@ -20,6 +20,12 @@
#define NBD_SET_SIZE_BLOCKS _IO( 0xab, 7 )
#define NBD_SET_SIZE_BLOCKS _IO( 0xab, 7 )
#define NBD_DISCONNECT _IO( 0xab, 8 )
#define NBD_DISCONNECT _IO( 0xab, 8 )
enum
{
NBD_CMD_READ
=
0
,
NBD_CMD_WRITE
=
1
,
NBD_CMD_DISC
=
2
};
#ifdef MAJOR_NR
#ifdef MAJOR_NR
#include <asm/semaphore.h>
#include <asm/semaphore.h>
...
@@ -33,6 +39,8 @@ extern int requests_in;
...
@@ -33,6 +39,8 @@ extern int requests_in;
extern
int
requests_out
;
extern
int
requests_out
;
#endif
#endif
#define nbd_cmd(req) ((req)->cmd[0])
static
void
static
void
nbd_end_request
(
struct
request
*
req
)
nbd_end_request
(
struct
request
*
req
)
{
{
...
@@ -68,6 +76,7 @@ struct nbd_device {
...
@@ -68,6 +76,7 @@ struct nbd_device {
struct
socket
*
sock
;
struct
socket
*
sock
;
struct
file
*
file
;
/* If == NULL, device is not ready, yet */
struct
file
*
file
;
/* If == NULL, device is not ready, yet */
int
magic
;
/* FIXME: not if debugging is off */
int
magic
;
/* FIXME: not if debugging is off */
spinlock_t
queue_lock
;
struct
list_head
queue_head
;
/* Requests are added here... */
struct
list_head
queue_head
;
/* Requests are added here... */
struct
semaphore
tx_lock
;
struct
semaphore
tx_lock
;
};
};
...
...
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