diff options
Diffstat (limited to 'fs/autofs4')
-rw-r--r-- | fs/autofs4/autofs_i.h | 54 | ||||
-rw-r--r-- | fs/autofs4/expire.c | 292 | ||||
-rw-r--r-- | fs/autofs4/inode.c | 103 | ||||
-rw-r--r-- | fs/autofs4/root.c | 328 | ||||
-rw-r--r-- | fs/autofs4/waitq.c | 111 |
5 files changed, 570 insertions, 318 deletions
diff --git a/fs/autofs4/autofs_i.h b/fs/autofs4/autofs_i.h index f54c5b21f876..57c4903614e5 100644 --- a/fs/autofs4/autofs_i.h +++ b/fs/autofs4/autofs_i.h | |||
@@ -3,6 +3,7 @@ | |||
3 | * linux/fs/autofs/autofs_i.h | 3 | * linux/fs/autofs/autofs_i.h |
4 | * | 4 | * |
5 | * Copyright 1997-1998 Transmeta Corporation - All Rights Reserved | 5 | * Copyright 1997-1998 Transmeta Corporation - All Rights Reserved |
6 | * Copyright 2005-2006 Ian Kent <raven@themaw.net> | ||
6 | * | 7 | * |
7 | * This file is part of the Linux kernel and is made available under | 8 | * This file is part of the Linux kernel and is made available under |
8 | * the terms of the GNU General Public License, version 2, or at your | 9 | * the terms of the GNU General Public License, version 2, or at your |
@@ -41,14 +42,6 @@ | |||
41 | 42 | ||
42 | #define AUTOFS_SUPER_MAGIC 0x0187 | 43 | #define AUTOFS_SUPER_MAGIC 0x0187 |
43 | 44 | ||
44 | /* | ||
45 | * If the daemon returns a negative response (AUTOFS_IOC_FAIL) then the | ||
46 | * kernel will keep the negative response cached for up to the time given | ||
47 | * here, although the time can be shorter if the kernel throws the dcache | ||
48 | * entry away. This probably should be settable from user space. | ||
49 | */ | ||
50 | #define AUTOFS_NEGATIVE_TIMEOUT (60*HZ) /* 1 minute */ | ||
51 | |||
52 | /* Unified info structure. This is pointed to by both the dentry and | 45 | /* Unified info structure. This is pointed to by both the dentry and |
53 | inode structures. Each file in the filesystem has an instance of this | 46 | inode structures. Each file in the filesystem has an instance of this |
54 | structure. It holds a reference to the dentry, so dentries are never | 47 | structure. It holds a reference to the dentry, so dentries are never |
@@ -63,6 +56,7 @@ struct autofs_info { | |||
63 | 56 | ||
64 | struct autofs_sb_info *sbi; | 57 | struct autofs_sb_info *sbi; |
65 | unsigned long last_used; | 58 | unsigned long last_used; |
59 | atomic_t count; | ||
66 | 60 | ||
67 | mode_t mode; | 61 | mode_t mode; |
68 | size_t size; | 62 | size_t size; |
@@ -83,23 +77,37 @@ struct autofs_wait_queue { | |||
83 | int hash; | 77 | int hash; |
84 | int len; | 78 | int len; |
85 | char *name; | 79 | char *name; |
80 | u32 dev; | ||
81 | u64 ino; | ||
82 | uid_t uid; | ||
83 | gid_t gid; | ||
84 | pid_t pid; | ||
85 | pid_t tgid; | ||
86 | /* This is for status reporting upon return */ | 86 | /* This is for status reporting upon return */ |
87 | int status; | 87 | int status; |
88 | atomic_t notified; | 88 | atomic_t notify; |
89 | atomic_t wait_ctr; | 89 | atomic_t wait_ctr; |
90 | }; | 90 | }; |
91 | 91 | ||
92 | #define AUTOFS_SBI_MAGIC 0x6d4a556d | 92 | #define AUTOFS_SBI_MAGIC 0x6d4a556d |
93 | 93 | ||
94 | #define AUTOFS_TYPE_INDIRECT 0x0001 | ||
95 | #define AUTOFS_TYPE_DIRECT 0x0002 | ||
96 | #define AUTOFS_TYPE_OFFSET 0x0004 | ||
97 | |||
94 | struct autofs_sb_info { | 98 | struct autofs_sb_info { |
95 | u32 magic; | 99 | u32 magic; |
96 | struct dentry *root; | 100 | struct dentry *root; |
101 | int pipefd; | ||
97 | struct file *pipe; | 102 | struct file *pipe; |
98 | pid_t oz_pgrp; | 103 | pid_t oz_pgrp; |
99 | int catatonic; | 104 | int catatonic; |
100 | int version; | 105 | int version; |
101 | int sub_version; | 106 | int sub_version; |
107 | int min_proto; | ||
108 | int max_proto; | ||
102 | unsigned long exp_timeout; | 109 | unsigned long exp_timeout; |
110 | unsigned int type; | ||
103 | int reghost_enabled; | 111 | int reghost_enabled; |
104 | int needs_reghost; | 112 | int needs_reghost; |
105 | struct super_block *sb; | 113 | struct super_block *sb; |
@@ -166,8 +174,10 @@ int autofs4_expire_multi(struct super_block *, struct vfsmount *, | |||
166 | extern struct inode_operations autofs4_symlink_inode_operations; | 174 | extern struct inode_operations autofs4_symlink_inode_operations; |
167 | extern struct inode_operations autofs4_dir_inode_operations; | 175 | extern struct inode_operations autofs4_dir_inode_operations; |
168 | extern struct inode_operations autofs4_root_inode_operations; | 176 | extern struct inode_operations autofs4_root_inode_operations; |
169 | extern struct file_operations autofs4_dir_operations; | 177 | extern struct inode_operations autofs4_indirect_root_inode_operations; |
170 | extern struct file_operations autofs4_root_operations; | 178 | extern struct inode_operations autofs4_direct_root_inode_operations; |
179 | extern const struct file_operations autofs4_dir_operations; | ||
180 | extern const struct file_operations autofs4_root_operations; | ||
171 | 181 | ||
172 | /* Initializing function */ | 182 | /* Initializing function */ |
173 | 183 | ||
@@ -176,13 +186,6 @@ struct autofs_info *autofs4_init_ino(struct autofs_info *, struct autofs_sb_info | |||
176 | 186 | ||
177 | /* Queue management functions */ | 187 | /* Queue management functions */ |
178 | 188 | ||
179 | enum autofs_notify | ||
180 | { | ||
181 | NFY_NONE, | ||
182 | NFY_MOUNT, | ||
183 | NFY_EXPIRE | ||
184 | }; | ||
185 | |||
186 | int autofs4_wait(struct autofs_sb_info *,struct dentry *, enum autofs_notify); | 189 | int autofs4_wait(struct autofs_sb_info *,struct dentry *, enum autofs_notify); |
187 | int autofs4_wait_release(struct autofs_sb_info *,autofs_wqt_t,int); | 190 | int autofs4_wait_release(struct autofs_sb_info *,autofs_wqt_t,int); |
188 | void autofs4_catatonic_mode(struct autofs_sb_info *); | 191 | void autofs4_catatonic_mode(struct autofs_sb_info *); |
@@ -200,12 +203,22 @@ static inline int autofs4_follow_mount(struct vfsmount **mnt, struct dentry **de | |||
200 | return res; | 203 | return res; |
201 | } | 204 | } |
202 | 205 | ||
206 | static inline u32 autofs4_get_dev(struct autofs_sb_info *sbi) | ||
207 | { | ||
208 | return new_encode_dev(sbi->sb->s_dev); | ||
209 | } | ||
210 | |||
211 | static inline u64 autofs4_get_ino(struct autofs_sb_info *sbi) | ||
212 | { | ||
213 | return sbi->sb->s_root->d_inode->i_ino; | ||
214 | } | ||
215 | |||
203 | static inline int simple_positive(struct dentry *dentry) | 216 | static inline int simple_positive(struct dentry *dentry) |
204 | { | 217 | { |
205 | return dentry->d_inode && !d_unhashed(dentry); | 218 | return dentry->d_inode && !d_unhashed(dentry); |
206 | } | 219 | } |
207 | 220 | ||
208 | static inline int simple_empty_nolock(struct dentry *dentry) | 221 | static inline int __simple_empty(struct dentry *dentry) |
209 | { | 222 | { |
210 | struct dentry *child; | 223 | struct dentry *child; |
211 | int ret = 0; | 224 | int ret = 0; |
@@ -217,3 +230,6 @@ static inline int simple_empty_nolock(struct dentry *dentry) | |||
217 | out: | 230 | out: |
218 | return ret; | 231 | return ret; |
219 | } | 232 | } |
233 | |||
234 | void autofs4_dentry_release(struct dentry *); | ||
235 | |||
diff --git a/fs/autofs4/expire.c b/fs/autofs4/expire.c index dc39589df165..b8ce02607d66 100644 --- a/fs/autofs4/expire.c +++ b/fs/autofs4/expire.c | |||
@@ -4,7 +4,7 @@ | |||
4 | * | 4 | * |
5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved | 5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved |
6 | * Copyright 1999-2000 Jeremy Fitzhardinge <jeremy@goop.org> | 6 | * Copyright 1999-2000 Jeremy Fitzhardinge <jeremy@goop.org> |
7 | * Copyright 2001-2003 Ian Kent <raven@themaw.net> | 7 | * Copyright 2001-2006 Ian Kent <raven@themaw.net> |
8 | * | 8 | * |
9 | * This file is part of the Linux kernel and is made available under | 9 | * This file is part of the Linux kernel and is made available under |
10 | * the terms of the GNU General Public License, version 2, or at your | 10 | * the terms of the GNU General Public License, version 2, or at your |
@@ -16,7 +16,7 @@ | |||
16 | 16 | ||
17 | static unsigned long now; | 17 | static unsigned long now; |
18 | 18 | ||
19 | /* Check if a dentry can be expired return 1 if it can else return 0 */ | 19 | /* Check if a dentry can be expired */ |
20 | static inline int autofs4_can_expire(struct dentry *dentry, | 20 | static inline int autofs4_can_expire(struct dentry *dentry, |
21 | unsigned long timeout, int do_now) | 21 | unsigned long timeout, int do_now) |
22 | { | 22 | { |
@@ -41,14 +41,14 @@ static inline int autofs4_can_expire(struct dentry *dentry, | |||
41 | attempts if expire fails the first time */ | 41 | attempts if expire fails the first time */ |
42 | ino->last_used = now; | 42 | ino->last_used = now; |
43 | } | 43 | } |
44 | |||
45 | return 1; | 44 | return 1; |
46 | } | 45 | } |
47 | 46 | ||
48 | /* Check a mount point for busyness return 1 if not busy, otherwise */ | 47 | /* Check a mount point for busyness */ |
49 | static int autofs4_check_mount(struct vfsmount *mnt, struct dentry *dentry) | 48 | static int autofs4_mount_busy(struct vfsmount *mnt, struct dentry *dentry) |
50 | { | 49 | { |
51 | int status = 0; | 50 | struct dentry *top = dentry; |
51 | int status = 1; | ||
52 | 52 | ||
53 | DPRINTK("dentry %p %.*s", | 53 | DPRINTK("dentry %p %.*s", |
54 | dentry, (int)dentry->d_name.len, dentry->d_name.name); | 54 | dentry, (int)dentry->d_name.len, dentry->d_name.name); |
@@ -63,9 +63,14 @@ static int autofs4_check_mount(struct vfsmount *mnt, struct dentry *dentry) | |||
63 | if (is_autofs4_dentry(dentry)) | 63 | if (is_autofs4_dentry(dentry)) |
64 | goto done; | 64 | goto done; |
65 | 65 | ||
66 | /* The big question */ | 66 | /* Update the expiry counter if fs is busy */ |
67 | if (may_umount_tree(mnt) == 0) | 67 | if (!may_umount_tree(mnt)) { |
68 | status = 1; | 68 | struct autofs_info *ino = autofs4_dentry_ino(top); |
69 | ino->last_used = jiffies; | ||
70 | goto done; | ||
71 | } | ||
72 | |||
73 | status = 0; | ||
69 | done: | 74 | done: |
70 | DPRINTK("returning = %d", status); | 75 | DPRINTK("returning = %d", status); |
71 | mntput(mnt); | 76 | mntput(mnt); |
@@ -73,78 +78,124 @@ done: | |||
73 | return status; | 78 | return status; |
74 | } | 79 | } |
75 | 80 | ||
81 | /* | ||
82 | * Calculate next entry in top down tree traversal. | ||
83 | * From next_mnt in namespace.c - elegant. | ||
84 | */ | ||
85 | static struct dentry *next_dentry(struct dentry *p, struct dentry *root) | ||
86 | { | ||
87 | struct list_head *next = p->d_subdirs.next; | ||
88 | |||
89 | if (next == &p->d_subdirs) { | ||
90 | while (1) { | ||
91 | if (p == root) | ||
92 | return NULL; | ||
93 | next = p->d_u.d_child.next; | ||
94 | if (next != &p->d_parent->d_subdirs) | ||
95 | break; | ||
96 | p = p->d_parent; | ||
97 | } | ||
98 | } | ||
99 | return list_entry(next, struct dentry, d_u.d_child); | ||
100 | } | ||
101 | |||
102 | /* | ||
103 | * Check a direct mount point for busyness. | ||
104 | * Direct mounts have similar expiry semantics to tree mounts. | ||
105 | * The tree is not busy iff no mountpoints are busy and there are no | ||
106 | * autofs submounts. | ||
107 | */ | ||
108 | static int autofs4_direct_busy(struct vfsmount *mnt, | ||
109 | struct dentry *top, | ||
110 | unsigned long timeout, | ||
111 | int do_now) | ||
112 | { | ||
113 | DPRINTK("top %p %.*s", | ||
114 | top, (int) top->d_name.len, top->d_name.name); | ||
115 | |||
116 | /* If it's busy update the expiry counters */ | ||
117 | if (!may_umount_tree(mnt)) { | ||
118 | struct autofs_info *ino = autofs4_dentry_ino(top); | ||
119 | if (ino) | ||
120 | ino->last_used = jiffies; | ||
121 | return 1; | ||
122 | } | ||
123 | |||
124 | /* Timeout of a direct mount is determined by its top dentry */ | ||
125 | if (!autofs4_can_expire(top, timeout, do_now)) | ||
126 | return 1; | ||
127 | |||
128 | return 0; | ||
129 | } | ||
130 | |||
76 | /* Check a directory tree of mount points for busyness | 131 | /* Check a directory tree of mount points for busyness |
77 | * The tree is not busy iff no mountpoints are busy | 132 | * The tree is not busy iff no mountpoints are busy |
78 | * Return 1 if the tree is busy or 0 otherwise | ||
79 | */ | 133 | */ |
80 | static int autofs4_check_tree(struct vfsmount *mnt, | 134 | static int autofs4_tree_busy(struct vfsmount *mnt, |
81 | struct dentry *top, | 135 | struct dentry *top, |
82 | unsigned long timeout, | 136 | unsigned long timeout, |
83 | int do_now) | 137 | int do_now) |
84 | { | 138 | { |
85 | struct dentry *this_parent = top; | 139 | struct autofs_info *top_ino = autofs4_dentry_ino(top); |
86 | struct list_head *next; | 140 | struct dentry *p; |
87 | 141 | ||
88 | DPRINTK("parent %p %.*s", | 142 | DPRINTK("top %p %.*s", |
89 | top, (int)top->d_name.len, top->d_name.name); | 143 | top, (int)top->d_name.len, top->d_name.name); |
90 | 144 | ||
91 | /* Negative dentry - give up */ | 145 | /* Negative dentry - give up */ |
92 | if (!simple_positive(top)) | 146 | if (!simple_positive(top)) |
93 | return 0; | 147 | return 1; |
94 | |||
95 | /* Timeout of a tree mount is determined by its top dentry */ | ||
96 | if (!autofs4_can_expire(top, timeout, do_now)) | ||
97 | return 0; | ||
98 | |||
99 | /* Is someone visiting anywhere in the tree ? */ | ||
100 | if (may_umount_tree(mnt)) | ||
101 | return 0; | ||
102 | 148 | ||
103 | spin_lock(&dcache_lock); | 149 | spin_lock(&dcache_lock); |
104 | repeat: | 150 | for (p = top; p; p = next_dentry(p, top)) { |
105 | next = this_parent->d_subdirs.next; | ||
106 | resume: | ||
107 | while (next != &this_parent->d_subdirs) { | ||
108 | struct dentry *dentry = list_entry(next, struct dentry, d_u.d_child); | ||
109 | |||
110 | /* Negative dentry - give up */ | 151 | /* Negative dentry - give up */ |
111 | if (!simple_positive(dentry)) { | 152 | if (!simple_positive(p)) |
112 | next = next->next; | ||
113 | continue; | 153 | continue; |
114 | } | ||
115 | 154 | ||
116 | DPRINTK("dentry %p %.*s", | 155 | DPRINTK("dentry %p %.*s", |
117 | dentry, (int)dentry->d_name.len, dentry->d_name.name); | 156 | p, (int) p->d_name.len, p->d_name.name); |
118 | |||
119 | if (!simple_empty_nolock(dentry)) { | ||
120 | this_parent = dentry; | ||
121 | goto repeat; | ||
122 | } | ||
123 | 157 | ||
124 | dentry = dget(dentry); | 158 | p = dget(p); |
125 | spin_unlock(&dcache_lock); | 159 | spin_unlock(&dcache_lock); |
126 | 160 | ||
127 | if (d_mountpoint(dentry)) { | 161 | /* |
128 | /* First busy => tree busy */ | 162 | * Is someone visiting anywhere in the subtree ? |
129 | if (!autofs4_check_mount(mnt, dentry)) { | 163 | * If there's no mount we need to check the usage |
130 | dput(dentry); | 164 | * count for the autofs dentry. |
131 | return 0; | 165 | * If the fs is busy update the expiry counter. |
166 | */ | ||
167 | if (d_mountpoint(p)) { | ||
168 | if (autofs4_mount_busy(mnt, p)) { | ||
169 | top_ino->last_used = jiffies; | ||
170 | dput(p); | ||
171 | return 1; | ||
172 | } | ||
173 | } else { | ||
174 | struct autofs_info *ino = autofs4_dentry_ino(p); | ||
175 | unsigned int ino_count = atomic_read(&ino->count); | ||
176 | |||
177 | /* allow for dget above and top is already dgot */ | ||
178 | if (p == top) | ||
179 | ino_count += 2; | ||
180 | else | ||
181 | ino_count++; | ||
182 | |||
183 | if (atomic_read(&p->d_count) > ino_count) { | ||
184 | top_ino->last_used = jiffies; | ||
185 | dput(p); | ||
186 | return 1; | ||
132 | } | 187 | } |
133 | } | 188 | } |
134 | 189 | dput(p); | |
135 | dput(dentry); | ||
136 | spin_lock(&dcache_lock); | 190 | spin_lock(&dcache_lock); |
137 | next = next->next; | ||
138 | } | ||
139 | |||
140 | if (this_parent != top) { | ||
141 | next = this_parent->d_u.d_child.next; | ||
142 | this_parent = this_parent->d_parent; | ||
143 | goto resume; | ||
144 | } | 191 | } |
145 | spin_unlock(&dcache_lock); | 192 | spin_unlock(&dcache_lock); |
146 | 193 | ||
147 | return 1; | 194 | /* Timeout of a tree mount is ultimately determined by its top dentry */ |
195 | if (!autofs4_can_expire(top, timeout, do_now)) | ||
196 | return 1; | ||
197 | |||
198 | return 0; | ||
148 | } | 199 | } |
149 | 200 | ||
150 | static struct dentry *autofs4_check_leaves(struct vfsmount *mnt, | 201 | static struct dentry *autofs4_check_leaves(struct vfsmount *mnt, |
@@ -152,58 +203,68 @@ static struct dentry *autofs4_check_leaves(struct vfsmount *mnt, | |||
152 | unsigned long timeout, | 203 | unsigned long timeout, |
153 | int do_now) | 204 | int do_now) |
154 | { | 205 | { |
155 | struct dentry *this_parent = parent; | 206 | struct dentry *p; |
156 | struct list_head *next; | ||
157 | 207 | ||
158 | DPRINTK("parent %p %.*s", | 208 | DPRINTK("parent %p %.*s", |
159 | parent, (int)parent->d_name.len, parent->d_name.name); | 209 | parent, (int)parent->d_name.len, parent->d_name.name); |
160 | 210 | ||
161 | spin_lock(&dcache_lock); | 211 | spin_lock(&dcache_lock); |
162 | repeat: | 212 | for (p = parent; p; p = next_dentry(p, parent)) { |
163 | next = this_parent->d_subdirs.next; | ||
164 | resume: | ||
165 | while (next != &this_parent->d_subdirs) { | ||
166 | struct dentry *dentry = list_entry(next, struct dentry, d_u.d_child); | ||
167 | |||
168 | /* Negative dentry - give up */ | 213 | /* Negative dentry - give up */ |
169 | if (!simple_positive(dentry)) { | 214 | if (!simple_positive(p)) |
170 | next = next->next; | ||
171 | continue; | 215 | continue; |
172 | } | ||
173 | 216 | ||
174 | DPRINTK("dentry %p %.*s", | 217 | DPRINTK("dentry %p %.*s", |
175 | dentry, (int)dentry->d_name.len, dentry->d_name.name); | 218 | p, (int) p->d_name.len, p->d_name.name); |
176 | |||
177 | if (!list_empty(&dentry->d_subdirs)) { | ||
178 | this_parent = dentry; | ||
179 | goto repeat; | ||
180 | } | ||
181 | 219 | ||
182 | dentry = dget(dentry); | 220 | p = dget(p); |
183 | spin_unlock(&dcache_lock); | 221 | spin_unlock(&dcache_lock); |
184 | 222 | ||
185 | if (d_mountpoint(dentry)) { | 223 | if (d_mountpoint(p)) { |
186 | /* Can we expire this guy */ | ||
187 | if (!autofs4_can_expire(dentry, timeout, do_now)) | ||
188 | goto cont; | ||
189 | |||
190 | /* Can we umount this guy */ | 224 | /* Can we umount this guy */ |
191 | if (autofs4_check_mount(mnt, dentry)) | 225 | if (autofs4_mount_busy(mnt, p)) |
192 | return dentry; | 226 | goto cont; |
193 | 227 | ||
228 | /* Can we expire this guy */ | ||
229 | if (autofs4_can_expire(p, timeout, do_now)) | ||
230 | return p; | ||
194 | } | 231 | } |
195 | cont: | 232 | cont: |
196 | dput(dentry); | 233 | dput(p); |
197 | spin_lock(&dcache_lock); | 234 | spin_lock(&dcache_lock); |
198 | next = next->next; | ||
199 | } | 235 | } |
236 | spin_unlock(&dcache_lock); | ||
237 | return NULL; | ||
238 | } | ||
239 | |||
240 | /* Check if we can expire a direct mount (possibly a tree) */ | ||
241 | static struct dentry *autofs4_expire_direct(struct super_block *sb, | ||
242 | struct vfsmount *mnt, | ||
243 | struct autofs_sb_info *sbi, | ||
244 | int how) | ||
245 | { | ||
246 | unsigned long timeout; | ||
247 | struct dentry *root = dget(sb->s_root); | ||
248 | int do_now = how & AUTOFS_EXP_IMMEDIATE; | ||
249 | |||
250 | if (!sbi->exp_timeout || !root) | ||
251 | return NULL; | ||
252 | |||
253 | now = jiffies; | ||
254 | timeout = sbi->exp_timeout; | ||
255 | |||
256 | /* Lock the tree as we must expire as a whole */ | ||
257 | spin_lock(&sbi->fs_lock); | ||
258 | if (!autofs4_direct_busy(mnt, root, timeout, do_now)) { | ||
259 | struct autofs_info *ino = autofs4_dentry_ino(root); | ||
200 | 260 | ||
201 | if (this_parent != parent) { | 261 | /* Set this flag early to catch sys_chdir and the like */ |
202 | next = this_parent->d_u.d_child.next; | 262 | ino->flags |= AUTOFS_INF_EXPIRING; |
203 | this_parent = this_parent->d_parent; | 263 | spin_unlock(&sbi->fs_lock); |
204 | goto resume; | 264 | return root; |
205 | } | 265 | } |
206 | spin_unlock(&dcache_lock); | 266 | spin_unlock(&sbi->fs_lock); |
267 | dput(root); | ||
207 | 268 | ||
208 | return NULL; | 269 | return NULL; |
209 | } | 270 | } |
@@ -214,10 +275,10 @@ cont: | |||
214 | * - it is unused by any user process | 275 | * - it is unused by any user process |
215 | * - it has been unused for exp_timeout time | 276 | * - it has been unused for exp_timeout time |
216 | */ | 277 | */ |
217 | static struct dentry *autofs4_expire(struct super_block *sb, | 278 | static struct dentry *autofs4_expire_indirect(struct super_block *sb, |
218 | struct vfsmount *mnt, | 279 | struct vfsmount *mnt, |
219 | struct autofs_sb_info *sbi, | 280 | struct autofs_sb_info *sbi, |
220 | int how) | 281 | int how) |
221 | { | 282 | { |
222 | unsigned long timeout; | 283 | unsigned long timeout; |
223 | struct dentry *root = sb->s_root; | 284 | struct dentry *root = sb->s_root; |
@@ -241,7 +302,7 @@ static struct dentry *autofs4_expire(struct super_block *sb, | |||
241 | struct dentry *dentry = list_entry(next, struct dentry, d_u.d_child); | 302 | struct dentry *dentry = list_entry(next, struct dentry, d_u.d_child); |
242 | 303 | ||
243 | /* Negative dentry - give up */ | 304 | /* Negative dentry - give up */ |
244 | if ( !simple_positive(dentry) ) { | 305 | if (!simple_positive(dentry)) { |
245 | next = next->next; | 306 | next = next->next; |
246 | continue; | 307 | continue; |
247 | } | 308 | } |
@@ -249,31 +310,36 @@ static struct dentry *autofs4_expire(struct super_block *sb, | |||
249 | dentry = dget(dentry); | 310 | dentry = dget(dentry); |
250 | spin_unlock(&dcache_lock); | 311 | spin_unlock(&dcache_lock); |
251 | 312 | ||
252 | /* Case 1: indirect mount or top level direct mount */ | 313 | /* |
314 | * Case 1: (i) indirect mount or top level pseudo direct mount | ||
315 | * (autofs-4.1). | ||
316 | * (ii) indirect mount with offset mount, check the "/" | ||
317 | * offset (autofs-5.0+). | ||
318 | */ | ||
253 | if (d_mountpoint(dentry)) { | 319 | if (d_mountpoint(dentry)) { |
254 | DPRINTK("checking mountpoint %p %.*s", | 320 | DPRINTK("checking mountpoint %p %.*s", |
255 | dentry, (int)dentry->d_name.len, dentry->d_name.name); | 321 | dentry, (int)dentry->d_name.len, dentry->d_name.name); |
256 | 322 | ||
257 | /* Can we expire this guy */ | 323 | /* Can we umount this guy */ |
258 | if (!autofs4_can_expire(dentry, timeout, do_now)) | 324 | if (autofs4_mount_busy(mnt, dentry)) |
259 | goto next; | 325 | goto next; |
260 | 326 | ||
261 | /* Can we umount this guy */ | 327 | /* Can we expire this guy */ |
262 | if (autofs4_check_mount(mnt, dentry)) { | 328 | if (autofs4_can_expire(dentry, timeout, do_now)) { |
263 | expired = dentry; | 329 | expired = dentry; |
264 | break; | 330 | break; |
265 | } | 331 | } |
266 | goto next; | 332 | goto next; |
267 | } | 333 | } |
268 | 334 | ||
269 | if ( simple_empty(dentry) ) | 335 | if (simple_empty(dentry)) |
270 | goto next; | 336 | goto next; |
271 | 337 | ||
272 | /* Case 2: tree mount, expire iff entire tree is not busy */ | 338 | /* Case 2: tree mount, expire iff entire tree is not busy */ |
273 | if (!exp_leaves) { | 339 | if (!exp_leaves) { |
274 | /* Lock the tree as we must expire as a whole */ | 340 | /* Lock the tree as we must expire as a whole */ |
275 | spin_lock(&sbi->fs_lock); | 341 | spin_lock(&sbi->fs_lock); |
276 | if (autofs4_check_tree(mnt, dentry, timeout, do_now)) { | 342 | if (!autofs4_tree_busy(mnt, dentry, timeout, do_now)) { |
277 | struct autofs_info *inf = autofs4_dentry_ino(dentry); | 343 | struct autofs_info *inf = autofs4_dentry_ino(dentry); |
278 | 344 | ||
279 | /* Set this flag early to catch sys_chdir and the like */ | 345 | /* Set this flag early to catch sys_chdir and the like */ |
@@ -283,7 +349,10 @@ static struct dentry *autofs4_expire(struct super_block *sb, | |||
283 | break; | 349 | break; |
284 | } | 350 | } |
285 | spin_unlock(&sbi->fs_lock); | 351 | spin_unlock(&sbi->fs_lock); |
286 | /* Case 3: direct mount, expire individual leaves */ | 352 | /* |
353 | * Case 3: pseudo direct mount, expire individual leaves | ||
354 | * (autofs-4.1). | ||
355 | */ | ||
287 | } else { | 356 | } else { |
288 | expired = autofs4_check_leaves(mnt, dentry, timeout, do_now); | 357 | expired = autofs4_check_leaves(mnt, dentry, timeout, do_now); |
289 | if (expired) { | 358 | if (expired) { |
@@ -297,7 +366,7 @@ next: | |||
297 | next = next->next; | 366 | next = next->next; |
298 | } | 367 | } |
299 | 368 | ||
300 | if ( expired ) { | 369 | if (expired) { |
301 | DPRINTK("returning %p %.*s", | 370 | DPRINTK("returning %p %.*s", |
302 | expired, (int)expired->d_name.len, expired->d_name.name); | 371 | expired, (int)expired->d_name.len, expired->d_name.name); |
303 | spin_lock(&dcache_lock); | 372 | spin_lock(&dcache_lock); |
@@ -325,7 +394,7 @@ int autofs4_expire_run(struct super_block *sb, | |||
325 | pkt.hdr.proto_version = sbi->version; | 394 | pkt.hdr.proto_version = sbi->version; |
326 | pkt.hdr.type = autofs_ptype_expire; | 395 | pkt.hdr.type = autofs_ptype_expire; |
327 | 396 | ||
328 | if ((dentry = autofs4_expire(sb, mnt, sbi, 0)) == NULL) | 397 | if ((dentry = autofs4_expire_indirect(sb, mnt, sbi, 0)) == NULL) |
329 | return -EAGAIN; | 398 | return -EAGAIN; |
330 | 399 | ||
331 | pkt.len = dentry->d_name.len; | 400 | pkt.len = dentry->d_name.len; |
@@ -351,17 +420,22 @@ int autofs4_expire_multi(struct super_block *sb, struct vfsmount *mnt, | |||
351 | if (arg && get_user(do_now, arg)) | 420 | if (arg && get_user(do_now, arg)) |
352 | return -EFAULT; | 421 | return -EFAULT; |
353 | 422 | ||
354 | if ((dentry = autofs4_expire(sb, mnt, sbi, do_now)) != NULL) { | 423 | if (sbi->type & AUTOFS_TYPE_DIRECT) |
355 | struct autofs_info *de_info = autofs4_dentry_ino(dentry); | 424 | dentry = autofs4_expire_direct(sb, mnt, sbi, do_now); |
425 | else | ||
426 | dentry = autofs4_expire_indirect(sb, mnt, sbi, do_now); | ||
427 | |||
428 | if (dentry) { | ||
429 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | ||
356 | 430 | ||
357 | /* This is synchronous because it makes the daemon a | 431 | /* This is synchronous because it makes the daemon a |
358 | little easier */ | 432 | little easier */ |
359 | de_info->flags |= AUTOFS_INF_EXPIRING; | 433 | ino->flags |= AUTOFS_INF_EXPIRING; |
360 | ret = autofs4_wait(sbi, dentry, NFY_EXPIRE); | 434 | ret = autofs4_wait(sbi, dentry, NFY_EXPIRE); |
361 | de_info->flags &= ~AUTOFS_INF_EXPIRING; | 435 | ino->flags &= ~AUTOFS_INF_EXPIRING; |
362 | dput(dentry); | 436 | dput(dentry); |
363 | } | 437 | } |
364 | 438 | ||
365 | return ret; | 439 | return ret; |
366 | } | 440 | } |
367 | 441 | ||
diff --git a/fs/autofs4/inode.c b/fs/autofs4/inode.c index 1ad98d48e550..fde78b110ddd 100644 --- a/fs/autofs4/inode.c +++ b/fs/autofs4/inode.c | |||
@@ -3,6 +3,7 @@ | |||
3 | * linux/fs/autofs/inode.c | 3 | * linux/fs/autofs/inode.c |
4 | * | 4 | * |
5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved | 5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved |
6 | * Copyright 2005-2006 Ian Kent <raven@themaw.net> | ||
6 | * | 7 | * |
7 | * This file is part of the Linux kernel and is made available under | 8 | * This file is part of the Linux kernel and is made available under |
8 | * the terms of the GNU General Public License, version 2, or at your | 9 | * the terms of the GNU General Public License, version 2, or at your |
@@ -13,6 +14,7 @@ | |||
13 | #include <linux/kernel.h> | 14 | #include <linux/kernel.h> |
14 | #include <linux/slab.h> | 15 | #include <linux/slab.h> |
15 | #include <linux/file.h> | 16 | #include <linux/file.h> |
17 | #include <linux/seq_file.h> | ||
16 | #include <linux/pagemap.h> | 18 | #include <linux/pagemap.h> |
17 | #include <linux/parser.h> | 19 | #include <linux/parser.h> |
18 | #include <linux/bitops.h> | 20 | #include <linux/bitops.h> |
@@ -46,6 +48,7 @@ struct autofs_info *autofs4_init_ino(struct autofs_info *ino, | |||
46 | ino->size = 0; | 48 | ino->size = 0; |
47 | 49 | ||
48 | ino->last_used = jiffies; | 50 | ino->last_used = jiffies; |
51 | atomic_set(&ino->count, 0); | ||
49 | 52 | ||
50 | ino->sbi = sbi; | 53 | ino->sbi = sbi; |
51 | 54 | ||
@@ -64,10 +67,19 @@ struct autofs_info *autofs4_init_ino(struct autofs_info *ino, | |||
64 | 67 | ||
65 | void autofs4_free_ino(struct autofs_info *ino) | 68 | void autofs4_free_ino(struct autofs_info *ino) |
66 | { | 69 | { |
70 | struct autofs_info *p_ino; | ||
71 | |||
67 | if (ino->dentry) { | 72 | if (ino->dentry) { |
68 | ino->dentry->d_fsdata = NULL; | 73 | ino->dentry->d_fsdata = NULL; |
69 | if (ino->dentry->d_inode) | 74 | if (ino->dentry->d_inode) { |
75 | struct dentry *parent = ino->dentry->d_parent; | ||
76 | if (atomic_dec_and_test(&ino->count)) { | ||
77 | p_ino = autofs4_dentry_ino(parent); | ||
78 | if (p_ino && parent != ino->dentry) | ||
79 | atomic_dec(&p_ino->count); | ||
80 | } | ||
70 | dput(ino->dentry); | 81 | dput(ino->dentry); |
82 | } | ||
71 | ino->dentry = NULL; | 83 | ino->dentry = NULL; |
72 | } | 84 | } |
73 | if (ino->free) | 85 | if (ino->free) |
@@ -145,20 +157,44 @@ static void autofs4_put_super(struct super_block *sb) | |||
145 | autofs4_catatonic_mode(sbi); /* Free wait queues, close pipe */ | 157 | autofs4_catatonic_mode(sbi); /* Free wait queues, close pipe */ |
146 | 158 | ||
147 | /* Clean up and release dangling references */ | 159 | /* Clean up and release dangling references */ |
148 | if (sbi) | 160 | autofs4_force_release(sbi); |
149 | autofs4_force_release(sbi); | ||
150 | 161 | ||
151 | kfree(sbi); | 162 | kfree(sbi); |
152 | 163 | ||
153 | DPRINTK("shutting down"); | 164 | DPRINTK("shutting down"); |
154 | } | 165 | } |
155 | 166 | ||
167 | static int autofs4_show_options(struct seq_file *m, struct vfsmount *mnt) | ||
168 | { | ||
169 | struct autofs_sb_info *sbi = autofs4_sbi(mnt->mnt_sb); | ||
170 | |||
171 | if (!sbi) | ||
172 | return 0; | ||
173 | |||
174 | seq_printf(m, ",fd=%d", sbi->pipefd); | ||
175 | seq_printf(m, ",pgrp=%d", sbi->oz_pgrp); | ||
176 | seq_printf(m, ",timeout=%lu", sbi->exp_timeout/HZ); | ||
177 | seq_printf(m, ",minproto=%d", sbi->min_proto); | ||
178 | seq_printf(m, ",maxproto=%d", sbi->max_proto); | ||
179 | |||
180 | if (sbi->type & AUTOFS_TYPE_OFFSET) | ||
181 | seq_printf(m, ",offset"); | ||
182 | else if (sbi->type & AUTOFS_TYPE_DIRECT) | ||
183 | seq_printf(m, ",direct"); | ||
184 | else | ||
185 | seq_printf(m, ",indirect"); | ||
186 | |||
187 | return 0; | ||
188 | } | ||
189 | |||
156 | static struct super_operations autofs4_sops = { | 190 | static struct super_operations autofs4_sops = { |
157 | .put_super = autofs4_put_super, | 191 | .put_super = autofs4_put_super, |
158 | .statfs = simple_statfs, | 192 | .statfs = simple_statfs, |
193 | .show_options = autofs4_show_options, | ||
159 | }; | 194 | }; |
160 | 195 | ||
161 | enum {Opt_err, Opt_fd, Opt_uid, Opt_gid, Opt_pgrp, Opt_minproto, Opt_maxproto}; | 196 | enum {Opt_err, Opt_fd, Opt_uid, Opt_gid, Opt_pgrp, Opt_minproto, Opt_maxproto, |
197 | Opt_indirect, Opt_direct, Opt_offset}; | ||
162 | 198 | ||
163 | static match_table_t tokens = { | 199 | static match_table_t tokens = { |
164 | {Opt_fd, "fd=%u"}, | 200 | {Opt_fd, "fd=%u"}, |
@@ -167,11 +203,15 @@ static match_table_t tokens = { | |||
167 | {Opt_pgrp, "pgrp=%u"}, | 203 | {Opt_pgrp, "pgrp=%u"}, |
168 | {Opt_minproto, "minproto=%u"}, | 204 | {Opt_minproto, "minproto=%u"}, |
169 | {Opt_maxproto, "maxproto=%u"}, | 205 | {Opt_maxproto, "maxproto=%u"}, |
206 | {Opt_indirect, "indirect"}, | ||
207 | {Opt_direct, "direct"}, | ||
208 | {Opt_offset, "offset"}, | ||
170 | {Opt_err, NULL} | 209 | {Opt_err, NULL} |
171 | }; | 210 | }; |
172 | 211 | ||
173 | static int parse_options(char *options, int *pipefd, uid_t *uid, gid_t *gid, | 212 | static int parse_options(char *options, int *pipefd, uid_t *uid, gid_t *gid, |
174 | pid_t *pgrp, int *minproto, int *maxproto) | 213 | pid_t *pgrp, unsigned int *type, |
214 | int *minproto, int *maxproto) | ||
175 | { | 215 | { |
176 | char *p; | 216 | char *p; |
177 | substring_t args[MAX_OPT_ARGS]; | 217 | substring_t args[MAX_OPT_ARGS]; |
@@ -225,6 +265,15 @@ static int parse_options(char *options, int *pipefd, uid_t *uid, gid_t *gid, | |||
225 | return 1; | 265 | return 1; |
226 | *maxproto = option; | 266 | *maxproto = option; |
227 | break; | 267 | break; |
268 | case Opt_indirect: | ||
269 | *type = AUTOFS_TYPE_INDIRECT; | ||
270 | break; | ||
271 | case Opt_direct: | ||
272 | *type = AUTOFS_TYPE_DIRECT; | ||
273 | break; | ||
274 | case Opt_offset: | ||
275 | *type = AUTOFS_TYPE_DIRECT | AUTOFS_TYPE_OFFSET; | ||
276 | break; | ||
228 | default: | 277 | default: |
229 | return 1; | 278 | return 1; |
230 | } | 279 | } |
@@ -243,6 +292,10 @@ static struct autofs_info *autofs4_mkroot(struct autofs_sb_info *sbi) | |||
243 | return ino; | 292 | return ino; |
244 | } | 293 | } |
245 | 294 | ||
295 | static struct dentry_operations autofs4_sb_dentry_operations = { | ||
296 | .d_release = autofs4_dentry_release, | ||
297 | }; | ||
298 | |||
246 | int autofs4_fill_super(struct super_block *s, void *data, int silent) | 299 | int autofs4_fill_super(struct super_block *s, void *data, int silent) |
247 | { | 300 | { |
248 | struct inode * root_inode; | 301 | struct inode * root_inode; |
@@ -251,7 +304,6 @@ int autofs4_fill_super(struct super_block *s, void *data, int silent) | |||
251 | int pipefd; | 304 | int pipefd; |
252 | struct autofs_sb_info *sbi; | 305 | struct autofs_sb_info *sbi; |
253 | struct autofs_info *ino; | 306 | struct autofs_info *ino; |
254 | int minproto, maxproto; | ||
255 | 307 | ||
256 | sbi = (struct autofs_sb_info *) kmalloc(sizeof(*sbi), GFP_KERNEL); | 308 | sbi = (struct autofs_sb_info *) kmalloc(sizeof(*sbi), GFP_KERNEL); |
257 | if ( !sbi ) | 309 | if ( !sbi ) |
@@ -263,12 +315,16 @@ int autofs4_fill_super(struct super_block *s, void *data, int silent) | |||
263 | s->s_fs_info = sbi; | 315 | s->s_fs_info = sbi; |
264 | sbi->magic = AUTOFS_SBI_MAGIC; | 316 | sbi->magic = AUTOFS_SBI_MAGIC; |
265 | sbi->root = NULL; | 317 | sbi->root = NULL; |
318 | sbi->pipefd = -1; | ||
266 | sbi->catatonic = 0; | 319 | sbi->catatonic = 0; |
267 | sbi->exp_timeout = 0; | 320 | sbi->exp_timeout = 0; |
268 | sbi->oz_pgrp = process_group(current); | 321 | sbi->oz_pgrp = process_group(current); |
269 | sbi->sb = s; | 322 | sbi->sb = s; |
270 | sbi->version = 0; | 323 | sbi->version = 0; |
271 | sbi->sub_version = 0; | 324 | sbi->sub_version = 0; |
325 | sbi->type = 0; | ||
326 | sbi->min_proto = 0; | ||
327 | sbi->max_proto = 0; | ||
272 | mutex_init(&sbi->wq_mutex); | 328 | mutex_init(&sbi->wq_mutex); |
273 | spin_lock_init(&sbi->fs_lock); | 329 | spin_lock_init(&sbi->fs_lock); |
274 | sbi->queues = NULL; | 330 | sbi->queues = NULL; |
@@ -285,38 +341,46 @@ int autofs4_fill_super(struct super_block *s, void *data, int silent) | |||
285 | if (!ino) | 341 | if (!ino) |
286 | goto fail_free; | 342 | goto fail_free; |
287 | root_inode = autofs4_get_inode(s, ino); | 343 | root_inode = autofs4_get_inode(s, ino); |
288 | kfree(ino); | ||
289 | if (!root_inode) | 344 | if (!root_inode) |
290 | goto fail_free; | 345 | goto fail_ino; |
291 | 346 | ||
292 | root_inode->i_op = &autofs4_root_inode_operations; | ||
293 | root_inode->i_fop = &autofs4_root_operations; | ||
294 | root = d_alloc_root(root_inode); | 347 | root = d_alloc_root(root_inode); |
295 | pipe = NULL; | ||
296 | |||
297 | if (!root) | 348 | if (!root) |
298 | goto fail_iput; | 349 | goto fail_iput; |
350 | pipe = NULL; | ||
351 | |||
352 | root->d_op = &autofs4_sb_dentry_operations; | ||
353 | root->d_fsdata = ino; | ||
299 | 354 | ||
300 | /* Can this call block? */ | 355 | /* Can this call block? */ |
301 | if (parse_options(data, &pipefd, | 356 | if (parse_options(data, &pipefd, |
302 | &root_inode->i_uid, &root_inode->i_gid, | 357 | &root_inode->i_uid, &root_inode->i_gid, |
303 | &sbi->oz_pgrp, | 358 | &sbi->oz_pgrp, &sbi->type, |
304 | &minproto, &maxproto)) { | 359 | &sbi->min_proto, &sbi->max_proto)) { |
305 | printk("autofs: called with bogus options\n"); | 360 | printk("autofs: called with bogus options\n"); |
306 | goto fail_dput; | 361 | goto fail_dput; |
307 | } | 362 | } |
308 | 363 | ||
364 | root_inode->i_fop = &autofs4_root_operations; | ||
365 | root_inode->i_op = sbi->type & AUTOFS_TYPE_DIRECT ? | ||
366 | &autofs4_direct_root_inode_operations : | ||
367 | &autofs4_indirect_root_inode_operations; | ||
368 | |||
309 | /* Couldn't this be tested earlier? */ | 369 | /* Couldn't this be tested earlier? */ |
310 | if (maxproto < AUTOFS_MIN_PROTO_VERSION || | 370 | if (sbi->max_proto < AUTOFS_MIN_PROTO_VERSION || |
311 | minproto > AUTOFS_MAX_PROTO_VERSION) { | 371 | sbi->min_proto > AUTOFS_MAX_PROTO_VERSION) { |
312 | printk("autofs: kernel does not match daemon version " | 372 | printk("autofs: kernel does not match daemon version " |
313 | "daemon (%d, %d) kernel (%d, %d)\n", | 373 | "daemon (%d, %d) kernel (%d, %d)\n", |
314 | minproto, maxproto, | 374 | sbi->min_proto, sbi->max_proto, |
315 | AUTOFS_MIN_PROTO_VERSION, AUTOFS_MAX_PROTO_VERSION); | 375 | AUTOFS_MIN_PROTO_VERSION, AUTOFS_MAX_PROTO_VERSION); |
316 | goto fail_dput; | 376 | goto fail_dput; |
317 | } | 377 | } |
318 | 378 | ||
319 | sbi->version = maxproto > AUTOFS_MAX_PROTO_VERSION ? AUTOFS_MAX_PROTO_VERSION : maxproto; | 379 | /* Establish highest kernel protocol version */ |
380 | if (sbi->max_proto > AUTOFS_MAX_PROTO_VERSION) | ||
381 | sbi->version = AUTOFS_MAX_PROTO_VERSION; | ||
382 | else | ||
383 | sbi->version = sbi->max_proto; | ||
320 | sbi->sub_version = AUTOFS_PROTO_SUBVERSION; | 384 | sbi->sub_version = AUTOFS_PROTO_SUBVERSION; |
321 | 385 | ||
322 | DPRINTK("pipe fd = %d, pgrp = %u", pipefd, sbi->oz_pgrp); | 386 | DPRINTK("pipe fd = %d, pgrp = %u", pipefd, sbi->oz_pgrp); |
@@ -329,6 +393,7 @@ int autofs4_fill_super(struct super_block *s, void *data, int silent) | |||
329 | if ( !pipe->f_op || !pipe->f_op->write ) | 393 | if ( !pipe->f_op || !pipe->f_op->write ) |
330 | goto fail_fput; | 394 | goto fail_fput; |
331 | sbi->pipe = pipe; | 395 | sbi->pipe = pipe; |
396 | sbi->pipefd = pipefd; | ||
332 | 397 | ||
333 | /* | 398 | /* |
334 | * Take a reference to the root dentry so we get a chance to | 399 | * Take a reference to the root dentry so we get a chance to |
@@ -356,6 +421,8 @@ fail_dput: | |||
356 | fail_iput: | 421 | fail_iput: |
357 | printk("autofs: get root dentry failed\n"); | 422 | printk("autofs: get root dentry failed\n"); |
358 | iput(root_inode); | 423 | iput(root_inode); |
424 | fail_ino: | ||
425 | kfree(ino); | ||
359 | fail_free: | 426 | fail_free: |
360 | kfree(sbi); | 427 | kfree(sbi); |
361 | fail_unlock: | 428 | fail_unlock: |
diff --git a/fs/autofs4/root.c b/fs/autofs4/root.c index 62d8d4acb8bb..84e030c8ddd0 100644 --- a/fs/autofs4/root.c +++ b/fs/autofs4/root.c | |||
@@ -4,7 +4,7 @@ | |||
4 | * | 4 | * |
5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved | 5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved |
6 | * Copyright 1999-2000 Jeremy Fitzhardinge <jeremy@goop.org> | 6 | * Copyright 1999-2000 Jeremy Fitzhardinge <jeremy@goop.org> |
7 | * Copyright 2001-2003 Ian Kent <raven@themaw.net> | 7 | * Copyright 2001-2006 Ian Kent <raven@themaw.net> |
8 | * | 8 | * |
9 | * This file is part of the Linux kernel and is made available under | 9 | * This file is part of the Linux kernel and is made available under |
10 | * the terms of the GNU General Public License, version 2, or at your | 10 | * the terms of the GNU General Public License, version 2, or at your |
@@ -30,9 +30,9 @@ static int autofs4_dir_close(struct inode *inode, struct file *file); | |||
30 | static int autofs4_dir_readdir(struct file * filp, void * dirent, filldir_t filldir); | 30 | static int autofs4_dir_readdir(struct file * filp, void * dirent, filldir_t filldir); |
31 | static int autofs4_root_readdir(struct file * filp, void * dirent, filldir_t filldir); | 31 | static int autofs4_root_readdir(struct file * filp, void * dirent, filldir_t filldir); |
32 | static struct dentry *autofs4_lookup(struct inode *,struct dentry *, struct nameidata *); | 32 | static struct dentry *autofs4_lookup(struct inode *,struct dentry *, struct nameidata *); |
33 | static int autofs4_dcache_readdir(struct file *, void *, filldir_t); | 33 | static void *autofs4_follow_link(struct dentry *, struct nameidata *); |
34 | 34 | ||
35 | struct file_operations autofs4_root_operations = { | 35 | const struct file_operations autofs4_root_operations = { |
36 | .open = dcache_dir_open, | 36 | .open = dcache_dir_open, |
37 | .release = dcache_dir_close, | 37 | .release = dcache_dir_close, |
38 | .read = generic_read_dir, | 38 | .read = generic_read_dir, |
@@ -40,14 +40,14 @@ struct file_operations autofs4_root_operations = { | |||
40 | .ioctl = autofs4_root_ioctl, | 40 | .ioctl = autofs4_root_ioctl, |
41 | }; | 41 | }; |
42 | 42 | ||
43 | struct file_operations autofs4_dir_operations = { | 43 | const struct file_operations autofs4_dir_operations = { |
44 | .open = autofs4_dir_open, | 44 | .open = autofs4_dir_open, |
45 | .release = autofs4_dir_close, | 45 | .release = autofs4_dir_close, |
46 | .read = generic_read_dir, | 46 | .read = generic_read_dir, |
47 | .readdir = autofs4_dir_readdir, | 47 | .readdir = autofs4_dir_readdir, |
48 | }; | 48 | }; |
49 | 49 | ||
50 | struct inode_operations autofs4_root_inode_operations = { | 50 | struct inode_operations autofs4_indirect_root_inode_operations = { |
51 | .lookup = autofs4_lookup, | 51 | .lookup = autofs4_lookup, |
52 | .unlink = autofs4_dir_unlink, | 52 | .unlink = autofs4_dir_unlink, |
53 | .symlink = autofs4_dir_symlink, | 53 | .symlink = autofs4_dir_symlink, |
@@ -55,6 +55,14 @@ struct inode_operations autofs4_root_inode_operations = { | |||
55 | .rmdir = autofs4_dir_rmdir, | 55 | .rmdir = autofs4_dir_rmdir, |
56 | }; | 56 | }; |
57 | 57 | ||
58 | struct inode_operations autofs4_direct_root_inode_operations = { | ||
59 | .lookup = autofs4_lookup, | ||
60 | .unlink = autofs4_dir_unlink, | ||
61 | .mkdir = autofs4_dir_mkdir, | ||
62 | .rmdir = autofs4_dir_rmdir, | ||
63 | .follow_link = autofs4_follow_link, | ||
64 | }; | ||
65 | |||
58 | struct inode_operations autofs4_dir_inode_operations = { | 66 | struct inode_operations autofs4_dir_inode_operations = { |
59 | .lookup = autofs4_lookup, | 67 | .lookup = autofs4_lookup, |
60 | .unlink = autofs4_dir_unlink, | 68 | .unlink = autofs4_dir_unlink, |
@@ -82,87 +90,7 @@ static int autofs4_root_readdir(struct file *file, void *dirent, | |||
82 | 90 | ||
83 | DPRINTK("needs_reghost = %d", sbi->needs_reghost); | 91 | DPRINTK("needs_reghost = %d", sbi->needs_reghost); |
84 | 92 | ||
85 | return autofs4_dcache_readdir(file, dirent, filldir); | 93 | return dcache_readdir(file, dirent, filldir); |
86 | } | ||
87 | |||
88 | /* Update usage from here to top of tree, so that scan of | ||
89 | top-level directories will give a useful result */ | ||
90 | static void autofs4_update_usage(struct vfsmount *mnt, struct dentry *dentry) | ||
91 | { | ||
92 | struct dentry *top = dentry->d_sb->s_root; | ||
93 | |||
94 | spin_lock(&dcache_lock); | ||
95 | for(; dentry != top; dentry = dentry->d_parent) { | ||
96 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | ||
97 | |||
98 | if (ino) { | ||
99 | touch_atime(mnt, dentry); | ||
100 | ino->last_used = jiffies; | ||
101 | } | ||
102 | } | ||
103 | spin_unlock(&dcache_lock); | ||
104 | } | ||
105 | |||
106 | /* | ||
107 | * From 2.4 kernel readdir.c | ||
108 | */ | ||
109 | static int autofs4_dcache_readdir(struct file * filp, void * dirent, filldir_t filldir) | ||
110 | { | ||
111 | int i; | ||
112 | struct dentry *dentry = filp->f_dentry; | ||
113 | |||
114 | i = filp->f_pos; | ||
115 | switch (i) { | ||
116 | case 0: | ||
117 | if (filldir(dirent, ".", 1, i, dentry->d_inode->i_ino, DT_DIR) < 0) | ||
118 | break; | ||
119 | i++; | ||
120 | filp->f_pos++; | ||
121 | /* fallthrough */ | ||
122 | case 1: | ||
123 | if (filldir(dirent, "..", 2, i, dentry->d_parent->d_inode->i_ino, DT_DIR) < 0) | ||
124 | break; | ||
125 | i++; | ||
126 | filp->f_pos++; | ||
127 | /* fallthrough */ | ||
128 | default: { | ||
129 | struct list_head *list; | ||
130 | int j = i-2; | ||
131 | |||
132 | spin_lock(&dcache_lock); | ||
133 | list = dentry->d_subdirs.next; | ||
134 | |||
135 | for (;;) { | ||
136 | if (list == &dentry->d_subdirs) { | ||
137 | spin_unlock(&dcache_lock); | ||
138 | return 0; | ||
139 | } | ||
140 | if (!j) | ||
141 | break; | ||
142 | j--; | ||
143 | list = list->next; | ||
144 | } | ||
145 | |||
146 | while(1) { | ||
147 | struct dentry *de = list_entry(list, | ||
148 | struct dentry, d_u.d_child); | ||
149 | |||
150 | if (!d_unhashed(de) && de->d_inode) { | ||
151 | spin_unlock(&dcache_lock); | ||
152 | if (filldir(dirent, de->d_name.name, de->d_name.len, filp->f_pos, de->d_inode->i_ino, DT_UNKNOWN) < 0) | ||
153 | break; | ||
154 | spin_lock(&dcache_lock); | ||
155 | } | ||
156 | filp->f_pos++; | ||
157 | list = list->next; | ||
158 | if (list != &dentry->d_subdirs) | ||
159 | continue; | ||
160 | spin_unlock(&dcache_lock); | ||
161 | break; | ||
162 | } | ||
163 | } | ||
164 | } | ||
165 | return 0; | ||
166 | } | 94 | } |
167 | 95 | ||
168 | static int autofs4_dir_open(struct inode *inode, struct file *file) | 96 | static int autofs4_dir_open(struct inode *inode, struct file *file) |
@@ -170,8 +98,16 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) | |||
170 | struct dentry *dentry = file->f_dentry; | 98 | struct dentry *dentry = file->f_dentry; |
171 | struct vfsmount *mnt = file->f_vfsmnt; | 99 | struct vfsmount *mnt = file->f_vfsmnt; |
172 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); | 100 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); |
101 | struct dentry *cursor; | ||
173 | int status; | 102 | int status; |
174 | 103 | ||
104 | status = dcache_dir_open(inode, file); | ||
105 | if (status) | ||
106 | goto out; | ||
107 | |||
108 | cursor = file->private_data; | ||
109 | cursor->d_fsdata = NULL; | ||
110 | |||
175 | DPRINTK("file=%p dentry=%p %.*s", | 111 | DPRINTK("file=%p dentry=%p %.*s", |
176 | file, dentry, dentry->d_name.len, dentry->d_name.name); | 112 | file, dentry, dentry->d_name.len, dentry->d_name.name); |
177 | 113 | ||
@@ -180,12 +116,15 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) | |||
180 | 116 | ||
181 | if (autofs4_ispending(dentry)) { | 117 | if (autofs4_ispending(dentry)) { |
182 | DPRINTK("dentry busy"); | 118 | DPRINTK("dentry busy"); |
183 | return -EBUSY; | 119 | dcache_dir_close(inode, file); |
120 | status = -EBUSY; | ||
121 | goto out; | ||
184 | } | 122 | } |
185 | 123 | ||
124 | status = -ENOENT; | ||
186 | if (!d_mountpoint(dentry) && dentry->d_op && dentry->d_op->d_revalidate) { | 125 | if (!d_mountpoint(dentry) && dentry->d_op && dentry->d_op->d_revalidate) { |
187 | struct nameidata nd; | 126 | struct nameidata nd; |
188 | int empty; | 127 | int empty, ret; |
189 | 128 | ||
190 | /* In case there are stale directory dentrys from a failed mount */ | 129 | /* In case there are stale directory dentrys from a failed mount */ |
191 | spin_lock(&dcache_lock); | 130 | spin_lock(&dcache_lock); |
@@ -195,13 +134,13 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) | |||
195 | if (!empty) | 134 | if (!empty) |
196 | d_invalidate(dentry); | 135 | d_invalidate(dentry); |
197 | 136 | ||
198 | nd.dentry = dentry; | ||
199 | nd.mnt = mnt; | ||
200 | nd.flags = LOOKUP_DIRECTORY; | 137 | nd.flags = LOOKUP_DIRECTORY; |
201 | status = (dentry->d_op->d_revalidate)(dentry, &nd); | 138 | ret = (dentry->d_op->d_revalidate)(dentry, &nd); |
202 | 139 | ||
203 | if (!status) | 140 | if (!ret) { |
204 | return -ENOENT; | 141 | dcache_dir_close(inode, file); |
142 | goto out; | ||
143 | } | ||
205 | } | 144 | } |
206 | 145 | ||
207 | if (d_mountpoint(dentry)) { | 146 | if (d_mountpoint(dentry)) { |
@@ -212,25 +151,29 @@ static int autofs4_dir_open(struct inode *inode, struct file *file) | |||
212 | if (!autofs4_follow_mount(&fp_mnt, &fp_dentry)) { | 151 | if (!autofs4_follow_mount(&fp_mnt, &fp_dentry)) { |
213 | dput(fp_dentry); | 152 | dput(fp_dentry); |
214 | mntput(fp_mnt); | 153 | mntput(fp_mnt); |
215 | return -ENOENT; | 154 | dcache_dir_close(inode, file); |
155 | goto out; | ||
216 | } | 156 | } |
217 | 157 | ||
218 | fp = dentry_open(fp_dentry, fp_mnt, file->f_flags); | 158 | fp = dentry_open(fp_dentry, fp_mnt, file->f_flags); |
219 | status = PTR_ERR(fp); | 159 | status = PTR_ERR(fp); |
220 | if (IS_ERR(fp)) { | 160 | if (IS_ERR(fp)) { |
221 | file->private_data = NULL; | 161 | dcache_dir_close(inode, file); |
222 | return status; | 162 | goto out; |
223 | } | 163 | } |
224 | file->private_data = fp; | 164 | cursor->d_fsdata = fp; |
225 | } | 165 | } |
226 | out: | ||
227 | return 0; | 166 | return 0; |
167 | out: | ||
168 | return status; | ||
228 | } | 169 | } |
229 | 170 | ||
230 | static int autofs4_dir_close(struct inode *inode, struct file *file) | 171 | static int autofs4_dir_close(struct inode *inode, struct file *file) |
231 | { | 172 | { |
232 | struct dentry *dentry = file->f_dentry; | 173 | struct dentry *dentry = file->f_dentry; |
233 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); | 174 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); |
175 | struct dentry *cursor = file->private_data; | ||
176 | int status = 0; | ||
234 | 177 | ||
235 | DPRINTK("file=%p dentry=%p %.*s", | 178 | DPRINTK("file=%p dentry=%p %.*s", |
236 | file, dentry, dentry->d_name.len, dentry->d_name.name); | 179 | file, dentry, dentry->d_name.len, dentry->d_name.name); |
@@ -240,26 +183,28 @@ static int autofs4_dir_close(struct inode *inode, struct file *file) | |||
240 | 183 | ||
241 | if (autofs4_ispending(dentry)) { | 184 | if (autofs4_ispending(dentry)) { |
242 | DPRINTK("dentry busy"); | 185 | DPRINTK("dentry busy"); |
243 | return -EBUSY; | 186 | status = -EBUSY; |
187 | goto out; | ||
244 | } | 188 | } |
245 | 189 | ||
246 | if (d_mountpoint(dentry)) { | 190 | if (d_mountpoint(dentry)) { |
247 | struct file *fp = file->private_data; | 191 | struct file *fp = cursor->d_fsdata; |
248 | 192 | if (!fp) { | |
249 | if (!fp) | 193 | status = -ENOENT; |
250 | return -ENOENT; | 194 | goto out; |
251 | 195 | } | |
252 | filp_close(fp, current->files); | 196 | filp_close(fp, current->files); |
253 | file->private_data = NULL; | ||
254 | } | 197 | } |
255 | out: | 198 | out: |
256 | return 0; | 199 | dcache_dir_close(inode, file); |
200 | return status; | ||
257 | } | 201 | } |
258 | 202 | ||
259 | static int autofs4_dir_readdir(struct file *file, void *dirent, filldir_t filldir) | 203 | static int autofs4_dir_readdir(struct file *file, void *dirent, filldir_t filldir) |
260 | { | 204 | { |
261 | struct dentry *dentry = file->f_dentry; | 205 | struct dentry *dentry = file->f_dentry; |
262 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); | 206 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); |
207 | struct dentry *cursor = file->private_data; | ||
263 | int status; | 208 | int status; |
264 | 209 | ||
265 | DPRINTK("file=%p dentry=%p %.*s", | 210 | DPRINTK("file=%p dentry=%p %.*s", |
@@ -274,7 +219,7 @@ static int autofs4_dir_readdir(struct file *file, void *dirent, filldir_t filldi | |||
274 | } | 219 | } |
275 | 220 | ||
276 | if (d_mountpoint(dentry)) { | 221 | if (d_mountpoint(dentry)) { |
277 | struct file *fp = file->private_data; | 222 | struct file *fp = cursor->d_fsdata; |
278 | 223 | ||
279 | if (!fp) | 224 | if (!fp) |
280 | return -ENOENT; | 225 | return -ENOENT; |
@@ -289,27 +234,26 @@ static int autofs4_dir_readdir(struct file *file, void *dirent, filldir_t filldi | |||
289 | return status; | 234 | return status; |
290 | } | 235 | } |
291 | out: | 236 | out: |
292 | return autofs4_dcache_readdir(file, dirent, filldir); | 237 | return dcache_readdir(file, dirent, filldir); |
293 | } | 238 | } |
294 | 239 | ||
295 | static int try_to_fill_dentry(struct vfsmount *mnt, struct dentry *dentry, int flags) | 240 | static int try_to_fill_dentry(struct dentry *dentry, int flags) |
296 | { | 241 | { |
297 | struct super_block *sb = mnt->mnt_sb; | 242 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); |
298 | struct autofs_sb_info *sbi = autofs4_sbi(sb); | 243 | struct autofs_info *ino = autofs4_dentry_ino(dentry); |
299 | struct autofs_info *de_info = autofs4_dentry_ino(dentry); | ||
300 | int status = 0; | 244 | int status = 0; |
301 | 245 | ||
302 | /* Block on any pending expiry here; invalidate the dentry | 246 | /* Block on any pending expiry here; invalidate the dentry |
303 | when expiration is done to trigger mount request with a new | 247 | when expiration is done to trigger mount request with a new |
304 | dentry */ | 248 | dentry */ |
305 | if (de_info && (de_info->flags & AUTOFS_INF_EXPIRING)) { | 249 | if (ino && (ino->flags & AUTOFS_INF_EXPIRING)) { |
306 | DPRINTK("waiting for expire %p name=%.*s", | 250 | DPRINTK("waiting for expire %p name=%.*s", |
307 | dentry, dentry->d_name.len, dentry->d_name.name); | 251 | dentry, dentry->d_name.len, dentry->d_name.name); |
308 | 252 | ||
309 | status = autofs4_wait(sbi, dentry, NFY_NONE); | 253 | status = autofs4_wait(sbi, dentry, NFY_NONE); |
310 | 254 | ||
311 | DPRINTK("expire done status=%d", status); | 255 | DPRINTK("expire done status=%d", status); |
312 | 256 | ||
313 | /* | 257 | /* |
314 | * If the directory still exists the mount request must | 258 | * If the directory still exists the mount request must |
315 | * continue otherwise it can't be followed at the right | 259 | * continue otherwise it can't be followed at the right |
@@ -317,34 +261,36 @@ static int try_to_fill_dentry(struct vfsmount *mnt, struct dentry *dentry, int f | |||
317 | */ | 261 | */ |
318 | status = d_invalidate(dentry); | 262 | status = d_invalidate(dentry); |
319 | if (status != -EBUSY) | 263 | if (status != -EBUSY) |
320 | return 0; | 264 | return -ENOENT; |
321 | } | 265 | } |
322 | 266 | ||
323 | DPRINTK("dentry=%p %.*s ino=%p", | 267 | DPRINTK("dentry=%p %.*s ino=%p", |
324 | dentry, dentry->d_name.len, dentry->d_name.name, dentry->d_inode); | 268 | dentry, dentry->d_name.len, dentry->d_name.name, dentry->d_inode); |
325 | 269 | ||
326 | /* Wait for a pending mount, triggering one if there isn't one already */ | 270 | /* |
271 | * Wait for a pending mount, triggering one if there | ||
272 | * isn't one already | ||
273 | */ | ||
327 | if (dentry->d_inode == NULL) { | 274 | if (dentry->d_inode == NULL) { |
328 | DPRINTK("waiting for mount name=%.*s", | 275 | DPRINTK("waiting for mount name=%.*s", |
329 | dentry->d_name.len, dentry->d_name.name); | 276 | dentry->d_name.len, dentry->d_name.name); |
330 | 277 | ||
331 | status = autofs4_wait(sbi, dentry, NFY_MOUNT); | 278 | status = autofs4_wait(sbi, dentry, NFY_MOUNT); |
332 | 279 | ||
333 | DPRINTK("mount done status=%d", status); | 280 | DPRINTK("mount done status=%d", status); |
334 | 281 | ||
335 | if (status && dentry->d_inode) | 282 | if (status && dentry->d_inode) |
336 | return 0; /* Try to get the kernel to invalidate this dentry */ | 283 | return status; /* Try to get the kernel to invalidate this dentry */ |
337 | 284 | ||
338 | /* Turn this into a real negative dentry? */ | 285 | /* Turn this into a real negative dentry? */ |
339 | if (status == -ENOENT) { | 286 | if (status == -ENOENT) { |
340 | dentry->d_time = jiffies + AUTOFS_NEGATIVE_TIMEOUT; | ||
341 | spin_lock(&dentry->d_lock); | 287 | spin_lock(&dentry->d_lock); |
342 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; | 288 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; |
343 | spin_unlock(&dentry->d_lock); | 289 | spin_unlock(&dentry->d_lock); |
344 | return 1; | 290 | return status; |
345 | } else if (status) { | 291 | } else if (status) { |
346 | /* Return a negative dentry, but leave it "pending" */ | 292 | /* Return a negative dentry, but leave it "pending" */ |
347 | return 1; | 293 | return status; |
348 | } | 294 | } |
349 | /* Trigger mount for path component or follow link */ | 295 | /* Trigger mount for path component or follow link */ |
350 | } else if (flags & (LOOKUP_CONTINUE | LOOKUP_DIRECTORY) || | 296 | } else if (flags & (LOOKUP_CONTINUE | LOOKUP_DIRECTORY) || |
@@ -363,19 +309,87 @@ static int try_to_fill_dentry(struct vfsmount *mnt, struct dentry *dentry, int f | |||
363 | spin_lock(&dentry->d_lock); | 309 | spin_lock(&dentry->d_lock); |
364 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; | 310 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; |
365 | spin_unlock(&dentry->d_lock); | 311 | spin_unlock(&dentry->d_lock); |
366 | return 0; | 312 | return status; |
367 | } | 313 | } |
368 | } | 314 | } |
369 | 315 | ||
370 | /* We don't update the usages for the autofs daemon itself, this | 316 | /* Initialize expiry counter after successful mount */ |
371 | is necessary for recursive autofs mounts */ | 317 | if (ino) |
372 | if (!autofs4_oz_mode(sbi)) | 318 | ino->last_used = jiffies; |
373 | autofs4_update_usage(mnt, dentry); | ||
374 | 319 | ||
375 | spin_lock(&dentry->d_lock); | 320 | spin_lock(&dentry->d_lock); |
376 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; | 321 | dentry->d_flags &= ~DCACHE_AUTOFS_PENDING; |
377 | spin_unlock(&dentry->d_lock); | 322 | spin_unlock(&dentry->d_lock); |
378 | return 1; | 323 | return status; |
324 | } | ||
325 | |||
326 | /* For autofs direct mounts the follow link triggers the mount */ | ||
327 | static void *autofs4_follow_link(struct dentry *dentry, struct nameidata *nd) | ||
328 | { | ||
329 | struct autofs_sb_info *sbi = autofs4_sbi(dentry->d_sb); | ||
330 | int oz_mode = autofs4_oz_mode(sbi); | ||
331 | unsigned int lookup_type; | ||
332 | int status; | ||
333 | |||
334 | DPRINTK("dentry=%p %.*s oz_mode=%d nd->flags=%d", | ||
335 | dentry, dentry->d_name.len, dentry->d_name.name, oz_mode, | ||
336 | nd->flags); | ||
337 | |||
338 | /* If it's our master or we shouldn't trigger a mount we're done */ | ||
339 | lookup_type = nd->flags & (LOOKUP_CONTINUE | LOOKUP_DIRECTORY); | ||
340 | if (oz_mode || !lookup_type) | ||
341 | goto done; | ||
342 | |||
343 | /* | ||
344 | * If a request is pending wait for it. | ||
345 | * If it's a mount then it won't be expired till at least | ||
346 | * a liitle later and if it's an expire then we might need | ||
347 | * to mount it again. | ||
348 | */ | ||
349 | if (autofs4_ispending(dentry)) { | ||
350 | DPRINTK("waiting for active request %p name=%.*s", | ||
351 | dentry, dentry->d_name.len, dentry->d_name.name); | ||
352 | |||
353 | status = autofs4_wait(sbi, dentry, NFY_NONE); | ||
354 | |||
355 | DPRINTK("request done status=%d", status); | ||
356 | } | ||
357 | |||
358 | /* | ||
359 | * If the dentry contains directories then it is an | ||
360 | * autofs multi-mount with no root mount offset. So | ||
361 | * don't try to mount it again. | ||
362 | */ | ||
363 | spin_lock(&dcache_lock); | ||
364 | if (!d_mountpoint(dentry) && list_empty(&dentry->d_subdirs)) { | ||
365 | spin_unlock(&dcache_lock); | ||
366 | |||
367 | status = try_to_fill_dentry(dentry, 0); | ||
368 | if (status) | ||
369 | goto out_error; | ||
370 | |||
371 | /* | ||
372 | * The mount succeeded but if there is no root mount | ||
373 | * it must be an autofs multi-mount with no root offset | ||
374 | * so we don't need to follow the mount. | ||
375 | */ | ||
376 | if (d_mountpoint(dentry)) { | ||
377 | if (!autofs4_follow_mount(&nd->mnt, &nd->dentry)) { | ||
378 | status = -ENOENT; | ||
379 | goto out_error; | ||
380 | } | ||
381 | } | ||
382 | |||
383 | goto done; | ||
384 | } | ||
385 | spin_unlock(&dcache_lock); | ||
386 | |||
387 | done: | ||
388 | return NULL; | ||
389 | |||
390 | out_error: | ||
391 | path_release(nd); | ||
392 | return ERR_PTR(status); | ||
379 | } | 393 | } |
380 | 394 | ||
381 | /* | 395 | /* |
@@ -384,47 +398,43 @@ static int try_to_fill_dentry(struct vfsmount *mnt, struct dentry *dentry, int f | |||
384 | * yet completely filled in, and revalidate has to delay such | 398 | * yet completely filled in, and revalidate has to delay such |
385 | * lookups.. | 399 | * lookups.. |
386 | */ | 400 | */ |
387 | static int autofs4_revalidate(struct dentry * dentry, struct nameidata *nd) | 401 | static int autofs4_revalidate(struct dentry *dentry, struct nameidata *nd) |
388 | { | 402 | { |
389 | struct inode * dir = dentry->d_parent->d_inode; | 403 | struct inode *dir = dentry->d_parent->d_inode; |
390 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); | 404 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); |
391 | int oz_mode = autofs4_oz_mode(sbi); | 405 | int oz_mode = autofs4_oz_mode(sbi); |
392 | int flags = nd ? nd->flags : 0; | 406 | int flags = nd ? nd->flags : 0; |
393 | int status = 1; | 407 | int status = 0; |
394 | 408 | ||
395 | /* Pending dentry */ | 409 | /* Pending dentry */ |
396 | if (autofs4_ispending(dentry)) { | 410 | if (autofs4_ispending(dentry)) { |
397 | if (!oz_mode) | 411 | if (!oz_mode) |
398 | status = try_to_fill_dentry(nd->mnt, dentry, flags); | 412 | status = try_to_fill_dentry(dentry, flags); |
399 | return status; | 413 | return !status; |
400 | } | 414 | } |
401 | 415 | ||
402 | /* Negative dentry.. invalidate if "old" */ | 416 | /* Negative dentry.. invalidate if "old" */ |
403 | if (dentry->d_inode == NULL) | 417 | if (dentry->d_inode == NULL) |
404 | return (dentry->d_time - jiffies <= AUTOFS_NEGATIVE_TIMEOUT); | 418 | return 0; |
405 | 419 | ||
406 | /* Check for a non-mountpoint directory with no contents */ | 420 | /* Check for a non-mountpoint directory with no contents */ |
407 | spin_lock(&dcache_lock); | 421 | spin_lock(&dcache_lock); |
408 | if (S_ISDIR(dentry->d_inode->i_mode) && | 422 | if (S_ISDIR(dentry->d_inode->i_mode) && |
409 | !d_mountpoint(dentry) && | 423 | !d_mountpoint(dentry) && |
410 | list_empty(&dentry->d_subdirs)) { | 424 | __simple_empty(dentry)) { |
411 | DPRINTK("dentry=%p %.*s, emptydir", | 425 | DPRINTK("dentry=%p %.*s, emptydir", |
412 | dentry, dentry->d_name.len, dentry->d_name.name); | 426 | dentry, dentry->d_name.len, dentry->d_name.name); |
413 | spin_unlock(&dcache_lock); | 427 | spin_unlock(&dcache_lock); |
414 | if (!oz_mode) | 428 | if (!oz_mode) |
415 | status = try_to_fill_dentry(nd->mnt, dentry, flags); | 429 | status = try_to_fill_dentry(dentry, flags); |
416 | return status; | 430 | return !status; |
417 | } | 431 | } |
418 | spin_unlock(&dcache_lock); | 432 | spin_unlock(&dcache_lock); |
419 | 433 | ||
420 | /* Update the usage list */ | ||
421 | if (!oz_mode) | ||
422 | autofs4_update_usage(nd->mnt, dentry); | ||
423 | |||
424 | return 1; | 434 | return 1; |
425 | } | 435 | } |
426 | 436 | ||
427 | static void autofs4_dentry_release(struct dentry *de) | 437 | void autofs4_dentry_release(struct dentry *de) |
428 | { | 438 | { |
429 | struct autofs_info *inf; | 439 | struct autofs_info *inf; |
430 | 440 | ||
@@ -462,12 +472,13 @@ static struct dentry *autofs4_lookup(struct inode *dir, struct dentry *dentry, s | |||
462 | DPRINTK("name = %.*s", | 472 | DPRINTK("name = %.*s", |
463 | dentry->d_name.len, dentry->d_name.name); | 473 | dentry->d_name.len, dentry->d_name.name); |
464 | 474 | ||
475 | /* File name too long to exist */ | ||
465 | if (dentry->d_name.len > NAME_MAX) | 476 | if (dentry->d_name.len > NAME_MAX) |
466 | return ERR_PTR(-ENAMETOOLONG);/* File name too long to exist */ | 477 | return ERR_PTR(-ENAMETOOLONG); |
467 | 478 | ||
468 | sbi = autofs4_sbi(dir->i_sb); | 479 | sbi = autofs4_sbi(dir->i_sb); |
469 | |||
470 | oz_mode = autofs4_oz_mode(sbi); | 480 | oz_mode = autofs4_oz_mode(sbi); |
481 | |||
471 | DPRINTK("pid = %u, pgrp = %u, catatonic = %d, oz_mode = %d", | 482 | DPRINTK("pid = %u, pgrp = %u, catatonic = %d, oz_mode = %d", |
472 | current->pid, process_group(current), sbi->catatonic, oz_mode); | 483 | current->pid, process_group(current), sbi->catatonic, oz_mode); |
473 | 484 | ||
@@ -519,7 +530,7 @@ static struct dentry *autofs4_lookup(struct inode *dir, struct dentry *dentry, s | |||
519 | * doesn't do the right thing for all system calls, but it should | 530 | * doesn't do the right thing for all system calls, but it should |
520 | * be OK for the operations we permit from an autofs. | 531 | * be OK for the operations we permit from an autofs. |
521 | */ | 532 | */ |
522 | if ( dentry->d_inode && d_unhashed(dentry) ) | 533 | if (dentry->d_inode && d_unhashed(dentry)) |
523 | return ERR_PTR(-ENOENT); | 534 | return ERR_PTR(-ENOENT); |
524 | 535 | ||
525 | return NULL; | 536 | return NULL; |
@@ -531,6 +542,7 @@ static int autofs4_dir_symlink(struct inode *dir, | |||
531 | { | 542 | { |
532 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); | 543 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); |
533 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | 544 | struct autofs_info *ino = autofs4_dentry_ino(dentry); |
545 | struct autofs_info *p_ino; | ||
534 | struct inode *inode; | 546 | struct inode *inode; |
535 | char *cp; | 547 | char *cp; |
536 | 548 | ||
@@ -564,6 +576,10 @@ static int autofs4_dir_symlink(struct inode *dir, | |||
564 | 576 | ||
565 | dentry->d_fsdata = ino; | 577 | dentry->d_fsdata = ino; |
566 | ino->dentry = dget(dentry); | 578 | ino->dentry = dget(dentry); |
579 | atomic_inc(&ino->count); | ||
580 | p_ino = autofs4_dentry_ino(dentry->d_parent); | ||
581 | if (p_ino && dentry->d_parent != dentry) | ||
582 | atomic_inc(&p_ino->count); | ||
567 | ino->inode = inode; | 583 | ino->inode = inode; |
568 | 584 | ||
569 | dir->i_mtime = CURRENT_TIME; | 585 | dir->i_mtime = CURRENT_TIME; |
@@ -590,11 +606,17 @@ static int autofs4_dir_unlink(struct inode *dir, struct dentry *dentry) | |||
590 | { | 606 | { |
591 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); | 607 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); |
592 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | 608 | struct autofs_info *ino = autofs4_dentry_ino(dentry); |
609 | struct autofs_info *p_ino; | ||
593 | 610 | ||
594 | /* This allows root to remove symlinks */ | 611 | /* This allows root to remove symlinks */ |
595 | if ( !autofs4_oz_mode(sbi) && !capable(CAP_SYS_ADMIN) ) | 612 | if ( !autofs4_oz_mode(sbi) && !capable(CAP_SYS_ADMIN) ) |
596 | return -EACCES; | 613 | return -EACCES; |
597 | 614 | ||
615 | if (atomic_dec_and_test(&ino->count)) { | ||
616 | p_ino = autofs4_dentry_ino(dentry->d_parent); | ||
617 | if (p_ino && dentry->d_parent != dentry) | ||
618 | atomic_dec(&p_ino->count); | ||
619 | } | ||
598 | dput(ino->dentry); | 620 | dput(ino->dentry); |
599 | 621 | ||
600 | dentry->d_inode->i_size = 0; | 622 | dentry->d_inode->i_size = 0; |
@@ -611,6 +633,7 @@ static int autofs4_dir_rmdir(struct inode *dir, struct dentry *dentry) | |||
611 | { | 633 | { |
612 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); | 634 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); |
613 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | 635 | struct autofs_info *ino = autofs4_dentry_ino(dentry); |
636 | struct autofs_info *p_ino; | ||
614 | 637 | ||
615 | if (!autofs4_oz_mode(sbi)) | 638 | if (!autofs4_oz_mode(sbi)) |
616 | return -EACCES; | 639 | return -EACCES; |
@@ -625,8 +648,12 @@ static int autofs4_dir_rmdir(struct inode *dir, struct dentry *dentry) | |||
625 | spin_unlock(&dentry->d_lock); | 648 | spin_unlock(&dentry->d_lock); |
626 | spin_unlock(&dcache_lock); | 649 | spin_unlock(&dcache_lock); |
627 | 650 | ||
651 | if (atomic_dec_and_test(&ino->count)) { | ||
652 | p_ino = autofs4_dentry_ino(dentry->d_parent); | ||
653 | if (p_ino && dentry->d_parent != dentry) | ||
654 | atomic_dec(&p_ino->count); | ||
655 | } | ||
628 | dput(ino->dentry); | 656 | dput(ino->dentry); |
629 | |||
630 | dentry->d_inode->i_size = 0; | 657 | dentry->d_inode->i_size = 0; |
631 | dentry->d_inode->i_nlink = 0; | 658 | dentry->d_inode->i_nlink = 0; |
632 | 659 | ||
@@ -640,6 +667,7 @@ static int autofs4_dir_mkdir(struct inode *dir, struct dentry *dentry, int mode) | |||
640 | { | 667 | { |
641 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); | 668 | struct autofs_sb_info *sbi = autofs4_sbi(dir->i_sb); |
642 | struct autofs_info *ino = autofs4_dentry_ino(dentry); | 669 | struct autofs_info *ino = autofs4_dentry_ino(dentry); |
670 | struct autofs_info *p_ino; | ||
643 | struct inode *inode; | 671 | struct inode *inode; |
644 | 672 | ||
645 | if ( !autofs4_oz_mode(sbi) ) | 673 | if ( !autofs4_oz_mode(sbi) ) |
@@ -662,6 +690,10 @@ static int autofs4_dir_mkdir(struct inode *dir, struct dentry *dentry, int mode) | |||
662 | 690 | ||
663 | dentry->d_fsdata = ino; | 691 | dentry->d_fsdata = ino; |
664 | ino->dentry = dget(dentry); | 692 | ino->dentry = dget(dentry); |
693 | atomic_inc(&ino->count); | ||
694 | p_ino = autofs4_dentry_ino(dentry->d_parent); | ||
695 | if (p_ino && dentry->d_parent != dentry) | ||
696 | atomic_inc(&p_ino->count); | ||
665 | ino->inode = inode; | 697 | ino->inode = inode; |
666 | dir->i_nlink++; | 698 | dir->i_nlink++; |
667 | dir->i_mtime = CURRENT_TIME; | 699 | dir->i_mtime = CURRENT_TIME; |
@@ -745,7 +777,7 @@ static inline int autofs4_ask_umount(struct vfsmount *mnt, int __user *p) | |||
745 | { | 777 | { |
746 | int status = 0; | 778 | int status = 0; |
747 | 779 | ||
748 | if (may_umount(mnt) == 0) | 780 | if (may_umount(mnt)) |
749 | status = 1; | 781 | status = 1; |
750 | 782 | ||
751 | DPRINTK("returning %d", status); | 783 | DPRINTK("returning %d", status); |
diff --git a/fs/autofs4/waitq.c b/fs/autofs4/waitq.c index be78e9378c03..142ab6aa2aa1 100644 --- a/fs/autofs4/waitq.c +++ b/fs/autofs4/waitq.c | |||
@@ -3,7 +3,7 @@ | |||
3 | * linux/fs/autofs/waitq.c | 3 | * linux/fs/autofs/waitq.c |
4 | * | 4 | * |
5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved | 5 | * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved |
6 | * Copyright 2001-2003 Ian Kent <raven@themaw.net> | 6 | * Copyright 2001-2006 Ian Kent <raven@themaw.net> |
7 | * | 7 | * |
8 | * This file is part of the Linux kernel and is made available under | 8 | * This file is part of the Linux kernel and is made available under |
9 | * the terms of the GNU General Public License, version 2, or at your | 9 | * the terms of the GNU General Public License, version 2, or at your |
@@ -33,7 +33,7 @@ void autofs4_catatonic_mode(struct autofs_sb_info *sbi) | |||
33 | sbi->catatonic = 1; | 33 | sbi->catatonic = 1; |
34 | wq = sbi->queues; | 34 | wq = sbi->queues; |
35 | sbi->queues = NULL; /* Erase all wait queues */ | 35 | sbi->queues = NULL; /* Erase all wait queues */ |
36 | while ( wq ) { | 36 | while (wq) { |
37 | nwq = wq->next; | 37 | nwq = wq->next; |
38 | wq->status = -ENOENT; /* Magic is gone - report failure */ | 38 | wq->status = -ENOENT; /* Magic is gone - report failure */ |
39 | kfree(wq->name); | 39 | kfree(wq->name); |
@@ -45,7 +45,6 @@ void autofs4_catatonic_mode(struct autofs_sb_info *sbi) | |||
45 | fput(sbi->pipe); /* Close the pipe */ | 45 | fput(sbi->pipe); /* Close the pipe */ |
46 | sbi->pipe = NULL; | 46 | sbi->pipe = NULL; |
47 | } | 47 | } |
48 | |||
49 | shrink_dcache_sb(sbi->sb); | 48 | shrink_dcache_sb(sbi->sb); |
50 | } | 49 | } |
51 | 50 | ||
@@ -98,7 +97,10 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi, | |||
98 | 97 | ||
99 | pkt.hdr.proto_version = sbi->version; | 98 | pkt.hdr.proto_version = sbi->version; |
100 | pkt.hdr.type = type; | 99 | pkt.hdr.type = type; |
101 | if (type == autofs_ptype_missing) { | 100 | switch (type) { |
101 | /* Kernel protocol v4 missing and expire packets */ | ||
102 | case autofs_ptype_missing: | ||
103 | { | ||
102 | struct autofs_packet_missing *mp = &pkt.missing; | 104 | struct autofs_packet_missing *mp = &pkt.missing; |
103 | 105 | ||
104 | pktsz = sizeof(*mp); | 106 | pktsz = sizeof(*mp); |
@@ -107,7 +109,10 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi, | |||
107 | mp->len = wq->len; | 109 | mp->len = wq->len; |
108 | memcpy(mp->name, wq->name, wq->len); | 110 | memcpy(mp->name, wq->name, wq->len); |
109 | mp->name[wq->len] = '\0'; | 111 | mp->name[wq->len] = '\0'; |
110 | } else if (type == autofs_ptype_expire_multi) { | 112 | break; |
113 | } | ||
114 | case autofs_ptype_expire_multi: | ||
115 | { | ||
111 | struct autofs_packet_expire_multi *ep = &pkt.expire_multi; | 116 | struct autofs_packet_expire_multi *ep = &pkt.expire_multi; |
112 | 117 | ||
113 | pktsz = sizeof(*ep); | 118 | pktsz = sizeof(*ep); |
@@ -116,7 +121,34 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi, | |||
116 | ep->len = wq->len; | 121 | ep->len = wq->len; |
117 | memcpy(ep->name, wq->name, wq->len); | 122 | memcpy(ep->name, wq->name, wq->len); |
118 | ep->name[wq->len] = '\0'; | 123 | ep->name[wq->len] = '\0'; |
119 | } else { | 124 | break; |
125 | } | ||
126 | /* | ||
127 | * Kernel protocol v5 packet for handling indirect and direct | ||
128 | * mount missing and expire requests | ||
129 | */ | ||
130 | case autofs_ptype_missing_indirect: | ||
131 | case autofs_ptype_expire_indirect: | ||
132 | case autofs_ptype_missing_direct: | ||
133 | case autofs_ptype_expire_direct: | ||
134 | { | ||
135 | struct autofs_v5_packet *packet = &pkt.v5_packet; | ||
136 | |||
137 | pktsz = sizeof(*packet); | ||
138 | |||
139 | packet->wait_queue_token = wq->wait_queue_token; | ||
140 | packet->len = wq->len; | ||
141 | memcpy(packet->name, wq->name, wq->len); | ||
142 | packet->name[wq->len] = '\0'; | ||
143 | packet->dev = wq->dev; | ||
144 | packet->ino = wq->ino; | ||
145 | packet->uid = wq->uid; | ||
146 | packet->gid = wq->gid; | ||
147 | packet->pid = wq->pid; | ||
148 | packet->tgid = wq->tgid; | ||
149 | break; | ||
150 | } | ||
151 | default: | ||
120 | printk("autofs4_notify_daemon: bad type %d!\n", type); | 152 | printk("autofs4_notify_daemon: bad type %d!\n", type); |
121 | return; | 153 | return; |
122 | } | 154 | } |
@@ -162,21 +194,29 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
162 | { | 194 | { |
163 | struct autofs_wait_queue *wq; | 195 | struct autofs_wait_queue *wq; |
164 | char *name; | 196 | char *name; |
165 | int len, status; | 197 | unsigned int len = 0; |
198 | unsigned int hash = 0; | ||
199 | int status; | ||
166 | 200 | ||
167 | /* In catatonic mode, we don't wait for nobody */ | 201 | /* In catatonic mode, we don't wait for nobody */ |
168 | if ( sbi->catatonic ) | 202 | if (sbi->catatonic) |
169 | return -ENOENT; | 203 | return -ENOENT; |
170 | 204 | ||
171 | name = kmalloc(NAME_MAX + 1, GFP_KERNEL); | 205 | name = kmalloc(NAME_MAX + 1, GFP_KERNEL); |
172 | if (!name) | 206 | if (!name) |
173 | return -ENOMEM; | 207 | return -ENOMEM; |
174 | 208 | ||
175 | len = autofs4_getpath(sbi, dentry, &name); | 209 | /* If this is a direct mount request create a dummy name */ |
176 | if (!len) { | 210 | if (IS_ROOT(dentry) && (sbi->type & AUTOFS_TYPE_DIRECT)) |
177 | kfree(name); | 211 | len = sprintf(name, "%p", dentry); |
178 | return -ENOENT; | 212 | else { |
213 | len = autofs4_getpath(sbi, dentry, &name); | ||
214 | if (!len) { | ||
215 | kfree(name); | ||
216 | return -ENOENT; | ||
217 | } | ||
179 | } | 218 | } |
219 | hash = full_name_hash(name, len); | ||
180 | 220 | ||
181 | if (mutex_lock_interruptible(&sbi->wq_mutex)) { | 221 | if (mutex_lock_interruptible(&sbi->wq_mutex)) { |
182 | kfree(name); | 222 | kfree(name); |
@@ -190,7 +230,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
190 | break; | 230 | break; |
191 | } | 231 | } |
192 | 232 | ||
193 | if ( !wq ) { | 233 | if (!wq) { |
194 | /* Can't wait for an expire if there's no mount */ | 234 | /* Can't wait for an expire if there's no mount */ |
195 | if (notify == NFY_NONE && !d_mountpoint(dentry)) { | 235 | if (notify == NFY_NONE && !d_mountpoint(dentry)) { |
196 | kfree(name); | 236 | kfree(name); |
@@ -200,7 +240,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
200 | 240 | ||
201 | /* Create a new wait queue */ | 241 | /* Create a new wait queue */ |
202 | wq = kmalloc(sizeof(struct autofs_wait_queue),GFP_KERNEL); | 242 | wq = kmalloc(sizeof(struct autofs_wait_queue),GFP_KERNEL); |
203 | if ( !wq ) { | 243 | if (!wq) { |
204 | kfree(name); | 244 | kfree(name); |
205 | mutex_unlock(&sbi->wq_mutex); | 245 | mutex_unlock(&sbi->wq_mutex); |
206 | return -ENOMEM; | 246 | return -ENOMEM; |
@@ -212,12 +252,18 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
212 | wq->next = sbi->queues; | 252 | wq->next = sbi->queues; |
213 | sbi->queues = wq; | 253 | sbi->queues = wq; |
214 | init_waitqueue_head(&wq->queue); | 254 | init_waitqueue_head(&wq->queue); |
215 | wq->hash = dentry->d_name.hash; | 255 | wq->hash = hash; |
216 | wq->name = name; | 256 | wq->name = name; |
217 | wq->len = len; | 257 | wq->len = len; |
258 | wq->dev = autofs4_get_dev(sbi); | ||
259 | wq->ino = autofs4_get_ino(sbi); | ||
260 | wq->uid = current->uid; | ||
261 | wq->gid = current->gid; | ||
262 | wq->pid = current->pid; | ||
263 | wq->tgid = current->tgid; | ||
218 | wq->status = -EINTR; /* Status return if interrupted */ | 264 | wq->status = -EINTR; /* Status return if interrupted */ |
219 | atomic_set(&wq->wait_ctr, 2); | 265 | atomic_set(&wq->wait_ctr, 2); |
220 | atomic_set(&wq->notified, 1); | 266 | atomic_set(&wq->notify, 1); |
221 | mutex_unlock(&sbi->wq_mutex); | 267 | mutex_unlock(&sbi->wq_mutex); |
222 | } else { | 268 | } else { |
223 | atomic_inc(&wq->wait_ctr); | 269 | atomic_inc(&wq->wait_ctr); |
@@ -227,9 +273,26 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
227 | (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify); | 273 | (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify); |
228 | } | 274 | } |
229 | 275 | ||
230 | if (notify != NFY_NONE && atomic_dec_and_test(&wq->notified)) { | 276 | if (notify != NFY_NONE && atomic_read(&wq->notify)) { |
231 | int type = (notify == NFY_MOUNT ? | 277 | int type; |
232 | autofs_ptype_missing : autofs_ptype_expire_multi); | 278 | |
279 | atomic_dec(&wq->notify); | ||
280 | |||
281 | if (sbi->version < 5) { | ||
282 | if (notify == NFY_MOUNT) | ||
283 | type = autofs_ptype_missing; | ||
284 | else | ||
285 | type = autofs_ptype_expire_multi; | ||
286 | } else { | ||
287 | if (notify == NFY_MOUNT) | ||
288 | type = (sbi->type & AUTOFS_TYPE_DIRECT) ? | ||
289 | autofs_ptype_missing_direct : | ||
290 | autofs_ptype_missing_indirect; | ||
291 | else | ||
292 | type = (sbi->type & AUTOFS_TYPE_DIRECT) ? | ||
293 | autofs_ptype_expire_direct : | ||
294 | autofs_ptype_expire_indirect; | ||
295 | } | ||
233 | 296 | ||
234 | DPRINTK("new wait id = 0x%08lx, name = %.*s, nfy=%d\n", | 297 | DPRINTK("new wait id = 0x%08lx, name = %.*s, nfy=%d\n", |
235 | (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify); | 298 | (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify); |
@@ -240,14 +303,14 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry, | |||
240 | 303 | ||
241 | /* wq->name is NULL if and only if the lock is already released */ | 304 | /* wq->name is NULL if and only if the lock is already released */ |
242 | 305 | ||
243 | if ( sbi->catatonic ) { | 306 | if (sbi->catatonic) { |
244 | /* We might have slept, so check again for catatonic mode */ | 307 | /* We might have slept, so check again for catatonic mode */ |
245 | wq->status = -ENOENT; | 308 | wq->status = -ENOENT; |
246 | kfree(wq->name); | 309 | kfree(wq->name); |
247 | wq->name = NULL; | 310 | wq->name = NULL; |
248 | } | 311 | } |
249 | 312 | ||
250 | if ( wq->name ) { | 313 | if (wq->name) { |
251 | /* Block all but "shutdown" signals while waiting */ | 314 | /* Block all but "shutdown" signals while waiting */ |
252 | sigset_t oldset; | 315 | sigset_t oldset; |
253 | unsigned long irqflags; | 316 | unsigned long irqflags; |
@@ -283,12 +346,12 @@ int autofs4_wait_release(struct autofs_sb_info *sbi, autofs_wqt_t wait_queue_tok | |||
283 | struct autofs_wait_queue *wq, **wql; | 346 | struct autofs_wait_queue *wq, **wql; |
284 | 347 | ||
285 | mutex_lock(&sbi->wq_mutex); | 348 | mutex_lock(&sbi->wq_mutex); |
286 | for ( wql = &sbi->queues ; (wq = *wql) != 0 ; wql = &wq->next ) { | 349 | for (wql = &sbi->queues ; (wq = *wql) != 0 ; wql = &wq->next) { |
287 | if ( wq->wait_queue_token == wait_queue_token ) | 350 | if (wq->wait_queue_token == wait_queue_token) |
288 | break; | 351 | break; |
289 | } | 352 | } |
290 | 353 | ||
291 | if ( !wq ) { | 354 | if (!wq) { |
292 | mutex_unlock(&sbi->wq_mutex); | 355 | mutex_unlock(&sbi->wq_mutex); |
293 | return -EINVAL; | 356 | return -EINVAL; |
294 | } | 357 | } |