Blame view

fs/proc_namespace.c 7.55 KB
0226f4923   Al Viro   vfs: take /proc/*...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
  /*
   * fs/proc_namespace.c - handling of /proc/<pid>/{mounts,mountinfo,mountstats}
   *
   * In fact, that's a piece of procfs; it's *almost* isolated from
   * the rest of fs/proc, but has rather close relationships with
   * fs/namespace.c, thus here instead of fs/proc
   *
   */
  #include <linux/mnt_namespace.h>
  #include <linux/nsproxy.h>
  #include <linux/security.h>
  #include <linux/fs_struct.h>
  #include "proc/internal.h" /* only for get_proc_task() in ->open() */
  
  #include "pnode.h"
  #include "internal.h"
  
  static unsigned mounts_poll(struct file *file, poll_table *wait)
  {
6ce6e24e7   Al Viro   get rid of magic ...
20
  	struct proc_mounts *p = proc_mounts(file->private_data);
0226f4923   Al Viro   vfs: take /proc/*...
21
22
23
24
  	struct mnt_namespace *ns = p->ns;
  	unsigned res = POLLIN | POLLRDNORM;
  
  	poll_wait(file, &p->ns->poll, wait);
962830df3   Andi Kleen   brlocks/lglocks: ...
25
  	br_read_lock(&vfsmount_lock);
0226f4923   Al Viro   vfs: take /proc/*...
26
27
28
29
  	if (p->m.poll_event != ns->event) {
  		p->m.poll_event = ns->event;
  		res |= POLLERR | POLLPRI;
  	}
962830df3   Andi Kleen   brlocks/lglocks: ...
30
  	br_read_unlock(&vfsmount_lock);
0226f4923   Al Viro   vfs: take /proc/*...
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
  
  	return res;
  }
  
  struct proc_fs_info {
  	int flag;
  	const char *str;
  };
  
  static int show_sb_opts(struct seq_file *m, struct super_block *sb)
  {
  	static const struct proc_fs_info fs_info[] = {
  		{ MS_SYNCHRONOUS, ",sync" },
  		{ MS_DIRSYNC, ",dirsync" },
  		{ MS_MANDLOCK, ",mand" },
  		{ 0, NULL }
  	};
  	const struct proc_fs_info *fs_infop;
  
  	for (fs_infop = fs_info; fs_infop->flag; fs_infop++) {
  		if (sb->s_flags & fs_infop->flag)
  			seq_puts(m, fs_infop->str);
  	}
  
  	return security_sb_show_options(m, sb);
  }
  
  static void show_mnt_opts(struct seq_file *m, struct vfsmount *mnt)
  {
  	static const struct proc_fs_info mnt_info[] = {
  		{ MNT_NOSUID, ",nosuid" },
  		{ MNT_NODEV, ",nodev" },
  		{ MNT_NOEXEC, ",noexec" },
  		{ MNT_NOATIME, ",noatime" },
  		{ MNT_NODIRATIME, ",nodiratime" },
  		{ MNT_RELATIME, ",relatime" },
  		{ 0, NULL }
  	};
  	const struct proc_fs_info *fs_infop;
  
  	for (fs_infop = mnt_info; fs_infop->flag; fs_infop++) {
  		if (mnt->mnt_flags & fs_infop->flag)
  			seq_puts(m, fs_infop->str);
  	}
  }
  
  static inline void mangle(struct seq_file *m, const char *s)
  {
  	seq_escape(m, s, " \t
  \\");
  }
  
  static void show_type(struct seq_file *m, struct super_block *sb)
  {
  	mangle(m, sb->s_type->name);
  	if (sb->s_subtype && sb->s_subtype[0]) {
  		seq_putc(m, '.');
  		mangle(m, sb->s_subtype);
  	}
  }
  
  static int show_vfsmnt(struct seq_file *m, struct vfsmount *mnt)
  {
  	struct mount *r = real_mount(mnt);
  	int err = 0;
  	struct path mnt_path = { .dentry = mnt->mnt_root, .mnt = mnt };
d861c630e   Al Viro   vfs: switch ->sho...
97
  	struct super_block *sb = mnt_path.dentry->d_sb;
0226f4923   Al Viro   vfs: take /proc/*...
98

d861c630e   Al Viro   vfs: switch ->sho...
99
100
  	if (sb->s_op->show_devname) {
  		err = sb->s_op->show_devname(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
101
102
103
104
105
106
107
108
109
  		if (err)
  			goto out;
  	} else {
  		mangle(m, r->mnt_devname ? r->mnt_devname : "none");
  	}
  	seq_putc(m, ' ');
  	seq_path(m, &mnt_path, " \t
  \\");
  	seq_putc(m, ' ');
d861c630e   Al Viro   vfs: switch ->sho...
110
  	show_type(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
111
  	seq_puts(m, __mnt_is_readonly(mnt) ? " ro" : " rw");
d861c630e   Al Viro   vfs: switch ->sho...
112
  	err = show_sb_opts(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
113
114
115
  	if (err)
  		goto out;
  	show_mnt_opts(m, mnt);
d861c630e   Al Viro   vfs: switch ->sho...
116
  	if (sb->s_op->show_options)
34c80b1d9   Al Viro   vfs: switch ->sho...
117
  		err = sb->s_op->show_options(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
118
119
120
121
122
123
124
125
  	seq_puts(m, " 0 0
  ");
  out:
  	return err;
  }
  
  static int show_mountinfo(struct seq_file *m, struct vfsmount *mnt)
  {
6ce6e24e7   Al Viro   get rid of magic ...
126
  	struct proc_mounts *p = proc_mounts(m);
0226f4923   Al Viro   vfs: take /proc/*...
127
128
129
130
131
132
133
134
135
  	struct mount *r = real_mount(mnt);
  	struct super_block *sb = mnt->mnt_sb;
  	struct path mnt_path = { .dentry = mnt->mnt_root, .mnt = mnt };
  	struct path root = p->root;
  	int err = 0;
  
  	seq_printf(m, "%i %i %u:%u ", r->mnt_id, r->mnt_parent->mnt_id,
  		   MAJOR(sb->s_dev), MINOR(sb->s_dev));
  	if (sb->s_op->show_path)
a6322de67   Al Viro   vfs: switch ->sho...
136
  		err = sb->s_op->show_path(m, mnt->mnt_root);
0226f4923   Al Viro   vfs: take /proc/*...
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
  	else
  		seq_dentry(m, mnt->mnt_root, " \t
  \\");
  	if (err)
  		goto out;
  	seq_putc(m, ' ');
  
  	/* mountpoints outside of chroot jail will give SEQ_SKIP on this */
  	err = seq_path_root(m, &mnt_path, &root, " \t
  \\");
  	if (err)
  		goto out;
  
  	seq_puts(m, mnt->mnt_flags & MNT_READONLY ? " ro" : " rw");
  	show_mnt_opts(m, mnt);
  
  	/* Tagged fields ("foo:X" or "bar") */
  	if (IS_MNT_SHARED(r))
  		seq_printf(m, " shared:%i", r->mnt_group_id);
  	if (IS_MNT_SLAVE(r)) {
  		int master = r->mnt_master->mnt_group_id;
  		int dom = get_dominating_id(r, &p->root);
  		seq_printf(m, " master:%i", master);
  		if (dom && dom != master)
  			seq_printf(m, " propagate_from:%i", dom);
  	}
  	if (IS_MNT_UNBINDABLE(r))
  		seq_puts(m, " unbindable");
  
  	/* Filesystem specific data */
  	seq_puts(m, " - ");
  	show_type(m, sb);
  	seq_putc(m, ' ');
  	if (sb->s_op->show_devname)
d861c630e   Al Viro   vfs: switch ->sho...
171
  		err = sb->s_op->show_devname(m, mnt->mnt_root);
0226f4923   Al Viro   vfs: take /proc/*...
172
173
174
175
176
177
178
179
180
  	else
  		mangle(m, r->mnt_devname ? r->mnt_devname : "none");
  	if (err)
  		goto out;
  	seq_puts(m, sb->s_flags & MS_RDONLY ? " ro" : " rw");
  	err = show_sb_opts(m, sb);
  	if (err)
  		goto out;
  	if (sb->s_op->show_options)
34c80b1d9   Al Viro   vfs: switch ->sho...
181
  		err = sb->s_op->show_options(m, mnt->mnt_root);
0226f4923   Al Viro   vfs: take /proc/*...
182
183
184
185
186
187
188
189
190
191
  	seq_putc(m, '
  ');
  out:
  	return err;
  }
  
  static int show_vfsstat(struct seq_file *m, struct vfsmount *mnt)
  {
  	struct mount *r = real_mount(mnt);
  	struct path mnt_path = { .dentry = mnt->mnt_root, .mnt = mnt };
64132379d   Al Viro   vfs: switch ->sho...
192
  	struct super_block *sb = mnt_path.dentry->d_sb;
0226f4923   Al Viro   vfs: take /proc/*...
193
194
195
  	int err = 0;
  
  	/* device */
64132379d   Al Viro   vfs: switch ->sho...
196
  	if (sb->s_op->show_devname) {
0226f4923   Al Viro   vfs: take /proc/*...
197
  		seq_puts(m, "device ");
d861c630e   Al Viro   vfs: switch ->sho...
198
  		err = sb->s_op->show_devname(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
  	} else {
  		if (r->mnt_devname) {
  			seq_puts(m, "device ");
  			mangle(m, r->mnt_devname);
  		} else
  			seq_puts(m, "no device");
  	}
  
  	/* mount point */
  	seq_puts(m, " mounted on ");
  	seq_path(m, &mnt_path, " \t
  \\");
  	seq_putc(m, ' ');
  
  	/* file system type */
  	seq_puts(m, "with fstype ");
64132379d   Al Viro   vfs: switch ->sho...
215
  	show_type(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
216
217
  
  	/* optional statistics */
64132379d   Al Viro   vfs: switch ->sho...
218
  	if (sb->s_op->show_stats) {
0226f4923   Al Viro   vfs: take /proc/*...
219
220
  		seq_putc(m, ' ');
  		if (!err)
64132379d   Al Viro   vfs: switch ->sho...
221
  			err = sb->s_op->show_stats(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
  	}
  
  	seq_putc(m, '
  ');
  	return err;
  }
  
  static int mounts_open_common(struct inode *inode, struct file *file,
  			      int (*show)(struct seq_file *, struct vfsmount *))
  {
  	struct task_struct *task = get_proc_task(inode);
  	struct nsproxy *nsp;
  	struct mnt_namespace *ns = NULL;
  	struct path root;
  	struct proc_mounts *p;
  	int ret = -EINVAL;
  
  	if (!task)
  		goto err;
  
  	rcu_read_lock();
  	nsp = task_nsproxy(task);
  	if (!nsp) {
  		rcu_read_unlock();
  		put_task_struct(task);
  		goto err;
  	}
  	ns = nsp->mnt_ns;
  	if (!ns) {
  		rcu_read_unlock();
  		put_task_struct(task);
  		goto err;
  	}
  	get_mnt_ns(ns);
  	rcu_read_unlock();
  	task_lock(task);
  	if (!task->fs) {
  		task_unlock(task);
  		put_task_struct(task);
  		ret = -ENOENT;
  		goto err_put_ns;
  	}
  	get_fs_root(task->fs, &root);
  	task_unlock(task);
  	put_task_struct(task);
  
  	ret = -ENOMEM;
  	p = kmalloc(sizeof(struct proc_mounts), GFP_KERNEL);
  	if (!p)
  		goto err_put_path;
  
  	file->private_data = &p->m;
  	ret = seq_open(file, &mounts_op);
  	if (ret)
  		goto err_free;
0226f4923   Al Viro   vfs: take /proc/*...
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
  	p->ns = ns;
  	p->root = root;
  	p->m.poll_event = ns->event;
  	p->show = show;
  
  	return 0;
  
   err_free:
  	kfree(p);
   err_put_path:
  	path_put(&root);
   err_put_ns:
  	put_mnt_ns(ns);
   err:
  	return ret;
  }
  
  static int mounts_release(struct inode *inode, struct file *file)
  {
6ce6e24e7   Al Viro   get rid of magic ...
296
  	struct proc_mounts *p = proc_mounts(file->private_data);
0226f4923   Al Viro   vfs: take /proc/*...
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
  	path_put(&p->root);
  	put_mnt_ns(p->ns);
  	return seq_release(inode, file);
  }
  
  static int mounts_open(struct inode *inode, struct file *file)
  {
  	return mounts_open_common(inode, file, show_vfsmnt);
  }
  
  static int mountinfo_open(struct inode *inode, struct file *file)
  {
  	return mounts_open_common(inode, file, show_mountinfo);
  }
  
  static int mountstats_open(struct inode *inode, struct file *file)
  {
  	return mounts_open_common(inode, file, show_vfsstat);
  }
  
  const struct file_operations proc_mounts_operations = {
  	.open		= mounts_open,
  	.read		= seq_read,
  	.llseek		= seq_lseek,
  	.release	= mounts_release,
  	.poll		= mounts_poll,
  };
  
  const struct file_operations proc_mountinfo_operations = {
  	.open		= mountinfo_open,
  	.read		= seq_read,
  	.llseek		= seq_lseek,
  	.release	= mounts_release,
  	.poll		= mounts_poll,
  };
  
  const struct file_operations proc_mountstats_operations = {
  	.open		= mountstats_open,
  	.read		= seq_read,
  	.llseek		= seq_lseek,
  	.release	= mounts_release,
  };