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
Kirill Smelkov
linux
Commits
05c0ae21
Commit
05c0ae21
authored
Apr 04, 2013
by
Al Viro
1
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
try a saner locking for pde_opener...
Signed-off-by:
Al Viro
<
viro@zeniv.linux.org.uk
>
parent
ca469f35
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
24 additions
and
44 deletions
+24
-44
fs/proc/inode.c
fs/proc/inode.c
+21
-41
fs/proc/internal.h
fs/proc/internal.h
+2
-2
include/linux/proc_fs.h
include/linux/proc_fs.h
+1
-1
No files found.
fs/proc/inode.c
View file @
05c0ae21
...
@@ -133,67 +133,48 @@ enum {BIAS = -1U<<31};
...
@@ -133,67 +133,48 @@ enum {BIAS = -1U<<31};
static
inline
int
use_pde
(
struct
proc_dir_entry
*
pde
)
static
inline
int
use_pde
(
struct
proc_dir_entry
*
pde
)
{
{
int
res
=
1
;
return
atomic_inc_unless_negative
(
&
pde
->
in_use
);
spin_lock
(
&
pde
->
pde_unload_lock
);
if
(
unlikely
(
pde
->
pde_users
<
0
))
res
=
0
;
else
pde
->
pde_users
++
;
spin_unlock
(
&
pde
->
pde_unload_lock
);
return
res
;
}
static
void
__pde_users_dec
(
struct
proc_dir_entry
*
pde
)
{
if
(
--
pde
->
pde_users
==
BIAS
)
complete
(
pde
->
pde_unload_completion
);
}
}
static
void
unuse_pde
(
struct
proc_dir_entry
*
pde
)
static
void
unuse_pde
(
struct
proc_dir_entry
*
pde
)
{
{
spin_lock
(
&
pde
->
pde_unload_lock
);
if
(
atomic_dec_return
(
&
pde
->
in_use
)
==
BIAS
)
__pde_users_dec
(
pde
);
complete
(
pde
->
pde_unload_completion
);
spin_unlock
(
&
pde
->
pde_unload_lock
);
}
}
/* pde is locked */
/* pde is locked */
static
void
close_pdeo
(
struct
proc_dir_entry
*
pde
,
struct
pde_opener
*
pdeo
)
static
void
close_pdeo
(
struct
proc_dir_entry
*
pde
,
struct
pde_opener
*
pdeo
)
{
{
pdeo
->
count
++
;
if
(
pdeo
->
closing
)
{
if
(
!
mutex_trylock
(
&
pdeo
->
mutex
))
{
/* somebody else is doing that, just wait */
/* somebody else is doing that, just wait */
DECLARE_COMPLETION_ONSTACK
(
c
);
pdeo
->
c
=
&
c
;
spin_unlock
(
&
pde
->
pde_unload_lock
);
spin_unlock
(
&
pde
->
pde_unload_lock
);
mutex_lock
(
&
pdeo
->
mutex
);
wait_for_completion
(
&
c
);
spin_lock
(
&
pde
->
pde_unload_lock
);
spin_lock
(
&
pde
->
pde_unload_lock
);
WARN_ON
(
!
list_empty
(
&
pdeo
->
lh
));
}
else
{
}
else
{
struct
file
*
file
;
struct
file
*
file
;
pdeo
->
closing
=
1
;
spin_unlock
(
&
pde
->
pde_unload_lock
);
spin_unlock
(
&
pde
->
pde_unload_lock
);
file
=
pdeo
->
file
;
file
=
pdeo
->
file
;
pde
->
proc_fops
->
release
(
file_inode
(
file
),
file
);
pde
->
proc_fops
->
release
(
file_inode
(
file
),
file
);
spin_lock
(
&
pde
->
pde_unload_lock
);
spin_lock
(
&
pde
->
pde_unload_lock
);
list_del_init
(
&
pdeo
->
lh
);
list_del_init
(
&
pdeo
->
lh
);
}
if
(
pdeo
->
c
)
mutex_unlock
(
&
pdeo
->
mutex
);
complete
(
pdeo
->
c
);
if
(
!--
pdeo
->
count
)
kfree
(
pdeo
);
kfree
(
pdeo
);
}
}
}
void
proc_entry_rundown
(
struct
proc_dir_entry
*
de
)
void
proc_entry_rundown
(
struct
proc_dir_entry
*
de
)
{
{
spin_lock
(
&
de
->
pde_unload_lock
);
DECLARE_COMPLETION_ONSTACK
(
c
);
de
->
pde_users
+=
BIAS
;
/* Wait until all existing callers into module are done. */
/* Wait until all existing callers into module are done. */
if
(
de
->
pde_users
!=
BIAS
)
{
de
->
pde_unload_completion
=
&
c
;
DECLARE_COMPLETION_ONSTACK
(
c
);
if
(
atomic_add_return
(
BIAS
,
&
de
->
in_use
)
!=
BIAS
)
de
->
pde_unload_completion
=
&
c
;
wait_for_completion
(
&
c
);
spin_unlock
(
&
de
->
pde_unload_lock
);
wait_for_completion
(
de
->
pde_unload_completion
);
spin_lock
(
&
de
->
pde_unload_lock
);
}
spin_lock
(
&
de
->
pde_unload_lock
);
while
(
!
list_empty
(
&
de
->
pde_openers
))
{
while
(
!
list_empty
(
&
de
->
pde_openers
))
{
struct
pde_opener
*
pdeo
;
struct
pde_opener
*
pdeo
;
pdeo
=
list_first_entry
(
&
de
->
pde_openers
,
struct
pde_opener
,
lh
);
pdeo
=
list_first_entry
(
&
de
->
pde_openers
,
struct
pde_opener
,
lh
);
...
@@ -356,7 +337,7 @@ static int proc_reg_open(struct inode *inode, struct file *file)
...
@@ -356,7 +337,7 @@ static int proc_reg_open(struct inode *inode, struct file *file)
* by hand in remove_proc_entry(). For this, save opener's credentials
* by hand in remove_proc_entry(). For this, save opener's credentials
* for later.
* for later.
*/
*/
pdeo
=
k
m
alloc
(
sizeof
(
struct
pde_opener
),
GFP_KERNEL
);
pdeo
=
k
z
alloc
(
sizeof
(
struct
pde_opener
),
GFP_KERNEL
);
if
(
!
pdeo
)
if
(
!
pdeo
)
return
-
ENOMEM
;
return
-
ENOMEM
;
...
@@ -370,18 +351,17 @@ static int proc_reg_open(struct inode *inode, struct file *file)
...
@@ -370,18 +351,17 @@ static int proc_reg_open(struct inode *inode, struct file *file)
if
(
open
)
if
(
open
)
rv
=
open
(
inode
,
file
);
rv
=
open
(
inode
,
file
);
spin_lock
(
&
pde
->
pde_unload_lock
);
if
(
rv
==
0
&&
release
)
{
if
(
rv
==
0
&&
release
)
{
/* To know what to release. */
/* To know what to release. */
mutex_init
(
&
pdeo
->
mutex
);
pdeo
->
count
=
0
;
pdeo
->
file
=
file
;
pdeo
->
file
=
file
;
/* Strictly for "too late" ->release in proc_reg_release(). */
/* Strictly for "too late" ->release in proc_reg_release(). */
spin_lock
(
&
pde
->
pde_unload_lock
);
list_add
(
&
pdeo
->
lh
,
&
pde
->
pde_openers
);
list_add
(
&
pdeo
->
lh
,
&
pde
->
pde_openers
);
spin_unlock
(
&
pde
->
pde_unload_lock
);
}
else
}
else
kfree
(
pdeo
);
kfree
(
pdeo
);
__pde_users_dec
(
pde
);
spin_unlock
(
&
pde
->
pde_unload_lock
);
unuse_pde
(
pde
);
return
rv
;
return
rv
;
}
}
...
...
fs/proc/internal.h
View file @
05c0ae21
...
@@ -153,8 +153,8 @@ int proc_readdir_de(struct proc_dir_entry *de, struct file *filp, void *dirent,
...
@@ -153,8 +153,8 @@ int proc_readdir_de(struct proc_dir_entry *de, struct file *filp, void *dirent,
struct
pde_opener
{
struct
pde_opener
{
struct
file
*
file
;
struct
file
*
file
;
struct
list_head
lh
;
struct
list_head
lh
;
int
c
ount
;
/* number of threads in close_pdeo() */
int
c
losing
;
struct
mutex
mutex
;
struct
completion
*
c
;
};
};
ssize_t
__proc_file_read
(
struct
file
*
,
char
__user
*
,
size_t
,
loff_t
*
);
ssize_t
__proc_file_read
(
struct
file
*
,
char
__user
*
,
size_t
,
loff_t
*
);
...
...
include/linux/proc_fs.h
View file @
05c0ae21
...
@@ -65,7 +65,7 @@ struct proc_dir_entry {
...
@@ -65,7 +65,7 @@ struct proc_dir_entry {
void
*
data
;
void
*
data
;
read_proc_t
*
read_proc
;
read_proc_t
*
read_proc
;
atomic_t
count
;
/* use count */
atomic_t
count
;
/* use count */
int
pde_users
;
/* number of callers into module in progress; */
atomic_t
in_use
;
/* number of callers into module in progress; */
/* negative -> it's going away RSN */
/* negative -> it's going away RSN */
struct
completion
*
pde_unload_completion
;
struct
completion
*
pde_unload_completion
;
struct
list_head
pde_openers
;
/* who did ->open, but not ->release */
struct
list_head
pde_openers
;
/* who did ->open, but not ->release */
...
...
Kirill Smelkov
@kirr
mentioned in commit
492b2da6
·
Feb 26, 2017
mentioned in commit
492b2da6
mentioned in commit 492b2da6056e7051917516368e75e062422c3557
Toggle commit list
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