1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19#include <linux/module.h>
20#include <linux/fs.h>
21#include <linux/mount.h>
22#include <linux/pagemap.h>
23#include <linux/init.h>
24#include <linux/kobject.h>
25#include <linux/namei.h>
26#include <linux/debugfs.h>
27#include <linux/fsnotify.h>
28#include <linux/string.h>
29#include <linux/magic.h>
30
31static struct vfsmount *debugfs_mount;
32static int debugfs_mount_count;
33static bool debugfs_registered;
34
35static struct inode *debugfs_get_inode(struct super_block *sb, int mode, dev_t dev)
36{
37 struct inode *inode = new_inode(sb);
38
39 if (inode) {
40 inode->i_mode = mode;
41 inode->i_atime = inode->i_mtime = inode->i_ctime = CURRENT_TIME;
42 switch (mode & S_IFMT) {
43 default:
44 init_special_inode(inode, mode, dev);
45 break;
46 case S_IFREG:
47 inode->i_fop = &debugfs_file_operations;
48 break;
49 case S_IFLNK:
50 inode->i_op = &debugfs_link_operations;
51 break;
52 case S_IFDIR:
53 inode->i_op = &simple_dir_inode_operations;
54 inode->i_fop = &simple_dir_operations;
55
56
57
58 inc_nlink(inode);
59 break;
60 }
61 }
62 return inode;
63}
64
65
66static int debugfs_mknod(struct inode *dir, struct dentry *dentry,
67 int mode, dev_t dev)
68{
69 struct inode *inode;
70 int error = -EPERM;
71
72 if (dentry->d_inode)
73 return -EEXIST;
74
75 inode = debugfs_get_inode(dir->i_sb, mode, dev);
76 if (inode) {
77 d_instantiate(dentry, inode);
78 dget(dentry);
79 error = 0;
80 }
81 return error;
82}
83
84static int debugfs_mkdir(struct inode *dir, struct dentry *dentry, int mode)
85{
86 int res;
87
88 mode = (mode & (S_IRWXUGO | S_ISVTX)) | S_IFDIR;
89 res = debugfs_mknod(dir, dentry, mode, 0);
90 if (!res) {
91 inc_nlink(dir);
92 fsnotify_mkdir(dir, dentry);
93 }
94 return res;
95}
96
97static int debugfs_link(struct inode *dir, struct dentry *dentry, int mode)
98{
99 mode = (mode & S_IALLUGO) | S_IFLNK;
100 return debugfs_mknod(dir, dentry, mode, 0);
101}
102
103static int debugfs_create(struct inode *dir, struct dentry *dentry, int mode)
104{
105 int res;
106
107 mode = (mode & S_IALLUGO) | S_IFREG;
108 res = debugfs_mknod(dir, dentry, mode, 0);
109 if (!res)
110 fsnotify_create(dir, dentry);
111 return res;
112}
113
114static inline int debugfs_positive(struct dentry *dentry)
115{
116 return dentry->d_inode && !d_unhashed(dentry);
117}
118
119static int debug_fill_super(struct super_block *sb, void *data, int silent)
120{
121 static struct tree_descr debug_files[] = {{""}};
122
123 return simple_fill_super(sb, DEBUGFS_MAGIC, debug_files);
124}
125
126static int debug_get_sb(struct file_system_type *fs_type,
127 int flags, const char *dev_name,
128 void *data, struct vfsmount *mnt)
129{
130 return get_sb_single(fs_type, flags, data, debug_fill_super, mnt);
131}
132
133static struct file_system_type debug_fs_type = {
134 .owner = THIS_MODULE,
135 .name = "debugfs",
136 .get_sb = debug_get_sb,
137 .kill_sb = kill_litter_super,
138};
139
140static int debugfs_create_by_name(const char *name, mode_t mode,
141 struct dentry *parent,
142 struct dentry **dentry)
143{
144 int error = 0;
145
146
147
148
149
150
151 if (!parent) {
152 if (debugfs_mount && debugfs_mount->mnt_sb) {
153 parent = debugfs_mount->mnt_sb->s_root;
154 }
155 }
156 if (!parent) {
157 pr_debug("debugfs: Ah! can not find a parent!\n");
158 return -EFAULT;
159 }
160
161 *dentry = NULL;
162 mutex_lock(&parent->d_inode->i_mutex);
163 *dentry = lookup_one_len(name, parent, strlen(name));
164 if (!IS_ERR(*dentry)) {
165 switch (mode & S_IFMT) {
166 case S_IFDIR:
167 error = debugfs_mkdir(parent->d_inode, *dentry, mode);
168 break;
169 case S_IFLNK:
170 error = debugfs_link(parent->d_inode, *dentry, mode);
171 break;
172 default:
173 error = debugfs_create(parent->d_inode, *dentry, mode);
174 break;
175 }
176 dput(*dentry);
177 } else
178 error = PTR_ERR(*dentry);
179 mutex_unlock(&parent->d_inode->i_mutex);
180
181 return error;
182}
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210struct dentry *debugfs_create_file(const char *name, mode_t mode,
211 struct dentry *parent, void *data,
212 const struct file_operations *fops)
213{
214 struct dentry *dentry = NULL;
215 int error;
216
217 pr_debug("debugfs: creating file '%s'\n",name);
218
219 error = simple_pin_fs(&debug_fs_type, &debugfs_mount,
220 &debugfs_mount_count);
221 if (error)
222 goto exit;
223
224 error = debugfs_create_by_name(name, mode, parent, &dentry);
225 if (error) {
226 dentry = NULL;
227 simple_release_fs(&debugfs_mount, &debugfs_mount_count);
228 goto exit;
229 }
230
231 if (dentry->d_inode) {
232 if (data)
233 dentry->d_inode->i_private = data;
234 if (fops)
235 dentry->d_inode->i_fop = fops;
236 }
237exit:
238 return dentry;
239}
240EXPORT_SYMBOL_GPL(debugfs_create_file);
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260struct dentry *debugfs_create_dir(const char *name, struct dentry *parent)
261{
262 return debugfs_create_file(name,
263 S_IFDIR | S_IRWXU | S_IRUGO | S_IXUGO,
264 parent, NULL, NULL);
265}
266EXPORT_SYMBOL_GPL(debugfs_create_dir);
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291struct dentry *debugfs_create_symlink(const char *name, struct dentry *parent,
292 const char *target)
293{
294 struct dentry *result;
295 char *link;
296
297 link = kstrdup(target, GFP_KERNEL);
298 if (!link)
299 return NULL;
300
301 result = debugfs_create_file(name, S_IFLNK | S_IRWXUGO, parent, link,
302 NULL);
303 if (!result)
304 kfree(link);
305 return result;
306}
307EXPORT_SYMBOL_GPL(debugfs_create_symlink);
308
309static void __debugfs_remove(struct dentry *dentry, struct dentry *parent)
310{
311 int ret = 0;
312
313 if (debugfs_positive(dentry)) {
314 if (dentry->d_inode) {
315 dget(dentry);
316 switch (dentry->d_inode->i_mode & S_IFMT) {
317 case S_IFDIR:
318 ret = simple_rmdir(parent->d_inode, dentry);
319 break;
320 case S_IFLNK:
321 kfree(dentry->d_inode->i_private);
322
323 default:
324 simple_unlink(parent->d_inode, dentry);
325 break;
326 }
327 if (!ret)
328 d_delete(dentry);
329 dput(dentry);
330 }
331 }
332}
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347void debugfs_remove(struct dentry *dentry)
348{
349 struct dentry *parent;
350
351 if (!dentry)
352 return;
353
354 parent = dentry->d_parent;
355 if (!parent || !parent->d_inode)
356 return;
357
358 mutex_lock(&parent->d_inode->i_mutex);
359 __debugfs_remove(dentry, parent);
360 mutex_unlock(&parent->d_inode->i_mutex);
361 simple_release_fs(&debugfs_mount, &debugfs_mount_count);
362}
363EXPORT_SYMBOL_GPL(debugfs_remove);
364
365
366
367
368
369
370
371
372
373
374
375
376
377void debugfs_remove_recursive(struct dentry *dentry)
378{
379 struct dentry *child;
380 struct dentry *parent;
381
382 if (!dentry)
383 return;
384
385 parent = dentry->d_parent;
386 if (!parent || !parent->d_inode)
387 return;
388
389 parent = dentry;
390 mutex_lock(&parent->d_inode->i_mutex);
391
392 while (1) {
393
394
395
396
397 if (list_empty(&parent->d_subdirs)) {
398 mutex_unlock(&parent->d_inode->i_mutex);
399 if (parent == dentry)
400 break;
401 parent = parent->d_parent;
402 mutex_lock(&parent->d_inode->i_mutex);
403 }
404 child = list_entry(parent->d_subdirs.next, struct dentry,
405 d_u.d_child);
406 next_sibling:
407
408
409
410
411
412 if (!list_empty(&child->d_subdirs)) {
413 mutex_unlock(&parent->d_inode->i_mutex);
414 parent = child;
415 mutex_lock(&parent->d_inode->i_mutex);
416 continue;
417 }
418 __debugfs_remove(child, parent);
419 if (parent->d_subdirs.next == &child->d_u.d_child) {
420
421
422
423 if (child->d_u.d_child.next != &parent->d_subdirs) {
424 child = list_entry(child->d_u.d_child.next,
425 struct dentry,
426 d_u.d_child);
427 goto next_sibling;
428 }
429
430
431
432
433
434 mutex_unlock(&parent->d_inode->i_mutex);
435 break;
436 }
437 simple_release_fs(&debugfs_mount, &debugfs_mount_count);
438 }
439
440 parent = dentry->d_parent;
441 mutex_lock(&parent->d_inode->i_mutex);
442 __debugfs_remove(dentry, parent);
443 mutex_unlock(&parent->d_inode->i_mutex);
444 simple_release_fs(&debugfs_mount, &debugfs_mount_count);
445}
446EXPORT_SYMBOL_GPL(debugfs_remove_recursive);
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467struct dentry *debugfs_rename(struct dentry *old_dir, struct dentry *old_dentry,
468 struct dentry *new_dir, const char *new_name)
469{
470 int error;
471 struct dentry *dentry = NULL, *trap;
472 const char *old_name;
473
474 trap = lock_rename(new_dir, old_dir);
475
476 if (!old_dir->d_inode || !new_dir->d_inode)
477 goto exit;
478
479 if (!old_dentry->d_inode || old_dentry == trap ||
480 d_mountpoint(old_dentry))
481 goto exit;
482 dentry = lookup_one_len(new_name, new_dir, strlen(new_name));
483
484 if (IS_ERR(dentry) || dentry == trap || dentry->d_inode)
485 goto exit;
486
487 old_name = fsnotify_oldname_init(old_dentry->d_name.name);
488
489 error = simple_rename(old_dir->d_inode, old_dentry, new_dir->d_inode,
490 dentry);
491 if (error) {
492 fsnotify_oldname_free(old_name);
493 goto exit;
494 }
495 d_move(old_dentry, dentry);
496 fsnotify_move(old_dir->d_inode, new_dir->d_inode, old_name,
497 old_dentry->d_name.name, S_ISDIR(old_dentry->d_inode->i_mode),
498 NULL, old_dentry);
499 fsnotify_oldname_free(old_name);
500 unlock_rename(new_dir, old_dir);
501 dput(dentry);
502 return old_dentry;
503exit:
504 if (dentry && !IS_ERR(dentry))
505 dput(dentry);
506 unlock_rename(new_dir, old_dir);
507 return NULL;
508}
509EXPORT_SYMBOL_GPL(debugfs_rename);
510
511
512
513
514bool debugfs_initialized(void)
515{
516 return debugfs_registered;
517}
518EXPORT_SYMBOL_GPL(debugfs_initialized);
519
520
521static struct kobject *debug_kobj;
522
523static int __init debugfs_init(void)
524{
525 int retval;
526
527 debug_kobj = kobject_create_and_add("debug", kernel_kobj);
528 if (!debug_kobj)
529 return -EINVAL;
530
531 retval = register_filesystem(&debug_fs_type);
532 if (retval)
533 kobject_put(debug_kobj);
534 else
535 debugfs_registered = true;
536
537 return retval;
538}
539
540static void __exit debugfs_exit(void)
541{
542 debugfs_registered = false;
543
544 simple_release_fs(&debugfs_mount, &debugfs_mount_count);
545 unregister_filesystem(&debug_fs_type);
546 kobject_put(debug_kobj);
547}
548
549core_initcall(debugfs_init);
550module_exit(debugfs_exit);
551MODULE_LICENSE("GPL");
552
553