mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 a068acf2ee
			
		
	
	
		a068acf2ee
		
	
	
	
	
		
			
			Many file systems that implement the show_options hook fail to correctly escape their output which could lead to unescaped characters (e.g. new lines) leaking into /proc/mounts and /proc/[pid]/mountinfo files. This could lead to confusion, spoofed entries (resulting in things like systemd issuing false d-bus "mount" notifications), and who knows what else. This looks like it would only be the root user stepping on themselves, but it's possible weird things could happen in containers or in other situations with delegated mount privileges. Here's an example using overlay with setuid fusermount trusting the contents of /proc/mounts (via the /etc/mtab symlink). Imagine the use of "sudo" is something more sneaky: $ BASE="ovl" $ MNT="$BASE/mnt" $ LOW="$BASE/lower" $ UP="$BASE/upper" $ WORK="$BASE/work/ 0 0 none /proc fuse.pwn user_id=1000" $ mkdir -p "$LOW" "$UP" "$WORK" $ sudo mount -t overlay -o "lowerdir=$LOW,upperdir=$UP,workdir=$WORK" none /mnt $ cat /proc/mounts none /root/ovl/mnt overlay rw,relatime,lowerdir=ovl/lower,upperdir=ovl/upper,workdir=ovl/work/ 0 0 none /proc fuse.pwn user_id=1000 0 0 $ fusermount -u /proc $ cat /proc/mounts cat: /proc/mounts: No such file or directory This fixes the problem by adding new seq_show_option and seq_show_option_n helpers, and updating the vulnerable show_option handlers to use them as needed. Some, like SELinux, need to be open coded due to unusual existing escape mechanisms. [akpm@linux-foundation.org: add lost chunk, per Kees] [keescook@chromium.org: seq_show_option should be using const parameters] Signed-off-by: Kees Cook <keescook@chromium.org> Acked-by: Serge Hallyn <serge.hallyn@canonical.com> Acked-by: Jan Kara <jack@suse.com> Acked-by: Paul Moore <paul@paul-moore.com> Cc: J. R. Okajima <hooanon05g@gmail.com> Signed-off-by: Kees Cook <keescook@chromium.org> Cc: <stable@vger.kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			239 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  linux/fs/hfsplus/options.c
 | |
|  *
 | |
|  * Copyright (C) 2001
 | |
|  * Brad Boyer (flar@allandria.com)
 | |
|  * (C) 2003 Ardis Technologies <roman@ardistech.com>
 | |
|  *
 | |
|  * Option parsing
 | |
|  */
 | |
| 
 | |
| #include <linux/string.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/sched.h>
 | |
| #include <linux/parser.h>
 | |
| #include <linux/nls.h>
 | |
| #include <linux/mount.h>
 | |
| #include <linux/seq_file.h>
 | |
| #include <linux/slab.h>
 | |
| #include "hfsplus_fs.h"
 | |
| 
 | |
| enum {
 | |
| 	opt_creator, opt_type,
 | |
| 	opt_umask, opt_uid, opt_gid,
 | |
| 	opt_part, opt_session, opt_nls,
 | |
| 	opt_nodecompose, opt_decompose,
 | |
| 	opt_barrier, opt_nobarrier,
 | |
| 	opt_force, opt_err
 | |
| };
 | |
| 
 | |
| static const match_table_t tokens = {
 | |
| 	{ opt_creator, "creator=%s" },
 | |
| 	{ opt_type, "type=%s" },
 | |
| 	{ opt_umask, "umask=%o" },
 | |
| 	{ opt_uid, "uid=%u" },
 | |
| 	{ opt_gid, "gid=%u" },
 | |
| 	{ opt_part, "part=%u" },
 | |
| 	{ opt_session, "session=%u" },
 | |
| 	{ opt_nls, "nls=%s" },
 | |
| 	{ opt_decompose, "decompose" },
 | |
| 	{ opt_nodecompose, "nodecompose" },
 | |
| 	{ opt_barrier, "barrier" },
 | |
| 	{ opt_nobarrier, "nobarrier" },
 | |
| 	{ opt_force, "force" },
 | |
| 	{ opt_err, NULL }
 | |
| };
 | |
| 
 | |
| /* Initialize an options object to reasonable defaults */
 | |
| void hfsplus_fill_defaults(struct hfsplus_sb_info *opts)
 | |
| {
 | |
| 	if (!opts)
 | |
| 		return;
 | |
| 
 | |
| 	opts->creator = HFSPLUS_DEF_CR_TYPE;
 | |
| 	opts->type = HFSPLUS_DEF_CR_TYPE;
 | |
| 	opts->umask = current_umask();
 | |
| 	opts->uid = current_uid();
 | |
| 	opts->gid = current_gid();
 | |
| 	opts->part = -1;
 | |
| 	opts->session = -1;
 | |
| }
 | |
| 
 | |
| /* convert a "four byte character" to a 32 bit int with error checks */
 | |
| static inline int match_fourchar(substring_t *arg, u32 *result)
 | |
