aboutsummaryrefslogtreecommitdiffstats
path: root/fs/configfs/dir.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/configfs/dir.c')
-rw-r--r--fs/configfs/dir.c28
1 files changed, 27 insertions, 1 deletions
diff --git a/fs/configfs/dir.c b/fs/configfs/dir.c
index a48dc7dd8765..2619f485bc3d 100644
--- a/fs/configfs/dir.c
+++ b/fs/configfs/dir.c
@@ -35,6 +35,14 @@
35#include "configfs_internal.h" 35#include "configfs_internal.h"
36 36
37DECLARE_RWSEM(configfs_rename_sem); 37DECLARE_RWSEM(configfs_rename_sem);
38/*
39 * Protects mutations of configfs_dirent linkage together with proper i_mutex
40 * Mutators of configfs_dirent linkage must *both* have the proper inode locked
41 * and configfs_dirent_lock locked, in that order.
42 * This allows one to safely traverse configfs_dirent trees without having to
43 * lock inodes.
44 */
45DEFINE_SPINLOCK(configfs_dirent_lock);
38 46
39static void configfs_d_iput(struct dentry * dentry, 47static void configfs_d_iput(struct dentry * dentry,
40 struct inode * inode) 48 struct inode * inode)
@@ -79,8 +87,10 @@ static struct configfs_dirent *configfs_new_dirent(struct configfs_dirent * pare
79 atomic_set(&sd->s_count, 1); 87 atomic_set(&sd->s_count, 1);
80 INIT_LIST_HEAD(&sd->s_links); 88 INIT_LIST_HEAD(&sd->s_links);
81 INIT_LIST_HEAD(&sd->s_children); 89 INIT_LIST_HEAD(&sd->s_children);
82 list_add(&sd->s_sibling, &parent_sd->s_children);
83 sd->s_element = element; 90 sd->s_element = element;
91 spin_lock(&configfs_dirent_lock);
92 list_add(&sd->s_sibling, &parent_sd->s_children);
93 spin_unlock(&configfs_dirent_lock);
84 94
85 return sd; 95 return sd;
86} 96}
@@ -173,7 +183,9 @@ static int create_dir(struct config_item * k, struct dentry * p,
173 } else { 183 } else {
174 struct configfs_dirent *sd = d->d_fsdata; 184 struct configfs_dirent *sd = d->d_fsdata;
175 if (sd) { 185 if (sd) {
186 spin_lock(&configfs_dirent_lock);
176 list_del_init(&sd->s_sibling); 187 list_del_init(&sd->s_sibling);
188 spin_unlock(&configfs_dirent_lock);
177 configfs_put(sd); 189 configfs_put(sd);
178 } 190 }
179 } 191 }
@@ -224,7 +236,9 @@ int configfs_create_link(struct configfs_symlink *sl,
224 else { 236 else {
225 struct configfs_dirent *sd = dentry->d_fsdata; 237 struct configfs_dirent *sd = dentry->d_fsdata;
226 if (sd) { 238 if (sd) {
239 spin_lock(&configfs_dirent_lock);
227 list_del_init(&sd->s_sibling); 240 list_del_init(&sd->s_sibling);
241 spin_unlock(&configfs_dirent_lock);
228 configfs_put(sd); 242 configfs_put(sd);
229 } 243 }
230 } 244 }
@@ -238,7 +252,9 @@ static void remove_dir(struct dentry * d)
238 struct configfs_dirent * sd; 252 struct configfs_dirent * sd;
239 253
240 sd = d->d_fsdata; 254 sd = d->d_fsdata;
255 spin_lock(&configfs_dirent_lock);
241 list_del_init(&sd->s_sibling); 256 list_del_init(&sd->s_sibling);
257 spin_unlock(&configfs_dirent_lock);
242 configfs_put(sd); 258 configfs_put(sd);
243 if (d->d_inode) 259 if (d->d_inode)
244 simple_rmdir(parent->d_inode,d); 260 simple_rmdir(parent->d_inode,d);
@@ -410,7 +426,9 @@ static void detach_attrs(struct config_item * item)
410 list_for_each_entry_safe(sd, tmp, &parent_sd->s_children, s_sibling) { 426 list_for_each_entry_safe(sd, tmp, &parent_sd->s_children, s_sibling) {
411 if (!sd->s_element || !(sd->s_type & CONFIGFS_NOT_PINNED)) 427 if (!sd->s_element || !(sd->s_type & CONFIGFS_NOT_PINNED))
412 continue; 428 continue;
429 spin_lock(&configfs_dirent_lock);
413 list_del_init(&sd->s_sibling); 430 list_del_init(&sd->s_sibling);
431 spin_unlock(&configfs_dirent_lock);
414 configfs_drop_dentry(sd, dentry); 432 configfs_drop_dentry(sd, dentry);
415 configfs_put(sd); 433 configfs_put(sd);
416 } 434 }
@@ -1268,7 +1286,9 @@ static int configfs_dir_close(struct inode *inode, struct file *file)
1268 struct configfs_dirent * cursor = file->private_data; 1286 struct configfs_dirent * cursor = file->private_data;
1269 1287
1270 mutex_lock(&dentry->d_inode->i_mutex); 1288 mutex_lock(&dentry->d_inode->i_mutex);
1289 spin_lock(&configfs_dirent_lock);
1271 list_del_init(&cursor->s_sibling); 1290 list_del_init(&cursor->s_sibling);
1291 spin_unlock(&configfs_dirent_lock);
1272 mutex_unlock(&dentry->d_inode->i_mutex); 1292 mutex_unlock(&dentry->d_inode->i_mutex);
1273 1293
1274 release_configfs_dirent(cursor); 1294 release_configfs_dirent(cursor);
@@ -1308,7 +1328,9 @@ static int configfs_readdir(struct file * filp, void * dirent, filldir_t filldir
1308 /* fallthrough */ 1328 /* fallthrough */
1309 default: 1329 default:
1310 if (filp->f_pos == 2) { 1330 if (filp->f_pos == 2) {
1331 spin_lock(&configfs_dirent_lock);
1311 list_move(q, &parent_sd->s_children); 1332 list_move(q, &parent_sd->s_children);
1333 spin_unlock(&configfs_dirent_lock);
1312 } 1334 }
1313 for (p=q->next; p!= &parent_sd->s_children; p=p->next) { 1335 for (p=q->next; p!= &parent_sd->s_children; p=p->next) {
1314 struct configfs_dirent *next; 1336 struct configfs_dirent *next;
@@ -1331,7 +1353,9 @@ static int configfs_readdir(struct file * filp, void * dirent, filldir_t filldir
1331 dt_type(next)) < 0) 1353 dt_type(next)) < 0)
1332 return 0; 1354 return 0;
1333 1355
1356 spin_lock(&configfs_dirent_lock);
1334 list_move(q, p); 1357 list_move(q, p);
1358 spin_unlock(&configfs_dirent_lock);
1335 p = q; 1359 p = q;
1336 filp->f_pos++; 1360 filp->f_pos++;
1337 } 1361 }
@@ -1362,6 +1386,7 @@ static loff_t configfs_dir_lseek(struct file * file, loff_t offset, int origin)
1362 struct list_head *p; 1386 struct list_head *p;
1363 loff_t n = file->f_pos - 2; 1387 loff_t n = file->f_pos - 2;
1364 1388
1389 spin_lock(&configfs_dirent_lock);
1365 list_del(&cursor->s_sibling); 1390 list_del(&cursor->s_sibling);
1366 p = sd->s_children.next; 1391 p = sd->s_children.next;
1367 while (n && p != &sd->s_children) { 1392 while (n && p != &sd->s_children) {
@@ -1373,6 +1398,7 @@ static loff_t configfs_dir_lseek(struct file * file, loff_t offset, int origin)
1373 p = p->next; 1398 p = p->next;
1374 } 1399 }
1375 list_add_tail(&cursor->s_sibling, p); 1400 list_add_tail(&cursor->s_sibling, p);
1401 spin_unlock(&configfs_dirent_lock);
1376 } 1402 }
1377 } 1403 }
1378 mutex_unlock(&dentry->d_inode->i_mutex); 1404 mutex_unlock(&dentry->d_inode->i_mutex);