# This is a BitKeeper generated patch for the following project: # Project Name: Linux kernel tree # This patch format is intended for GNU patch command version 2.5 or higher. # This patch includes the following deltas: # ChangeSet 1.1010 -> 1.1011 # include/linux/ext3_fs.h 1.6 -> 1.7 # include/linux/ext2_fs.h 1.6 -> 1.7 # fs/ext2/super.c 1.9 -> 1.10 # fs/ext3/super.c 1.9 -> 1.10 # # The following is the BitKeeper ChangeSet Log # -------------------------------------------- # 03/03/08 tytso@thank.thunk.org 1.1011 # Default mount options from superblock for ext2/3 filesystems # # This patch adds support for default mount options to be stored in the # superblock, so they don't have to be specified on the mount command line # (or in /etc/fstab). While I was in the code, I also cleaned up the # handling of how mount options are processed in the ext2 and ext3 # filesystems. # # Most mount options are now processed *after* the superblock has been # read in. This allows for a much cleaner handling of those default mount # option parameters that were already stored in the superblock: the # resuid, resgid, and s_errors fields were handled using some fairly gross # special cases. Now the only mount option which is processed first is # the sb option, which specifies the location of the superblock. This # allows the handling of all of the default mount parameters to be much # more cleanly and more generally handled. # # This does change the behaviour from earlier kernels, in that if the sb # mount option is specified, it must be specified *first*. However, this # option is rarely used, and if it is, it generally is specified first, so # this seems to be a reasonable restriction. # -------------------------------------------- # diff -Nru a/fs/ext2/super.c b/fs/ext2/super.c --- a/fs/ext2/super.c Thu Mar 13 00:36:39 2003 +++ b/fs/ext2/super.c Thu Mar 13 00:36:39 2003 @@ -48,16 +48,12 @@ va_start (args, fmt); vsprintf (error_buf, fmt, args); va_end (args); - if (test_opt (sb, ERRORS_PANIC) || - (le16_to_cpu(sb->u.ext2_sb.s_es->s_errors) == EXT2_ERRORS_PANIC && - !test_opt (sb, ERRORS_CONT) && !test_opt (sb, ERRORS_RO))) + if (test_opt (sb, ERRORS_PANIC)) panic ("EXT2-fs panic (device %s): %s: %s\n", bdevname(sb->s_dev), function, error_buf); printk (KERN_CRIT "EXT2-fs error (device %s): %s: %s\n", bdevname(sb->s_dev), function, error_buf); - if (test_opt (sb, ERRORS_RO) || - (le16_to_cpu(sb->u.ext2_sb.s_es->s_errors) == EXT2_ERRORS_RO && - !test_opt (sb, ERRORS_CONT) && !test_opt (sb, ERRORS_PANIC))) { + if (test_opt (sb, ERRORS_RO)) { printk ("Remounting filesystem read-only\n"); sb->s_flags |= MS_RDONLY; } @@ -158,12 +154,61 @@ remount_fs: ext2_remount, }; +static unsigned long get_sb_block(void **data) +{ + unsigned long sb_block; + char *options = (char *) *data; + + if (!options || strncmp(options, "sb=", 3) != 0) + return 1; /* Default location */ + options += 3; + sb_block = simple_strtoul(options, &options, 0); + if (*options && *options != ',') { + printk("EXT2-fs: Invalid sb specification: %s\n", + (char *) *data); + return 1; + } + if (*options == ',') + options++; + *data = (void *) options; + return sb_block; +} + +static int want_value(char *value, char *option) +{ + if (!value || !*value) { + printk(KERN_NOTICE "EXT2-fs: the %s option needs an argument\n", + option); + return -1; + } + return 0; +} + +static int want_null_value(char *value, char *option) +{ + if (*value) { + printk(KERN_NOTICE "EXT2-fs: Invalid %s argument: %s\n", + option, value); + return -1; + } + return 0; +} + +static int want_numeric(char *value, char *option, unsigned long *number) +{ + if (want_value(value, option)) + return -1; + *number = simple_strtoul(value, &value, 0); + if (want_null_value(value, option)) + return -1; + return 0; +} + /* * This function has been shamelessly adapted from the msdos fs */ -static int parse_options (char * options, unsigned long * sb_block, - unsigned short *resuid, unsigned short * resgid, - unsigned long * mount_options) +static int parse_options (char * options, + struct ext2_sb_info *sbi) { char * this_char; char * value; @@ -176,22 +221,22 @@ if ((value = strchr (this_char, '=')) != NULL) *value++ = 0; if (!strcmp (this_char, "bsddf")) - clear_opt (*mount_options, MINIX_DF); + clear_opt (sbi->s_mount_opt, MINIX_DF); else if (!strcmp (this_char, "nouid32")) { - set_opt (*mount_options, NO_UID32); + set_opt (sbi->s_mount_opt, NO_UID32); } else if (!strcmp (this_char, "check")) { if (!value || !*value || !strcmp (value, "none")) - clear_opt (*mount_options, CHECK); + clear_opt (sbi->s_mount_opt, CHECK); else #ifdef CONFIG_EXT2_CHECK - set_opt (*mount_options, CHECK); + set_opt (sbi->s_mount_opt, CHECK); #else printk("EXT2 Check option not supported\n"); #endif } else if (!strcmp (this_char, "debug")) - set_opt (*mount_options, DEBUG); + set_opt (sbi->s_mount_opt, DEBUG); else if (!strcmp (this_char, "errors")) { if (!value || !*value) { printk ("EXT2-fs: the errors option requires " @@ -199,19 +244,19 @@ return 0; } if (!strcmp (value, "continue")) { - clear_opt (*mount_options, ERRORS_RO); - clear_opt (*mount_options, ERRORS_PANIC); - set_opt (*mount_options, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_RO); + clear_opt (sbi->s_mount_opt, ERRORS_PANIC); + set_opt (sbi->s_mount_opt, ERRORS_CONT); } else if (!strcmp (value, "remount-ro")) { - clear_opt (*mount_options, ERRORS_CONT); - clear_opt (*mount_options, ERRORS_PANIC); - set_opt (*mount_options, ERRORS_RO); + clear_opt (sbi->s_mount_opt, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_PANIC); + set_opt (sbi->s_mount_opt, ERRORS_RO); } else if (!strcmp (value, "panic")) { - clear_opt (*mount_options, ERRORS_CONT); - clear_opt (*mount_options, ERRORS_RO); - set_opt (*mount_options, ERRORS_PANIC); + clear_opt (sbi->s_mount_opt, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_RO); + set_opt (sbi->s_mount_opt, ERRORS_PANIC); } else { printk ("EXT2-fs: Invalid errors option: %s\n", @@ -221,52 +266,25 @@ } else if (!strcmp (this_char, "grpid") || !strcmp (this_char, "bsdgroups")) - set_opt (*mount_options, GRPID); + set_opt (sbi->s_mount_opt, GRPID); else if (!strcmp (this_char, "minixdf")) - set_opt (*mount_options, MINIX_DF); + set_opt (sbi->s_mount_opt, MINIX_DF); else if (!strcmp (this_char, "nocheck")) - clear_opt (*mount_options, CHECK); + clear_opt (sbi->s_mount_opt, CHECK); else if (!strcmp (this_char, "nogrpid") || !strcmp (this_char, "sysvgroups")) - clear_opt (*mount_options, GRPID); + clear_opt (sbi->s_mount_opt, GRPID); else if (!strcmp (this_char, "resgid")) { - if (!value || !*value) { - printk ("EXT2-fs: the resgid option requires " - "an argument\n"); + unsigned long v; + if (want_numeric(value, "resgid", &v)) return 0; - } - *resgid = simple_strtoul (value, &value, 0); - if (*value) { - printk ("EXT2-fs: Invalid resgid option: %s\n", - value); - return 0; - } + sbi->s_resgid = v; } else if (!strcmp (this_char, "resuid")) { - if (!value || !*value) { - printk ("EXT2-fs: the resuid option requires " - "an argument"); - return 0; - } - *resuid = simple_strtoul (value, &value, 0); - if (*value) { - printk ("EXT2-fs: Invalid resuid option: %s\n", - value); - return 0; - } - } - else if (!strcmp (this_char, "sb")) { - if (!value || !*value) { - printk ("EXT2-fs: the sb option requires " - "an argument"); - return 0; - } - *sb_block = simple_strtoul (value, &value, 0); - if (*value) { - printk ("EXT2-fs: Invalid sb option: %s\n", - value); + unsigned long v; + if (want_numeric(value, "resuid", &v)) return 0; - } + sbi->s_resuid = v; } /* Silently ignore the quota options */ else if (!strcmp (this_char, "grpquota") @@ -401,13 +419,13 @@ int silent) { struct buffer_head * bh; + struct ext2_sb_info * sbi = EXT2_SB(sb); struct ext2_super_block * es; - unsigned long sb_block = 1; - unsigned short resuid = EXT2_DEF_RESUID; - unsigned short resgid = EXT2_DEF_RESGID; - unsigned long logic_sb_block = 1; - unsigned long offset = 0; + unsigned long sb_block = get_sb_block(&data); + unsigned long logic_sb_block; + unsigned long offset; kdev_t dev = sb->s_dev; + unsigned long def_mount_opts; int blocksize = BLOCK_SIZE; int db_count; int i, j; @@ -423,12 +441,6 @@ if(blocksize < BLOCK_SIZE ) blocksize = BLOCK_SIZE; - sb->u.ext2_sb.s_mount_opt = 0; - if (!parse_options ((char *) data, &sb_block, &resuid, &resgid, - &sb->u.ext2_sb.s_mount_opt)) { - return NULL; - } - if (set_blocksize(dev, blocksize) < 0) { printk ("EXT2-fs: unable to set blocksize %d\n", blocksize); return NULL; @@ -436,14 +448,11 @@ sb->s_blocksize = blocksize; /* - * If the superblock doesn't start on a sector boundary, - * calculate the offset. FIXME(eric) this doesn't make sense - * that we would have to do this. + * If the superblock doesn't start on a hardware sector boundary, + * calculate the offset. */ - if (blocksize != BLOCK_SIZE) { - logic_sb_block = (sb_block*BLOCK_SIZE) / blocksize; - offset = (sb_block*BLOCK_SIZE) % blocksize; - } + logic_sb_block = (sb_block*BLOCK_SIZE) / blocksize; + offset = (sb_block*BLOCK_SIZE) % blocksize; if (!(bh = sb_bread(sb, logic_sb_block))) { printk ("EXT2-fs: unable to read superblock\n"); @@ -462,6 +471,27 @@ bdevname(dev)); goto failed_mount; } + + /* Set defaults before we parse the mount options */ + def_mount_opts = le32_to_cpu(es->s_default_mount_opts); + if (def_mount_opts & EXT2_DEFM_DEBUG) + set_opt(sbi->s_mount_opt, DEBUG); + if (def_mount_opts & EXT2_DEFM_BSDGROUPS) + set_opt(sbi->s_mount_opt, GRPID); + if (def_mount_opts & EXT2_DEFM_UID16) + set_opt(sbi->s_mount_opt, NO_UID32); + + if (le16_to_cpu(sbi->s_es->s_errors) == EXT2_ERRORS_PANIC) + set_opt(sbi->s_mount_opt, ERRORS_PANIC); + else if (le16_to_cpu(sbi->s_es->s_errors) == EXT2_ERRORS_RO) + set_opt(sbi->s_mount_opt, ERRORS_RO); + + sbi->s_resuid = le16_to_cpu(es->s_def_resuid); + sbi->s_resgid = le16_to_cpu(es->s_def_resgid); + + if (!parse_options ((char *) data, sbi)) + goto failed_mount; + if (le32_to_cpu(es->s_rev_level) == EXT2_GOOD_OLD_REV && (EXT2_HAS_COMPAT_FEATURE(sb, ~0U) || EXT2_HAS_RO_COMPAT_FEATURE(sb, ~0U) || @@ -550,14 +580,6 @@ sb->u.ext2_sb.s_desc_per_block = sb->s_blocksize / sizeof (struct ext2_group_desc); sb->u.ext2_sb.s_sbh = bh; - if (resuid != EXT2_DEF_RESUID) - sb->u.ext2_sb.s_resuid = resuid; - else - sb->u.ext2_sb.s_resuid = le16_to_cpu(es->s_def_resuid); - if (resgid != EXT2_DEF_RESGID) - sb->u.ext2_sb.s_resgid = resgid; - else - sb->u.ext2_sb.s_resgid = le16_to_cpu(es->s_def_resgid); sb->u.ext2_sb.s_mount_state = le16_to_cpu(es->s_state); sb->u.ext2_sb.s_addr_per_block_bits = log2 (EXT2_ADDR_PER_BLOCK(sb)); @@ -709,24 +731,16 @@ int ext2_remount (struct super_block * sb, int * flags, char * data) { + struct ext2_sb_info * sbi = EXT2_SB(sb); struct ext2_super_block * es; - unsigned short resuid = sb->u.ext2_sb.s_resuid; - unsigned short resgid = sb->u.ext2_sb.s_resgid; - unsigned long new_mount_opt; - unsigned long tmp; /* * Allow the "check" option to be passed as a remount option. */ - new_mount_opt = sb->u.ext2_sb.s_mount_opt; - if (!parse_options (data, &tmp, &resuid, &resgid, - &new_mount_opt)) + if (!parse_options (data, sbi)) return -EINVAL; - sb->u.ext2_sb.s_mount_opt = new_mount_opt; - sb->u.ext2_sb.s_resuid = resuid; - sb->u.ext2_sb.s_resgid = resgid; - es = sb->u.ext2_sb.s_es; + es = sbi->s_es; if ((*flags & MS_RDONLY) == (sb->s_flags & MS_RDONLY)) return 0; if (*flags & MS_RDONLY) { diff -Nru a/fs/ext3/super.c b/fs/ext3/super.c --- a/fs/ext3/super.c Thu Mar 13 00:36:39 2003 +++ b/fs/ext3/super.c Thu Mar 13 00:36:39 2003 @@ -108,32 +108,6 @@ static char error_buf[1024]; -/* Determine the appropriate response to ext3_error on a given filesystem */ - -static int ext3_error_behaviour(struct super_block *sb) -{ - /* First check for mount-time options */ - if (test_opt (sb, ERRORS_PANIC)) - return EXT3_ERRORS_PANIC; - if (test_opt (sb, ERRORS_RO)) - return EXT3_ERRORS_RO; - if (test_opt (sb, ERRORS_CONT)) - return EXT3_ERRORS_CONTINUE; - - /* If no overrides were specified on the mount, then fall back - * to the default behaviour set in the filesystem's superblock - * on disk. */ - switch (le16_to_cpu(sb->u.ext3_sb.s_es->s_errors)) { - case EXT3_ERRORS_PANIC: - return EXT3_ERRORS_PANIC; - case EXT3_ERRORS_RO: - return EXT3_ERRORS_RO; - default: - break; - } - return EXT3_ERRORS_CONTINUE; -} - /* Deal with the reporting of failure conditions on a filesystem such as * inconsistencies detected or read IO failures. * @@ -159,18 +133,15 @@ if (sb->s_flags & MS_RDONLY) return; - if (ext3_error_behaviour(sb) != EXT3_ERRORS_CONTINUE) { - EXT3_SB(sb)->s_mount_opt |= EXT3_MOUNT_ABORT; - journal_abort(EXT3_SB(sb)->s_journal, -EIO); - } - - if (ext3_error_behaviour(sb) == EXT3_ERRORS_PANIC) + if (test_opt (sb, ERRORS_PANIC)) panic ("EXT3-fs (device %s): panic forced after error\n", bdevname(sb->s_dev)); - - if (ext3_error_behaviour(sb) == EXT3_ERRORS_RO) { + if (test_opt (sb, ERRORS_RO)) { printk (KERN_CRIT "Remounting filesystem read-only\n"); sb->s_flags |= MS_RDONLY; + } else { + EXT3_SB(sb)->s_mount_opt |= EXT3_MOUNT_ABORT; + journal_abort(EXT3_SB(sb)->s_journal, -EIO); } ext3_commit_super(sb, es, 1); @@ -260,7 +231,7 @@ vsprintf (error_buf, fmt, args); va_end (args); - if (ext3_error_behaviour(sb) == EXT3_ERRORS_PANIC) + if (test_opt (sb, ERRORS_PANIC)) panic ("EXT3-fs panic (device %s): %s: %s\n", bdevname(sb->s_dev), function, error_buf); @@ -493,17 +464,32 @@ return 0; } +static unsigned long get_sb_block(void **data) +{ + unsigned long sb_block; + char *options = (char *) *data; + + if (!options || strncmp(options, "sb=", 3) != 0) + return 1; /* Default location */ + options += 3; + sb_block = simple_strtoul(options, &options, 0); + if (*options && *options != ',') { + printk("EXT3-fs: Invalid sb specification: %s\n", + (char *) *data); + return 1; + } + if (*options == ',') + options++; + *data = (void *) options; + return sb_block; +} + /* * This function has been shamelessly adapted from the msdos fs */ -static int parse_options (char * options, unsigned long * sb_block, - struct ext3_sb_info *sbi, - unsigned long * inum, - int is_remount) -{ - unsigned long *mount_options = &sbi->s_mount_opt; - uid_t *resuid = &sbi->s_resuid; - gid_t *resgid = &sbi->s_resgid; +static int parse_options (char * options, struct ext3_sb_info *sbi, + unsigned long * inum, int is_remount) +{ char * this_char; char * value; @@ -515,42 +501,42 @@ if ((value = strchr (this_char, '=')) != NULL) *value++ = 0; if (!strcmp (this_char, "bsddf")) - clear_opt (*mount_options, MINIX_DF); + clear_opt (sbi->s_mount_opt, MINIX_DF); else if (!strcmp (this_char, "nouid32")) { - set_opt (*mount_options, NO_UID32); + set_opt (sbi->s_mount_opt, NO_UID32); } else if (!strcmp (this_char, "abort")) - set_opt (*mount_options, ABORT); + set_opt (sbi->s_mount_opt, ABORT); else if (!strcmp (this_char, "check")) { if (!value || !*value || !strcmp (value, "none")) - clear_opt (*mount_options, CHECK); + clear_opt (sbi->s_mount_opt, CHECK); else #ifdef CONFIG_EXT3_CHECK - set_opt (*mount_options, CHECK); + set_opt (sbi->s_mount_opt, CHECK); #else printk(KERN_ERR "EXT3 Check option not supported\n"); #endif } else if (!strcmp (this_char, "debug")) - set_opt (*mount_options, DEBUG); + set_opt (sbi->s_mount_opt, DEBUG); else if (!strcmp (this_char, "errors")) { if (want_value(value, "errors")) return 0; if (!strcmp (value, "continue")) { - clear_opt (*mount_options, ERRORS_RO); - clear_opt (*mount_options, ERRORS_PANIC); - set_opt (*mount_options, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_RO); + clear_opt (sbi->s_mount_opt, ERRORS_PANIC); + set_opt (sbi->s_mount_opt, ERRORS_CONT); } else if (!strcmp (value, "remount-ro")) { - clear_opt (*mount_options, ERRORS_CONT); - clear_opt (*mount_options, ERRORS_PANIC); - set_opt (*mount_options, ERRORS_RO); + clear_opt (sbi->s_mount_opt, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_PANIC); + set_opt (sbi->s_mount_opt, ERRORS_RO); } else if (!strcmp (value, "panic")) { - clear_opt (*mount_options, ERRORS_CONT); - clear_opt (*mount_options, ERRORS_RO); - set_opt (*mount_options, ERRORS_PANIC); + clear_opt (sbi->s_mount_opt, ERRORS_CONT); + clear_opt (sbi->s_mount_opt, ERRORS_RO); + set_opt (sbi->s_mount_opt, ERRORS_PANIC); } else { printk (KERN_ERR @@ -561,29 +547,25 @@ } else if (!strcmp (this_char, "grpid") || !strcmp (this_char, "bsdgroups")) - set_opt (*mount_options, GRPID); + set_opt (sbi->s_mount_opt, GRPID); else if (!strcmp (this_char, "minixdf")) - set_opt (*mount_options, MINIX_DF); + set_opt (sbi->s_mount_opt, MINIX_DF); else if (!strcmp (this_char, "nocheck")) - clear_opt (*mount_options, CHECK); + clear_opt (sbi->s_mount_opt, CHECK); else if (!strcmp (this_char, "nogrpid") || !strcmp (this_char, "sysvgroups")) - clear_opt (*mount_options, GRPID); + clear_opt (sbi->s_mount_opt, GRPID); else if (!strcmp (this_char, "resgid")) { unsigned long v; if (want_numeric(value, "resgid", &v)) return 0; - *resgid = v; + sbi->s_resgid = v; } else if (!strcmp (this_char, "resuid")) { unsigned long v; if (want_numeric(value, "resuid", &v)) return 0; - *resuid = v; - } - else if (!strcmp (this_char, "sb")) { - if (want_numeric(value, "sb", sb_block)) - return 0; + sbi->s_resuid = v; } #ifdef CONFIG_JBD_DEBUG else if (!strcmp (this_char, "ro-after")) { @@ -614,12 +596,12 @@ if (want_value(value, "journal")) return 0; if (!strcmp (value, "update")) - set_opt (*mount_options, UPDATE_JOURNAL); + set_opt (sbi->s_mount_opt, UPDATE_JOURNAL); else if (want_numeric(value, "journal", inum)) return 0; } else if (!strcmp (this_char, "noload")) - set_opt (*mount_options, NOLOAD); + set_opt (sbi->s_mount_opt, NOLOAD); else if (!strcmp (this_char, "data")) { int data_opt = 0; @@ -638,7 +620,7 @@ return 0; } if (is_remount) { - if ((*mount_options & EXT3_MOUNT_DATA_FLAGS) != + if ((sbi->s_mount_opt & EXT3_MOUNT_DATA_FLAGS) != data_opt) { printk(KERN_ERR "EXT3-fs: cannot change data " @@ -646,8 +628,8 @@ return 0; } } else { - *mount_options &= ~EXT3_MOUNT_DATA_FLAGS; - *mount_options |= data_opt; + sbi->s_mount_opt &= ~EXT3_MOUNT_DATA_FLAGS; + sbi->s_mount_opt |= data_opt; } } else if (!strcmp (this_char, "commit")) { unsigned long v; @@ -902,11 +884,12 @@ struct buffer_head * bh; struct ext3_super_block *es = 0; struct ext3_sb_info *sbi = EXT3_SB(sb); - unsigned long sb_block = 1; - unsigned long logic_sb_block = 1; - unsigned long offset = 0; + unsigned long sb_block = get_sb_block(&data); + unsigned long logic_sb_block; + unsigned long offset; unsigned long journal_inum = 0; kdev_t dev = sb->s_dev; + unsigned long def_mount_opts; int blocksize; int hblock; int db_count; @@ -928,14 +911,6 @@ if (blocksize < hblock) blocksize = hblock; - sbi->s_mount_opt = 0; - sbi->s_resuid = EXT3_DEF_RESUID; - sbi->s_resgid = EXT3_DEF_RESGID; - if (!parse_options ((char *) data, &sb_block, sbi, &journal_inum, 0)) { - sb->s_dev = 0; - goto out_fail; - } - sb->s_blocksize = blocksize; set_blocksize (dev, blocksize); @@ -943,10 +918,8 @@ * The ext3 superblock will not be buffer aligned for other than 1kB * block sizes. We need to calculate the offset from buffer start. */ - if (blocksize != EXT3_MIN_BLOCK_SIZE) { - logic_sb_block = (sb_block * EXT3_MIN_BLOCK_SIZE) / blocksize; - offset = (sb_block * EXT3_MIN_BLOCK_SIZE) % blocksize; - } + logic_sb_block = (sb_block * EXT3_MIN_BLOCK_SIZE) / blocksize; + offset = (sb_block * EXT3_MIN_BLOCK_SIZE) % blocksize; if (!(bh = sb_bread(sb, logic_sb_block))) { printk (KERN_ERR "EXT3-fs: unable to read superblock\n"); @@ -966,6 +939,34 @@ bdevname(dev)); goto failed_mount; } + + /* Set defaults before we parse the mount options */ + def_mount_opts = le32_to_cpu(es->s_default_mount_opts); + sbi->s_mount_opt = 0; + if (def_mount_opts & EXT3_DEFM_DEBUG) + set_opt(sbi->s_mount_opt, DEBUG); + if (def_mount_opts & EXT3_DEFM_BSDGROUPS) + set_opt(sbi->s_mount_opt, GRPID); + if (def_mount_opts & EXT3_DEFM_UID16) + set_opt(sbi->s_mount_opt, NO_UID32); + if ((def_mount_opts & EXT3_DEFM_JMODE) == EXT3_DEFM_JMODE_DATA) + sbi->s_mount_opt |= EXT3_MOUNT_JOURNAL_DATA; + else if ((def_mount_opts & EXT3_DEFM_JMODE) == EXT3_DEFM_JMODE_ORDERED) + sbi->s_mount_opt |= EXT3_MOUNT_ORDERED_DATA; + else if ((def_mount_opts & EXT3_DEFM_JMODE) == EXT3_DEFM_JMODE_WBACK) + sbi->s_mount_opt |= EXT3_MOUNT_WRITEBACK_DATA; + + if (le16_to_cpu(sbi->s_es->s_errors) == EXT3_ERRORS_PANIC) + set_opt(sbi->s_mount_opt, ERRORS_PANIC); + else if (le16_to_cpu(sbi->s_es->s_errors) == EXT3_ERRORS_RO) + set_opt(sbi->s_mount_opt, ERRORS_RO); + + sbi->s_resuid = le16_to_cpu(es->s_def_resuid); + sbi->s_resgid = le16_to_cpu(es->s_def_resgid); + + if (!parse_options ((char *) data, sbi, &journal_inum, 0)) + goto failed_mount; + if (le32_to_cpu(es->s_rev_level) == EXT3_GOOD_OLD_REV && (EXT3_HAS_COMPAT_FEATURE(sb, ~0U) || EXT3_HAS_RO_COMPAT_FEATURE(sb, ~0U) || @@ -1065,10 +1066,6 @@ sbi->s_itb_per_group = sbi->s_inodes_per_group /sbi->s_inodes_per_block; sbi->s_desc_per_block = blocksize / sizeof(struct ext3_group_desc); sbi->s_sbh = bh; - if (sbi->s_resuid == EXT3_DEF_RESUID) - sbi->s_resuid = le16_to_cpu(es->s_def_resuid); - if (sbi->s_resgid == EXT3_DEF_RESGID) - sbi->s_resgid = le16_to_cpu(es->s_def_resgid); sbi->s_mount_state = le16_to_cpu(es->s_state); sbi->s_addr_per_block_bits = log2(EXT3_ADDR_PER_BLOCK(sb)); sbi->s_desc_per_block_bits = log2(EXT3_DESC_PER_BLOCK(sb)); @@ -1658,7 +1655,7 @@ /* * Allow the "check" option to be passed as a remount option. */ - if (!parse_options(data, &tmp, sbi, &tmp, 1)) + if (!parse_options(data, sbi, &tmp, 1)) return -EINVAL; if (sbi->s_mount_opt & EXT3_MOUNT_ABORT) diff -Nru a/include/linux/ext2_fs.h b/include/linux/ext2_fs.h --- a/include/linux/ext2_fs.h Thu Mar 13 00:36:39 2003 +++ b/include/linux/ext2_fs.h Thu Mar 13 00:36:39 2003 @@ -392,7 +392,19 @@ __u8 s_prealloc_blocks; /* Nr of blocks to try to preallocate*/ __u8 s_prealloc_dir_blocks; /* Nr to preallocate for dirs */ __u16 s_padding1; - __u32 s_reserved[204]; /* Padding to the end of the block */ + /* + * Journaling support valid if EXT3_FEATURE_COMPAT_HAS_JOURNAL set. + */ + __u8 s_journal_uuid[16]; /* uuid of journal superblock */ + __u32 s_journal_inum; /* inode number of journal file */ + __u32 s_journal_dev; /* device number of journal file */ + __u32 s_last_orphan; /* start of list of inodes to delete */ + __u32 s_hash_seed[4]; /* HTREE hash seed */ + __u8 s_def_hash_version; /* Default hash version to use */ + __u8 s_reserved_char_pad; + __u16 s_reserved_word_pad; + __u32 s_default_mount_opts; + __u32 s_reserved[191]; /* Padding to the end of the block */ }; #ifdef __KERNEL__ @@ -475,10 +487,18 @@ #define EXT2_FEATURE_INCOMPAT_UNSUPPORTED ~EXT2_FEATURE_INCOMPAT_SUPP /* - * Default values for user and/or group using reserved blocks + * Default mount options */ -#define EXT2_DEF_RESUID 0 -#define EXT2_DEF_RESGID 0 +#define EXT2_DEFM_DEBUG 0x0001 +#define EXT2_DEFM_BSDGROUPS 0x0002 +#define EXT2_DEFM_XATTR_USER 0x0004 +#define EXT2_DEFM_ACL 0x0008 +#define EXT2_DEFM_UID16 0x0010 + /* Not used by ext2, but reserved for use by ext3 */ +#define EXT3_DEFM_JMODE 0x0060 +#define EXT3_DEFM_JMODE_DATA 0x0020 +#define EXT3_DEFM_JMODE_ORDERED 0x0040 +#define EXT3_DEFM_JMODE_WBACK 0x0060 /* * Structure of a directory entry diff -Nru a/include/linux/ext3_fs.h b/include/linux/ext3_fs.h --- a/include/linux/ext3_fs.h Thu Mar 13 00:36:39 2003 +++ b/include/linux/ext3_fs.h Thu Mar 13 00:36:39 2003 @@ -446,7 +446,8 @@ __u8 s_def_hash_version; /* Default hash version to use */ __u8 s_reserved_char_pad; __u16 s_reserved_word_pad; - __u32 s_reserved[192]; /* Padding to the end of the block */ + __u32 s_default_mount_opts; + __u32 s_reserved[191]; /* Padding to the end of the block */ }; #ifdef __KERNEL__ @@ -528,10 +529,17 @@ EXT3_FEATURE_RO_COMPAT_BTREE_DIR) /* - * Default values for user and/or group using reserved blocks + * Default mount options */ -#define EXT3_DEF_RESUID 0 -#define EXT3_DEF_RESGID 0 +#define EXT3_DEFM_DEBUG 0x0001 +#define EXT3_DEFM_BSDGROUPS 0x0002 +#define EXT3_DEFM_XATTR_USER 0x0004 +#define EXT3_DEFM_ACL 0x0008 +#define EXT3_DEFM_UID16 0x0010 +#define EXT3_DEFM_JMODE 0x0060 +#define EXT3_DEFM_JMODE_DATA 0x0020 +#define EXT3_DEFM_JMODE_ORDERED 0x0040 +#define EXT3_DEFM_JMODE_WBACK 0x0060 /* * Structure of a directory entry