aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ocfs2
diff options
context:
space:
mode:
authorPatrick J. LoPresti <lopresti@gmail.com>2010-07-22 18:05:57 -0400
committerJoel Becker <joel.becker@oracle.com>2010-09-10 11:42:10 -0400
commit3bdb8efd94a73bb137e3315cd831cbc874052b4b (patch)
treeab9d273ca7a066fbb36884a703016c0eaaabe2c7 /fs/ocfs2
parent1113e1b504f6e8d4364c0b73c9097828067d4617 (diff)
OCFS2: Allow huge (> 16 TiB) volumes to mount
The OCFS2 developers have already done all of the hard work to allow volumes larger than 16 TiB. But there is still a "sanity check" in fs/ocfs2/super.c that prevents the mounting of such volumes, even when the cluster size and journal options would allow it. This patch replaces that sanity check with a more sophisticated one to mount a huge volume provided that (a) it is addressable by the raw word/address size of the system (borrowing a test from ext4); (b) the volume is using JBD2; and (c) the JBD2_FEATURE_INCOMPAT_64BIT flag is set on the journal. I factored out the sanity check into its own function. I also moved it from ocfs2_initialize_super() down to ocfs2_check_volume(); any earlier, and the journal will not have been initialized yet. This patch is one of a pair, and it depends on the other ("JBD2: Allow feature checks before journal recovery"). I have tested this patch on small volumes, huge volumes, and huge volumes without 64-bit block support in the journal. All of them appear to work or to fail gracefully, as appropriate. Signed-off-by: Patrick LoPresti <lopresti@gmail.com> Signed-off-by: Joel Becker <joel.becker@oracle.com>
Diffstat (limited to 'fs/ocfs2')
-rw-r--r--fs/ocfs2/super.c51
1 files changed, 46 insertions, 5 deletions
diff --git a/fs/ocfs2/super.c b/fs/ocfs2/super.c
index fa1be1b304d1..47415398d56a 100644
--- a/fs/ocfs2/super.c
+++ b/fs/ocfs2/super.c
@@ -1990,6 +1990,36 @@ static int ocfs2_setup_osb_uuid(struct ocfs2_super *osb, const unsigned char *uu
1990 return 0; 1990 return 0;
1991} 1991}
1992 1992
1993/* Make sure entire volume is addressable by our journal. Requires
1994 osb_clusters_at_boot to be valid and for the journal to have been
1995 initialized by ocfs2_journal_init(). */
1996static int ocfs2_journal_addressable(struct ocfs2_super *osb)
1997{
1998 int status = 0;
1999 u64 max_block =
2000 ocfs2_clusters_to_blocks(osb->sb,
2001 osb->osb_clusters_at_boot) - 1;
2002
2003 /* 32-bit block number is always OK. */
2004 if (max_block <= (u32)~0ULL)
2005 goto out;
2006
2007 /* Volume is "huge", so see if our journal is new enough to
2008 support it. */
2009 if (!(OCFS2_HAS_COMPAT_FEATURE(osb->sb,
2010 OCFS2_FEATURE_COMPAT_JBD2_SB) &&
2011 jbd2_journal_check_used_features(osb->journal->j_journal, 0, 0,
2012 JBD2_FEATURE_INCOMPAT_64BIT))) {
2013 mlog(ML_ERROR, "The journal cannot address the entire volume. "
2014 "Enable the 'block64' journal option with tunefs.ocfs2");
2015 status = -EFBIG;
2016 goto out;
2017 }
2018
2019 out:
2020 return status;
2021}
2022
1993static int ocfs2_initialize_super(struct super_block *sb, 2023static int ocfs2_initialize_super(struct super_block *sb,
1994 struct buffer_head *bh, 2024 struct buffer_head *bh,
1995 int sector_size, 2025 int sector_size,
@@ -2002,6 +2032,7 @@ static int ocfs2_initialize_super(struct super_block *sb,
2002 struct ocfs2_journal *journal; 2032 struct ocfs2_journal *journal;
2003 __le32 uuid_net_key; 2033 __le32 uuid_net_key;
2004 struct ocfs2_super *osb; 2034 struct ocfs2_super *osb;
2035 u64 total_blocks;
2005 2036
2006 mlog_entry_void(); 2037 mlog_entry_void();
2007 2038
@@ -2214,11 +2245,15 @@ static int ocfs2_initialize_super(struct super_block *sb,
2214 goto bail; 2245 goto bail;
2215 } 2246 }
2216 2247
2217 if (ocfs2_clusters_to_blocks(osb->sb, le32_to_cpu(di->i_clusters) - 1) 2248 total_blocks = ocfs2_clusters_to_blocks(osb->sb,
2218 > (u32)~0UL) { 2249 le32_to_cpu(di->i_clusters));
2219 mlog(ML_ERROR, "Volume might try to write to blocks beyond " 2250
2220 "what jbd can address in 32 bits.\n"); 2251 status = generic_check_addressable(osb->sb->s_blocksize_bits,
2221 status = -EINVAL; 2252 total_blocks);
2253 if (status) {
2254 mlog(ML_ERROR, "Volume too large "
2255 "to mount safely on this system");
2256 status = -EFBIG;
2222 goto bail; 2257 goto bail;
2223 } 2258 }
2224 2259
@@ -2380,6 +2415,12 @@ static int ocfs2_check_volume(struct ocfs2_super *osb)
2380 goto finally; 2415 goto finally;
2381 } 2416 }
2382 2417
2418 /* Now that journal has been initialized, check to make sure
2419 entire volume is addressable. */
2420 status = ocfs2_journal_addressable(osb);
2421 if (status)
2422 goto finally;
2423
2383 /* If the journal was unmounted cleanly then we don't want to 2424 /* If the journal was unmounted cleanly then we don't want to
2384 * recover anything. Otherwise, journal_load will do that 2425 * recover anything. Otherwise, journal_load will do that
2385 * dirty work for us :) */ 2426 * dirty work for us :) */