diff options
Diffstat (limited to 'fs/sysfs/dir.c')
-rw-r--r-- | fs/sysfs/dir.c | 88 |
1 files changed, 88 insertions, 0 deletions
diff --git a/fs/sysfs/dir.c b/fs/sysfs/dir.c index 06dff2c30c9b..f5f0b936f181 100644 --- a/fs/sysfs/dir.c +++ b/fs/sysfs/dir.c | |||
@@ -20,6 +20,94 @@ spinlock_t kobj_sysfs_assoc_lock = SPIN_LOCK_UNLOCKED; | |||
20 | static spinlock_t sysfs_ino_lock = SPIN_LOCK_UNLOCKED; | 20 | static spinlock_t sysfs_ino_lock = SPIN_LOCK_UNLOCKED; |
21 | static DEFINE_IDA(sysfs_ino_ida); | 21 | static DEFINE_IDA(sysfs_ino_ida); |
22 | 22 | ||
23 | /** | ||
24 | * sysfs_get_active - get an active reference to sysfs_dirent | ||
25 | * @sd: sysfs_dirent to get an active reference to | ||
26 | * | ||
27 | * Get an active reference of @sd. This function is noop if @sd | ||
28 | * is NULL. | ||
29 | * | ||
30 | * RETURNS: | ||
31 | * Pointer to @sd on success, NULL on failure. | ||
32 | */ | ||
33 | struct sysfs_dirent *sysfs_get_active(struct sysfs_dirent *sd) | ||
34 | { | ||
35 | if (sd) { | ||
36 | if (unlikely(!down_read_trylock(&sd->s_active))) | ||
37 | sd = NULL; | ||
38 | } | ||
39 | return sd; | ||
40 | } | ||
41 | |||
42 | /** | ||
43 | * sysfs_put_active - put an active reference to sysfs_dirent | ||
44 | * @sd: sysfs_dirent to put an active reference to | ||
45 | * | ||
46 | * Put an active reference to @sd. This function is noop if @sd | ||
47 | * is NULL. | ||
48 | */ | ||
49 | void sysfs_put_active(struct sysfs_dirent *sd) | ||
50 | { | ||
51 | if (sd) | ||
52 | up_read(&sd->s_active); | ||
53 | } | ||
54 | |||
55 | /** | ||
56 | * sysfs_get_active_two - get active references to sysfs_dirent and parent | ||
57 | * @sd: sysfs_dirent of interest | ||
58 | * | ||
59 | * Get active reference to @sd and its parent. Parent's active | ||
60 | * reference is grabbed first. This function is noop if @sd is | ||
61 | * NULL. | ||
62 | * | ||
63 | * RETURNS: | ||
64 | * Pointer to @sd on success, NULL on failure. | ||
65 | */ | ||
66 | struct sysfs_dirent *sysfs_get_active_two(struct sysfs_dirent *sd) | ||
67 | { | ||
68 | if (sd) { | ||
69 | if (sd->s_parent && unlikely(!sysfs_get_active(sd->s_parent))) | ||
70 | return NULL; | ||
71 | if (unlikely(!sysfs_get_active(sd))) { | ||
72 | sysfs_put_active(sd->s_parent); | ||
73 | return NULL; | ||
74 | } | ||
75 | } | ||
76 | return sd; | ||
77 | } | ||
78 | |||
79 | /** | ||
80 | * sysfs_put_active_two - put active references to sysfs_dirent and parent | ||
81 | * @sd: sysfs_dirent of interest | ||
82 | * | ||
83 | * Put active references to @sd and its parent. This function is | ||
84 | * noop if @sd is NULL. | ||
85 | */ | ||
86 | void sysfs_put_active_two(struct sysfs_dirent *sd) | ||
87 | { | ||
88 | if (sd) { | ||
89 | sysfs_put_active(sd); | ||
90 | sysfs_put_active(sd->s_parent); | ||
91 | } | ||
92 | } | ||
93 | |||
94 | /** | ||
95 | * sysfs_deactivate - deactivate sysfs_dirent | ||
96 | * @sd: sysfs_dirent to deactivate | ||
97 | * | ||
98 | * Deny new active references and drain existing ones. s_active | ||
99 | * will be unlocked when the sysfs_dirent is released. | ||
100 | */ | ||
101 | void sysfs_deactivate(struct sysfs_dirent *sd) | ||
102 | { | ||
103 | down_write_nested(&sd->s_active, SYSFS_S_ACTIVE_DEACTIVATE); | ||
104 | |||
105 | /* s_active will be unlocked by the thread doing the final put | ||
106 | * on @sd. Lie to lockdep. | ||
107 | */ | ||
108 | rwsem_release(&sd->s_active.dep_map, 1, _RET_IP_); | ||
109 | } | ||
110 | |||
23 | static int sysfs_alloc_ino(ino_t *pino) | 111 | static int sysfs_alloc_ino(ino_t *pino) |
24 | { | 112 | { |
25 | int ino, rc; | 113 | int ino, rc; |