diff options
author | Al Viro <viro@zeniv.linux.org.uk> | 2011-08-02 21:32:13 -0400 |
---|---|---|
committer | Al Viro <viro@zeniv.linux.org.uk> | 2011-08-03 00:58:42 -0400 |
commit | 3567866bf26190d1e734c975c907eb06e923ba23 (patch) | |
tree | 60d6b71d8ec821b121e6ab0756833d79c912908e /include/linux/posix_acl.h | |
parent | 951c0d660a7c35286e401ca6d6ef38c9d49643c7 (diff) |
RCUify freeing acls, let check_acl() go ahead in RCU mode if acl is cached
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'include/linux/posix_acl.h')
-rw-r--r-- | include/linux/posix_acl.h | 18 |
1 files changed, 9 insertions, 9 deletions
diff --git a/include/linux/posix_acl.h b/include/linux/posix_acl.h index a9c2fb29be96..b7681102a4b9 100644 --- a/include/linux/posix_acl.h +++ b/include/linux/posix_acl.h | |||
@@ -9,6 +9,7 @@ | |||
9 | #define __LINUX_POSIX_ACL_H | 9 | #define __LINUX_POSIX_ACL_H |
10 | 10 | ||
11 | #include <linux/slab.h> | 11 | #include <linux/slab.h> |
12 | #include <linux/rcupdate.h> | ||
12 | 13 | ||
13 | #define ACL_UNDEFINED_ID (-1) | 14 | #define ACL_UNDEFINED_ID (-1) |
14 | 15 | ||
@@ -38,7 +39,10 @@ struct posix_acl_entry { | |||
38 | }; | 39 | }; |
39 | 40 | ||
40 | struct posix_acl { | 41 | struct posix_acl { |
41 | atomic_t a_refcount; | 42 | union { |
43 | atomic_t a_refcount; | ||
44 | struct rcu_head a_rcu; | ||
45 | }; | ||
42 | unsigned int a_count; | 46 | unsigned int a_count; |
43 | struct posix_acl_entry a_entries[0]; | 47 | struct posix_acl_entry a_entries[0]; |
44 | }; | 48 | }; |
@@ -65,7 +69,7 @@ static inline void | |||
65 | posix_acl_release(struct posix_acl *acl) | 69 | posix_acl_release(struct posix_acl *acl) |
66 | { | 70 | { |
67 | if (acl && atomic_dec_and_test(&acl->a_refcount)) | 71 | if (acl && atomic_dec_and_test(&acl->a_refcount)) |
68 | kfree(acl); | 72 | kfree_rcu(acl, a_rcu); |
69 | } | 73 | } |
70 | 74 | ||
71 | 75 | ||
@@ -110,13 +114,9 @@ static inline struct posix_acl *get_cached_acl(struct inode *inode, int type) | |||
110 | return acl; | 114 | return acl; |
111 | } | 115 | } |
112 | 116 | ||
113 | static inline int negative_cached_acl(struct inode *inode, int type) | 117 | static inline struct posix_acl *get_cached_acl_rcu(struct inode *inode, int type) |
114 | { | 118 | { |
115 | struct posix_acl **p = acl_by_type(inode, type); | 119 | return rcu_dereference(*acl_by_type(inode, type)); |
116 | struct posix_acl *acl = ACCESS_ONCE(*p); | ||
117 | if (acl) | ||
118 | return 0; | ||
119 | return 1; | ||
120 | } | 120 | } |
121 | 121 | ||
122 | static inline void set_cached_acl(struct inode *inode, | 122 | static inline void set_cached_acl(struct inode *inode, |
@@ -127,7 +127,7 @@ static inline void set_cached_acl(struct inode *inode, | |||
127 | struct posix_acl *old; | 127 | struct posix_acl *old; |
128 | spin_lock(&inode->i_lock); | 128 | spin_lock(&inode->i_lock); |
129 | old = *p; | 129 | old = *p; |
130 | *p = posix_dup_acl(acl); | 130 | rcu_assign_pointer(*p, posix_acl_dup(acl)); |
131 | spin_unlock(&inode->i_lock); | 131 | spin_unlock(&inode->i_lock); |
132 | if (old != ACL_NOT_CACHED) | 132 | if (old != ACL_NOT_CACHED) |
133 | posix_acl_release(old); | 133 | posix_acl_release(old); |