| {
 | |
| 	if (arg->to - arg->from != 4)
 | |
| 		return -EINVAL;
 | |
| 	memcpy(result, arg->from, 4);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int hfsplus_parse_options_remount(char *input, int *force)
 | |
| {
 | |
| 	char *p;
 | |
| 	substring_t args[MAX_OPT_ARGS];
 | |
| 	int token;
 | |
| 
 | |
| 	if (!input)
 | |
| 		return 1;
 | |
| 
 | |
| 	while ((p = strsep(&input, ",")) != NULL) {
 | |
| 		if (!*p)
 | |
| 			continue;
 | |
| 
 | |
| 		token = match_token(p, tokens, args);
 | |
| 		switch (token) {
 | |
| 		case opt_force:
 | |
| 			*force = 1;
 | |
| 			break;
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| /* Parse options from mount. Returns 0 on failure */
 | |
| /* input is the options passed to mount() as a string */
 | |
| int hfsplus_parse_options(char *input, struct hfsplus_sb_info *sbi)
 | |
| {
 | |
| 	char *p;
 | |
| 	substring_t args[MAX_OPT_ARGS];
 | |
| 	int tmp, token;
 | |
| 
 | |
| 	if (!input)
 | |
| 		goto done;
 | |
| 
 | |
| 	while ((p = strsep(&input, ",")) != NULL) {
 | |
| 		if (!*p)
 | |
| 			continue;
 | |
| 
 | |
| 		token = match_token(p, tokens, args);
 | |
| 		switch (token) {
 | |
| 		case opt_creator:
 | |
| 			if (match_fourchar(&args[0], &sbi->creator)) {
 | |
| 				pr_err("creator requires a 4 character value\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_type:
 | |
| 			if (match_fourchar(&args[0], &sbi->type)) {
 | |
| 				pr_err("type requires a 4 character value\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_umask:
 | |
| 			if (match_octal(&args[0], &tmp)) {
 | |
| 				pr_err("umask requires a value\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			sbi->umask = (umode_t)tmp;
 | |
| 			break;
 | |
| 		case opt_uid:
 | |
| 			if (match_int(&args[0], &tmp)) {
 | |
| 				pr_err("uid requires an argument\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			sbi->uid = make_kuid(current_user_ns(), (uid_t)tmp);
 | |
| 			if (!uid_valid(sbi->uid)) {
 | |
| 				pr_err("invalid uid specified\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_gid:
 | |
| 			if (match_int(&args[0], &tmp)) {
 | |
| 				pr_err("gid requires an argument\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			sbi->gid = make_kgid(current_user_ns(), (gid_t)tmp);
 | |
| 			if (!gid_valid(sbi->gid)) {
 | |
| 				pr_err("invalid gid specified\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_part:
 | |
| 			if (match_int(&args[0], &sbi->part)) {
 | |
| 				pr_err("part requires an argument\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_session:
 | |
| 			if (match_int(&args[0], &sbi->session)) {
 | |
| 				pr_err("session requires an argument\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case opt_nls:
 | |
| 			if (sbi->nls) {
 | |
| 				pr_err("unable to change nls mapping\n");
 | |
| 				return 0;
 | |
| 			}
 | |
| 			p = match_strdup(&args[0]);
 | |
| 			if (p)
 | |
| 				sbi->nls = load_nls(p);
 | |
| 			if (!sbi->nls) {
 | |
| 				pr_err("unable to load nls mapping \"%s\"\n",
 | |
| 				       p);
 | |
| 				kfree(p);
 | |
| 				return 0;
 | |
| 			}
 | |
| 			kfree(p);
 | |
| 			break;
 | |
| 		case opt_decompose:
 | |
| 			clear_bit(HFSPLUS_SB_NODECOMPOSE, &sbi->flags);
 | |
| 			break;
 | |
| 		case opt_nodecompose:
 | |
| 			set_bit(HFSPLUS_SB_NODECOMPOSE, &sbi->flags);
 | |
| 			break;
 | |
| 		case opt_barrier:
 | |
| 			clear_bit(HFSPLUS_SB_NOBARRIER, &sbi->flags);
 | |
| 			break;
 | |
| 		case opt_nobarrier:
 | |
| 			set_bit(HFSPLUS_SB_NOBARRIER, &sbi->flags);
 | |
| 			break;
 | |
| 		case opt_force:
 | |
| 			set_bit(HFSPLUS_SB_FORCE, &sbi->flags);
 | |
| 			break;
 | |
| 		default:
 | |
| 			return 0;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| done:
 | |
| 	if (!sbi->nls) {
 | |
| 		/* try utf8 first, as this is the old default behaviour */
 | |
| 		sbi->nls = load_nls("utf8");
 | |
| 		if (!sbi->nls)
 | |
| 			sbi->nls = load_nls_default();
 | |
| 		if (!sbi->nls)
 | |
| 			return 0;
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| int hfsplus_show_options(struct seq_file *seq, struct dentry *root)
 | |
| {
 | |
| 	struct hfsplus_sb_info *sbi = HFSPLUS_SB(root->d_sb);
 | |
| 
 | |
| 	if (sbi->creator != HFSPLUS_DEF_CR_TYPE)
 | |
| 		seq_show_option_n(seq, "creator", (char *)&sbi->creator, 4);
 | |
| 	if (sbi->type != HFSPLUS_DEF_CR_TYPE)
 | |
| 		seq_show_option_n(seq, "type", (char *)&sbi->type, 4);
 | |
| 	seq_printf(seq, ",umask=%o,uid=%u,gid=%u", sbi->umask,
 | |
| 			from_kuid_munged(&init_user_ns, sbi->uid),
 | |
| 			from_kgid_munged(&init_user_ns, sbi->gid));
 | |
| 	if (sbi->part >= 0)
 | |
| 		seq_printf(seq, ",part=%u", sbi->part);
 | |
| 	if (sbi->session >= 0)
 | |
| 		seq_printf(seq, ",session=%u", sbi->session);
 | |
| 	if (sbi->nls)
 | |
| 		seq_printf(seq, ",nls=%s", sbi->nls->charset);
 | |
| 	if (test_bit(HFSPLUS_SB_NODECOMPOSE, &sbi->flags))
 | |
| 		seq_puts(seq, ",nodecompose");
 | |
| 	if (test_bit(HFSPLUS_SB_NOBARRIER, &sbi->flags))
 | |
| 		seq_puts(seq, ",nobarrier");
 | |
| 	return 0;
 | |
| }
 |