Blame view

fs/proc_namespace.c 7.89 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
0226f4923   Al Viro   vfs: take /proc/*...
2
3
4
5
6
7
8
9
10
11
12
13
  /*
   * 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>
f719ff9bc   Ingo Molnar   sched/headers: Pr...
14
  #include <linux/sched/task.h>
0226f4923   Al Viro   vfs: take /proc/*...
15
16
17
18
  #include "proc/internal.h" /* only for get_proc_task() in ->open() */
  
  #include "pnode.h"
  #include "internal.h"
076ccb76e   Al Viro   fs: annotate ->po...
19
  static __poll_t mounts_poll(struct file *file, poll_table *wait)
0226f4923   Al Viro   vfs: take /proc/*...
20
  {
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
21
22
  	struct seq_file *m = file->private_data;
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
23
  	struct mnt_namespace *ns = p->ns;
a9a08845e   Linus Torvalds   vfs: do bulk POLL...
24
  	__poll_t res = EPOLLIN | EPOLLRDNORM;
aab407fc5   Al Viro   don't bother with...
25
  	int event;
0226f4923   Al Viro   vfs: take /proc/*...
26
27
  
  	poll_wait(file, &p->ns->poll, wait);
6aa7de059   Mark Rutland   locking/atomics: ...
28
  	event = READ_ONCE(ns->event);
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
29
30
  	if (m->poll_event != event) {
  		m->poll_event = event;
a9a08845e   Linus Torvalds   vfs: do bulk POLL...
31
  		res |= EPOLLERR | EPOLLPRI;
0226f4923   Al Viro   vfs: take /proc/*...
32
  	}
0226f4923   Al Viro   vfs: take /proc/*...
33
34
35
  
  	return res;
  }
1e88c4201   Alexey Gladkov   proc: rename stru...
36
  struct proc_fs_opts {
0226f4923   Al Viro   vfs: take /proc/*...
37
38
39
40
41
42
  	int flag;
  	const char *str;
  };
  
  static int show_sb_opts(struct seq_file *m, struct super_block *sb)
  {
1e88c4201   Alexey Gladkov   proc: rename stru...
43
  	static const struct proc_fs_opts fs_opts[] = {
1751e8a6c   Linus Torvalds   Rename superblock...
44
45
46
47
  		{ SB_SYNCHRONOUS, ",sync" },
  		{ SB_DIRSYNC, ",dirsync" },
  		{ SB_MANDLOCK, ",mand" },
  		{ SB_LAZYTIME, ",lazytime" },
0226f4923   Al Viro   vfs: take /proc/*...
48
49
  		{ 0, NULL }
  	};
1e88c4201   Alexey Gladkov   proc: rename stru...
50
  	const struct proc_fs_opts *fs_infop;
0226f4923   Al Viro   vfs: take /proc/*...
51

1e88c4201   Alexey Gladkov   proc: rename stru...
52
  	for (fs_infop = fs_opts; fs_infop->flag; fs_infop++) {
0226f4923   Al Viro   vfs: take /proc/*...
53
54
55
56
57
58
59
60
61
  		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)
  {
1e88c4201   Alexey Gladkov   proc: rename stru...
62
  	static const struct proc_fs_opts mnt_opts[] = {
0226f4923   Al Viro   vfs: take /proc/*...
63
64
65
66
67
68
69
70
  		{ MNT_NOSUID, ",nosuid" },
  		{ MNT_NODEV, ",nodev" },
  		{ MNT_NOEXEC, ",noexec" },
  		{ MNT_NOATIME, ",noatime" },
  		{ MNT_NODIRATIME, ",nodiratime" },
  		{ MNT_RELATIME, ",relatime" },
  		{ 0, NULL }
  	};
1e88c4201   Alexey Gladkov   proc: rename stru...
71
  	const struct proc_fs_opts *fs_infop;
0226f4923   Al Viro   vfs: take /proc/*...
72

1e88c4201   Alexey Gladkov   proc: rename stru...
73
  	for (fs_infop = mnt_opts; fs_infop->flag; fs_infop++) {
0226f4923   Al Viro   vfs: take /proc/*...
74
75
76
77
78
79
80
81
82
83
84
85
86
87
  		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);
c7eb68696   David Howells   vfs: subtype hand...
88
  	if (sb->s_subtype) {
0226f4923   Al Viro   vfs: take /proc/*...
89
90
91
92
93
94
95
  		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
  	}
bc98a42c1   David Howells   VFS: Convert sb->...
183
  	seq_puts(m, sb_rdonly(sb) ? " ro" : " rw");
0226f4923   Al Viro   vfs: take /proc/*...
184
185
186
187
  	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;
9f6c61f96   Miklos Szeredi   proc/mounts: add ...
283
284
  	INIT_LIST_HEAD(&p->cursor.mnt_list);
  	p->cursor.mnt.mnt_flags = MNT_CURSOR;
0226f4923   Al Viro   vfs: take /proc/*...
285
286
  
  	return 0;
0226f4923   Al Viro   vfs: take /proc/*...
287
288
289
290
291
292
293
294
295
296
   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_...
297
298
  	struct seq_file *m = file->private_data;
  	struct proc_mounts *p = m->private;
0226f4923   Al Viro   vfs: take /proc/*...
299
  	path_put(&p->root);
9f6c61f96   Miklos Szeredi   proc/mounts: add ...
300
  	mnt_cursor_del(p->ns, &p->cursor);
0226f4923   Al Viro   vfs: take /proc/*...
301
  	put_mnt_ns(p->ns);
ede1bf0dc   Yann Droneaud   fs: use seq_open_...
302
  	return seq_release_private(inode, file);
0226f4923   Al Viro   vfs: take /proc/*...
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
340
341
  }
  
  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,
  };