diff options
author | Arnd Bergmann <arnd@arndb.de> | 2014-02-26 06:01:54 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-02-26 16:06:13 -0500 |
commit | c11da83bdae210e1d40a6755b78f8543a9be9227 (patch) | |
tree | 5f3849f1681cd3c273030895c493e2c7180caa20 /drivers/isdn/divert/divert_procfs.c | |
parent | c728cc88cec52a6bf97679ed4353bc42ff25e6ab (diff) |
isdn: divert, hysdn: fix interruptible_sleep_on race
These two drivers use identical code for their procfs status
file handling, which contains a small race against status
data becoming available while reading the file.
This uses wait_event_interruptible instead to fix this
particular race and eventually get rid of all sleep_on
instances. There seems to be another race involving
multiple concurrent readers of the same procfs file, which
I don't try to fix here.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Cc: Karsten Keil <isdn@linux-pingi.de>
Cc: netdev@vger.kernel.org
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/isdn/divert/divert_procfs.c')
-rw-r--r-- | drivers/isdn/divert/divert_procfs.c | 7 |
1 files changed, 4 insertions, 3 deletions
diff --git a/drivers/isdn/divert/divert_procfs.c b/drivers/isdn/divert/divert_procfs.c index fb4f1bac0133..1c5dc345e7c5 100644 --- a/drivers/isdn/divert/divert_procfs.c +++ b/drivers/isdn/divert/divert_procfs.c | |||
@@ -86,12 +86,13 @@ isdn_divert_read(struct file *file, char __user *buf, size_t count, loff_t *off) | |||
86 | struct divert_info *inf; | 86 | struct divert_info *inf; |
87 | int len; | 87 | int len; |
88 | 88 | ||
89 | if (!*((struct divert_info **) file->private_data)) { | 89 | if (!(inf = *((struct divert_info **) file->private_data))) { |
90 | if (file->f_flags & O_NONBLOCK) | 90 | if (file->f_flags & O_NONBLOCK) |
91 | return -EAGAIN; | 91 | return -EAGAIN; |
92 | interruptible_sleep_on(&(rd_queue)); | 92 | wait_event_interruptible(rd_queue, (inf = |
93 | *((struct divert_info **) file->private_data))); | ||
93 | } | 94 | } |
94 | if (!(inf = *((struct divert_info **) file->private_data))) | 95 | if (!inf) |
95 | return (0); | 96 | return (0); |
96 | 97 | ||
97 | inf->usage_cnt--; /* new usage count */ | 98 | inf->usage_cnt--; /* new usage count */ |