Blame view

fs/proc_namespace.c 7.76 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)
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
20
21
  	struct seq_file *m = file->private_data;
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
22
23
  	struct mnt_namespace *ns = p->ns;
  	unsigned res = POLLIN | POLLRDNORM;
aab407fc5   Al Viro   don't bother with...
24
  	int event;
0226f4923   Al Viro   vfs: take /proc/*...
25
26
  
  	poll_wait(file, &p->ns->poll, wait);
aab407fc5   Al Viro   don't bother with...
27
  	event = ACCESS_ONCE(ns->event);
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
28
29
  	if (m->poll_event != event) {
  		m->poll_event = event;
0226f4923   Al Viro   vfs: take /proc/*...
30
31
  		res |= POLLERR | POLLPRI;
  	}
0226f4923   Al Viro   vfs: take /proc/*...
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
  
  	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" },
0ae45f63d   Theodore Ts'o   vfs: add support ...
47
  		{ MS_LAZYTIME, ",lazytime" },
0226f4923   Al Viro   vfs: take /proc/*...
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
  		{ 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)
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
96
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
97
  	struct mount *r = real_mount(mnt);
0226f4923   Al Viro   vfs: take /proc/*...
98
  	struct path mnt_path = { .dentry = mnt->mnt_root, .mnt = mnt };
d861c630e   Al Viro   vfs: switch ->sho...
99
  	struct super_block *sb = mnt_path.dentry->d_sb;
5d9f3c7b6   Dmitry V. Levin   vfs: show_vfsmnt:...
100
  	int err;
0226f4923   Al Viro   vfs: take /proc/*...
101

d861c630e   Al Viro   vfs: switch ->sho...
102
103
  	if (sb->s_op->show_devname) {
  		err = sb->s_op->show_devname(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
104
105
106
107
108
109
  		if (err)
  			goto out;
  	} else {
  		mangle(m, r->mnt_devname ? r->mnt_devname : "none");
  	}
  	seq_putc(m, ' ');
9d4d65748   Dmitry V. Levin   vfs: make mounts ...
110
111
112
113
114
  	/* mountpoints outside of chroot jail will give SEQ_SKIP on this */
  	err = seq_path_root(m, &mnt_path, &p->root, " \t
  \\");
  	if (err)
  		goto out;
0226f4923   Al Viro   vfs: take /proc/*...
115
  	seq_putc(m, ' ');
d861c630e   Al Viro   vfs: switch ->sho...
116
  	show_type(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
117
  	seq_puts(m, __mnt_is_readonly(mnt) ? " ro" : " rw");
d861c630e   Al Viro   vfs: switch ->sho...
118
  	err = show_sb_opts(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
119
120
121
  	if (err)
  		goto out;
  	show_mnt_opts(m, mnt);
d861c630e   Al Viro   vfs: switch ->sho...
122
  	if (sb->s_op->show_options)
34c80b1d9   Al Viro   vfs: switch ->sho...
123
  		err = sb->s_op->show_options(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
124
125
126
127
128
129
130
131
  	seq_puts(m, " 0 0
  ");
  out:
  	return err;
  }
  
  static int show_mountinfo(struct seq_file *m, struct vfsmount *mnt)
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
132
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
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 };
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
136
  	int err;
0226f4923   Al Viro   vfs: take /proc/*...
137
138
139
  
  	seq_printf(m, "%i %i %u:%u ", r->mnt_id, r->mnt_parent->mnt_id,
  		   MAJOR(sb->s_dev), MINOR(sb->s_dev));
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
140
  	if (sb->s_op->show_path) {
a6322de67   Al Viro   vfs: switch ->sho...
141
  		err = sb->s_op->show_path(m, mnt->mnt_root);
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
142
143
144
  		if (err)
  			goto out;
  	} else {
0226f4923   Al Viro   vfs: take /proc/*...
145
146
  		seq_dentry(m, mnt->mnt_root, " \t
  \\");
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
147
  	}
0226f4923   Al Viro   vfs: take /proc/*...
148
149
150
  	seq_putc(m, ' ');
  
  	/* mountpoints outside of chroot jail will give SEQ_SKIP on this */
9ad4dc4f7   Dmitry V. Levin   vfs: cleanup show...
151
152
  	err = seq_path_root(m, &mnt_path, &p->root, " \t
  \\");
0226f4923   Al Viro   vfs: take /proc/*...
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
  	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, ' ');
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
176
  	if (sb->s_op->show_devname) {
d861c630e   Al Viro   vfs: switch ->sho...
177
  		err = sb->s_op->show_devname(m, mnt->mnt_root);
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
178
179
180
  		if (err)
  			goto out;
  	} else {
0226f4923   Al Viro   vfs: take /proc/*...
181
  		mangle(m, r->mnt_devname ? r->mnt_devname : "none");
6ce4bca0a   Dmitry V. Levin   vfs: show_mountin...
182
  	}
0226f4923   Al Viro   vfs: take /proc/*...
183
184
185
186
187
  	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...
188
  		err = sb->s_op->show_options(m, mnt->mnt_root);
0226f4923   Al Viro   vfs: take /proc/*...
189
190
191
192
193
194
195
196
  	seq_putc(m, '
  ');
  out:
  	return err;
  }
  
  static int show_vfsstat(struct seq_file *m, struct vfsmount *mnt)
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
197
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
198
199
  	struct mount *r = real_mount(mnt);
  	struct path mnt_path = { .dentry = mnt->mnt_root, .mnt = mnt };
64132379d   Al Viro   vfs: switch ->sho...
200
  	struct super_block *sb = mnt_path.dentry->d_sb;
b896fb35c   Dmitry V. Levin   vfs: show_vfsstat...
201
  	int err;
0226f4923   Al Viro   vfs: take /proc/*...
202
203
  
  	/* device */
64132379d   Al Viro   vfs: switch ->sho...
204
  	if (sb->s_op->show_devname) {
0226f4923   Al Viro   vfs: take /proc/*...
205
  		seq_puts(m, "device ");
d861c630e   Al Viro   vfs: switch ->sho...
206
  		err = sb->s_op->show_devname(m, mnt_path.dentry);
5f8d498d4   Dmitry V. Levin   vfs: show_vfsstat...
207
208
  		if (err)
  			goto out;
0226f4923   Al Viro   vfs: take /proc/*...
209
210
211
212
213
214
215
216
217
218
  	} 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 ");
9d4d65748   Dmitry V. Levin   vfs: make mounts ...
219
220
221
222
223
  	/* mountpoints outside of chroot jail will give SEQ_SKIP on this */
  	err = seq_path_root(m, &mnt_path, &p->root, " \t
  \\");
  	if (err)
  		goto out;
0226f4923   Al Viro   vfs: take /proc/*...
224
225
226
227
  	seq_putc(m, ' ');
  
  	/* file system type */
  	seq_puts(m, "with fstype ");
64132379d   Al Viro   vfs: switch ->sho...
228
  	show_type(m, sb);
0226f4923   Al Viro   vfs: take /proc/*...
229
230
  
  	/* optional statistics */
64132379d   Al Viro   vfs: switch ->sho...
231
  	if (sb->s_op->show_stats) {
0226f4923   Al Viro   vfs: take /proc/*...
232
  		seq_putc(m, ' ');
b896fb35c   Dmitry V. Levin   vfs: show_vfsstat...
233
  		err = sb->s_op->show_stats(m, mnt_path.dentry);
0226f4923   Al Viro   vfs: take /proc/*...
234
235
236
237
  	}
  
  	seq_putc(m, '
  ');
9d4d65748   Dmitry V. Levin   vfs: make mounts ...
238
  out:
0226f4923   Al Viro   vfs: take /proc/*...
239
240
241
242
243
244
245
246
247
248
249
  	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;
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
250
  	struct seq_file *m;
0226f4923   Al Viro   vfs: take /proc/*...
251
252
253
254
  	int ret = -EINVAL;
  
  	if (!task)
  		goto err;
728dba3a3   Eric W. Biederman   namespaces: Use t...
255
256
  	task_lock(task);
  	nsp = task->nsproxy;
3d93116ce   Axel Lin   fs/proc_namespace...
257
  	if (!nsp || !nsp->mnt_ns) {
728dba3a3   Eric W. Biederman   namespaces: Use t...
258
  		task_unlock(task);
0226f4923   Al Viro   vfs: take /proc/*...
259
260
261
262
  		put_task_struct(task);
  		goto err;
  	}
  	ns = nsp->mnt_ns;
0226f4923   Al Viro   vfs: take /proc/*...
263
  	get_mnt_ns(ns);
0226f4923   Al Viro   vfs: take /proc/*...
264
265
266
267
268
269
270
271
272
  	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);
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
273
274
  	ret = seq_open_private(file, &mounts_op, sizeof(struct proc_mounts));
  	if (ret)
0226f4923   Al Viro   vfs: take /proc/*...
275
  		goto err_put_path;
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
276
277
  	m = file->private_data;
  	m->poll_event = ns->event;
0226f4923   Al Viro   vfs: take /proc/*...
278

ede1bf0dc   Yann Droneaud   fs: use seq_open_...
279
  	p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
280
281
  	p->ns = ns;
  	p->root = root;
0226f4923   Al Viro   vfs: take /proc/*...
282
  	p->show = show;
c7999c362   Al Viro   reduce m_start() ...
283
  	p->cached_event = ~0ULL;
0226f4923   Al Viro   vfs: take /proc/*...
284
285
  
  	return 0;
0226f4923   Al Viro   vfs: take /proc/*...
286
287
288
289
290
291
292
293
294
295
   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)
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
296
297
  	struct seq_file *m = file->private_data;
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
298
299
  	path_put(&p->root);
  	put_mnt_ns(p->ns);
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
300
  	return seq_release_private(inode, file);
0226f4923   Al Viro   vfs: take /proc/*...
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
339
  }
  
  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,
  };