Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 1 | /* |
| 2 | * openfs.c --- open an ext2 filesystem |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 3 | * |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 4 | * Copyright (C) 1993, 1994, 1995, 1996 Theodore Ts'o. |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 5 | * |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 6 | * %Begin-Header% |
Theodore Ts'o | 543547a | 2010-05-17 21:31:56 -0400 | [diff] [blame] | 7 | * This file may be redistributed under the terms of the GNU Library |
| 8 | * General Public License, version 2. |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 9 | * %End-Header% |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 10 | */ |
| 11 | |
| 12 | #include <stdio.h> |
| 13 | #include <string.h> |
Theodore Ts'o | 50e1e10 | 1997-04-26 13:58:21 +0000 | [diff] [blame] | 14 | #if HAVE_UNISTD_H |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 15 | #include <unistd.h> |
Theodore Ts'o | 50e1e10 | 1997-04-26 13:58:21 +0000 | [diff] [blame] | 16 | #endif |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 17 | #include <fcntl.h> |
| 18 | #include <time.h> |
Theodore Ts'o | 1d2ff46 | 1997-10-19 23:00:21 +0000 | [diff] [blame] | 19 | #if HAVE_SYS_STAT_H |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 20 | #include <sys/stat.h> |
Theodore Ts'o | 1d2ff46 | 1997-10-19 23:00:21 +0000 | [diff] [blame] | 21 | #endif |
| 22 | #if HAVE_SYS_TYPES_H |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 23 | #include <sys/types.h> |
Theodore Ts'o | 1d2ff46 | 1997-10-19 23:00:21 +0000 | [diff] [blame] | 24 | #endif |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 25 | #ifdef HAVE_ERRNO_H |
| 26 | #include <errno.h> |
| 27 | #endif |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 28 | |
Theodore Ts'o | b5abe6f | 1998-01-19 14:47:53 +0000 | [diff] [blame] | 29 | #include "ext2_fs.h" |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 30 | |
| 31 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 32 | #include "ext2fs.h" |
Theodore Ts'o | a78926e | 2001-05-03 04:02:29 +0000 | [diff] [blame] | 33 | #include "e2image.h" |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 34 | |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 35 | blk64_t ext2fs_descriptor_block_loc2(ext2_filsys fs, blk64_t group_block, |
| 36 | dgrp_t i) |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 37 | { |
| 38 | int bg; |
| 39 | int has_super = 0; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 40 | blk64_t ret_blk; |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 41 | |
| 42 | if (!(fs->super->s_feature_incompat & EXT2_FEATURE_INCOMPAT_META_BG) || |
| 43 | (i < fs->super->s_first_meta_bg)) |
| 44 | return (group_block + i + 1); |
| 45 | |
Valerie Clement | f2de1d3 | 2007-08-30 17:38:13 +0200 | [diff] [blame] | 46 | bg = EXT2_DESC_PER_BLOCK(fs->super) * i; |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 47 | if (ext2fs_bg_has_super(fs, bg)) |
| 48 | has_super = 1; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 49 | ret_blk = ext2fs_group_first_block2(fs, bg) + has_super; |
Theodore Ts'o | 9ed06a1 | 2002-10-31 11:45:06 -0500 | [diff] [blame] | 50 | /* |
| 51 | * If group_block is not the normal value, we're trying to use |
| 52 | * the backup group descriptors and superblock --- so use the |
| 53 | * alternate location of the second block group in the |
| 54 | * metablock group. Ideally we should be testing each bg |
| 55 | * descriptor block individually for correctness, but we don't |
| 56 | * have the infrastructure in place to do that. |
| 57 | */ |
| 58 | if (group_block != fs->super->s_first_data_block && |
| 59 | ((ret_blk + fs->super->s_blocks_per_group) < |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 60 | ext2fs_blocks_count(fs->super))) |
Theodore Ts'o | 9ed06a1 | 2002-10-31 11:45:06 -0500 | [diff] [blame] | 61 | ret_blk += fs->super->s_blocks_per_group; |
| 62 | return ret_blk; |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 63 | } |
| 64 | |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 65 | blk_t ext2fs_descriptor_block_loc(ext2_filsys fs, blk_t group_block, dgrp_t i) |
| 66 | { |
| 67 | return ext2fs_descriptor_block_loc2(fs, group_block, i); |
| 68 | } |
| 69 | |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 70 | errcode_t ext2fs_open(const char *name, int flags, int superblock, |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 71 | unsigned int block_size, io_manager manager, |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 72 | ext2_filsys *ret_fs) |
| 73 | { |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 74 | return ext2fs_open2(name, 0, flags, superblock, block_size, |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 75 | manager, ret_fs); |
| 76 | } |
| 77 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 78 | /* |
| 79 | * Note: if superblock is non-zero, block-size must also be non-zero. |
| 80 | * Superblock and block_size can be zero to use the default size. |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 81 | * |
| 82 | * Valid flags for ext2fs_open() |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 83 | * |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 84 | * EXT2_FLAG_RW - Open the filesystem for read/write. |
| 85 | * EXT2_FLAG_FORCE - Open the filesystem even if some of the |
Theodore Ts'o | a777397 | 2001-05-13 23:12:10 +0000 | [diff] [blame] | 86 | * features aren't supported. |
| 87 | * EXT2_FLAG_JOURNAL_DEV_OK - Open an ext3 journal device |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 88 | * EXT2_FLAG_SKIP_MMP - Open without multi-mount protection check. |
| 89 | * EXT2_FLAG_64BITS - Allow 64-bit bitfields (needed for large |
| 90 | * filesystems) |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 91 | */ |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 92 | errcode_t ext2fs_open2(const char *name, const char *io_options, |
| 93 | int flags, int superblock, |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 94 | unsigned int block_size, io_manager manager, |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 95 | ext2_filsys *ret_fs) |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 96 | { |
| 97 | ext2_filsys fs; |
| 98 | errcode_t retval; |
Theodore Ts'o | 8203fe5 | 2009-04-23 01:30:42 -0400 | [diff] [blame] | 99 | unsigned long i, first_meta_bg; |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 100 | __u32 features; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 101 | unsigned int blocks_per_group, io_flags; |
| 102 | blk64_t group_block, blk; |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 103 | char *dest, *cp; |
Theodore Ts'o | 2d328bb | 2008-03-17 23:17:13 -0400 | [diff] [blame] | 104 | #ifdef WORDS_BIGENDIAN |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 105 | unsigned int groups_per_block; |
Theodore Ts'o | 50e1e10 | 1997-04-26 13:58:21 +0000 | [diff] [blame] | 106 | struct ext2_group_desc *gdp; |
Theodore Ts'o | 2d328bb | 2008-03-17 23:17:13 -0400 | [diff] [blame] | 107 | int j; |
| 108 | #endif |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 109 | |
Theodore Ts'o | f3db356 | 1997-04-26 13:34:30 +0000 | [diff] [blame] | 110 | EXT2_CHECK_MAGIC(manager, EXT2_ET_MAGIC_IO_MANAGER); |
Theodore Ts'o | 7b4e453 | 1997-10-26 03:41:24 +0000 | [diff] [blame] | 111 | |
Theodore Ts'o | c4e3d3f | 2003-08-01 09:41:07 -0400 | [diff] [blame] | 112 | retval = ext2fs_get_mem(sizeof(struct struct_ext2_filsys), &fs); |
Theodore Ts'o | 7b4e453 | 1997-10-26 03:41:24 +0000 | [diff] [blame] | 113 | if (retval) |
| 114 | return retval; |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 115 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 116 | memset(fs, 0, sizeof(struct struct_ext2_filsys)); |
Theodore Ts'o | f3db356 | 1997-04-26 13:34:30 +0000 | [diff] [blame] | 117 | fs->magic = EXT2_ET_MAGIC_EXT2FS_FILSYS; |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 118 | fs->flags = flags; |
Theodore Ts'o | 058ad1c | 2007-06-18 18:26:50 -0400 | [diff] [blame] | 119 | /* don't overwrite sb backups unless flag is explicitly cleared */ |
| 120 | fs->flags |= EXT2_FLAG_MASTER_SB_ONLY; |
Theodore Ts'o | 6a52506 | 2001-12-24 09:40:00 -0500 | [diff] [blame] | 121 | fs->umask = 022; |
Theodore Ts'o | c4e3d3f | 2003-08-01 09:41:07 -0400 | [diff] [blame] | 122 | retval = ext2fs_get_mem(strlen(name)+1, &fs->device_name); |
Theodore Ts'o | 7b4e453 | 1997-10-26 03:41:24 +0000 | [diff] [blame] | 123 | if (retval) |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 124 | goto cleanup; |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 125 | strcpy(fs->device_name, name); |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 126 | cp = strchr(fs->device_name, '?'); |
| 127 | if (!io_options && cp) { |
| 128 | *cp++ = 0; |
| 129 | io_options = cp; |
| 130 | } |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 131 | |
Theodore Ts'o | 39c47ce | 2006-03-18 19:16:10 -0500 | [diff] [blame] | 132 | io_flags = 0; |
| 133 | if (flags & EXT2_FLAG_RW) |
| 134 | io_flags |= IO_FLAG_RW; |
| 135 | if (flags & EXT2_FLAG_EXCLUSIVE) |
| 136 | io_flags |= IO_FLAG_EXCLUSIVE; |
Theodore Ts'o | 7f1a1fb | 2010-09-24 10:02:25 -0400 | [diff] [blame] | 137 | if (flags & EXT2_FLAG_DIRECT_IO) |
| 138 | io_flags |= IO_FLAG_DIRECT_IO; |
Theodore Ts'o | 39c47ce | 2006-03-18 19:16:10 -0500 | [diff] [blame] | 139 | retval = manager->open(fs->device_name, io_flags, &fs->io); |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 140 | if (retval) |
| 141 | goto cleanup; |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 142 | if (io_options && |
Theodore Ts'o | 2e8ca9a | 2004-11-30 14:07:11 -0500 | [diff] [blame] | 143 | (retval = io_channel_set_options(fs->io, io_options))) |
| 144 | goto cleanup; |
| 145 | fs->image_io = fs->io; |
| 146 | fs->io->app_data = fs; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 147 | retval = io_channel_alloc_buf(fs->io, -SUPERBLOCK_SIZE, &fs->super); |
Theodore Ts'o | 7b4e453 | 1997-10-26 03:41:24 +0000 | [diff] [blame] | 148 | if (retval) |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 149 | goto cleanup; |
Theodore Ts'o | a78926e | 2001-05-03 04:02:29 +0000 | [diff] [blame] | 150 | if (flags & EXT2_FLAG_IMAGE_FILE) { |
| 151 | retval = ext2fs_get_mem(sizeof(struct ext2_image_hdr), |
Theodore Ts'o | c4e3d3f | 2003-08-01 09:41:07 -0400 | [diff] [blame] | 152 | &fs->image_header); |
Theodore Ts'o | a78926e | 2001-05-03 04:02:29 +0000 | [diff] [blame] | 153 | if (retval) |
| 154 | goto cleanup; |
| 155 | retval = io_channel_read_blk(fs->io, 0, |
Theodore Ts'o | 85f93ff | 2006-05-21 19:26:45 -0400 | [diff] [blame] | 156 | -(int)sizeof(struct ext2_image_hdr), |
Theodore Ts'o | a78926e | 2001-05-03 04:02:29 +0000 | [diff] [blame] | 157 | fs->image_header); |
| 158 | if (retval) |
| 159 | goto cleanup; |
| 160 | if (fs->image_header->magic_number != EXT2_ET_MAGIC_E2IMAGE) |
| 161 | return EXT2_ET_MAGIC_E2IMAGE; |
| 162 | superblock = 1; |
| 163 | block_size = fs->image_header->fs_blocksize; |
| 164 | } |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 165 | |
| 166 | /* |
| 167 | * If the user specifies a specific block # for the |
| 168 | * superblock, then he/she must also specify the block size! |
| 169 | * Otherwise, read the master superblock located at offset |
| 170 | * SUPERBLOCK_OFFSET from the start of the partition. |
Theodore Ts'o | c180ac8 | 2000-10-26 20:24:43 +0000 | [diff] [blame] | 171 | * |
| 172 | * Note: we only save a backup copy of the superblock if we |
| 173 | * are reading the superblock from the primary superblock location. |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 174 | */ |
| 175 | if (superblock) { |
| 176 | if (!block_size) { |
Theodore Ts'o | 1f0b6c1 | 1997-10-31 06:07:47 +0000 | [diff] [blame] | 177 | retval = EXT2_ET_INVALID_ARGUMENT; |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 178 | goto cleanup; |
| 179 | } |
| 180 | io_channel_set_blksize(fs->io, block_size); |
Theodore Ts'o | 9ed06a1 | 2002-10-31 11:45:06 -0500 | [diff] [blame] | 181 | group_block = superblock; |
Theodore Ts'o | c180ac8 | 2000-10-26 20:24:43 +0000 | [diff] [blame] | 182 | fs->orig_super = 0; |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 183 | } else { |
| 184 | io_channel_set_blksize(fs->io, SUPERBLOCK_OFFSET); |
| 185 | superblock = 1; |
Theodore Ts'o | f3db356 | 1997-04-26 13:34:30 +0000 | [diff] [blame] | 186 | group_block = 0; |
Theodore Ts'o | c4e3d3f | 2003-08-01 09:41:07 -0400 | [diff] [blame] | 187 | retval = ext2fs_get_mem(SUPERBLOCK_SIZE, &fs->orig_super); |
Theodore Ts'o | c180ac8 | 2000-10-26 20:24:43 +0000 | [diff] [blame] | 188 | if (retval) |
| 189 | goto cleanup; |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 190 | } |
| 191 | retval = io_channel_read_blk(fs->io, superblock, -SUPERBLOCK_SIZE, |
| 192 | fs->super); |
| 193 | if (retval) |
| 194 | goto cleanup; |
Theodore Ts'o | c180ac8 | 2000-10-26 20:24:43 +0000 | [diff] [blame] | 195 | if (fs->orig_super) |
| 196 | memcpy(fs->orig_super, fs->super, SUPERBLOCK_SIZE); |
Theodore Ts'o | 50e1e10 | 1997-04-26 13:58:21 +0000 | [diff] [blame] | 197 | |
Theodore Ts'o | 126a291 | 2007-08-11 01:56:48 -0400 | [diff] [blame] | 198 | #ifdef WORDS_BIGENDIAN |
| 199 | fs->flags |= EXT2_FLAG_SWAP_BYTES; |
| 200 | ext2fs_swap_super(fs->super); |
| 201 | #else |
| 202 | if (fs->flags & EXT2_FLAG_SWAP_BYTES) { |
| 203 | retval = EXT2_ET_UNIMPLEMENTED; |
| 204 | goto cleanup; |
Theodore Ts'o | 50e1e10 | 1997-04-26 13:58:21 +0000 | [diff] [blame] | 205 | } |
Theodore Ts'o | 5df55d7 | 2001-06-11 07:00:04 +0000 | [diff] [blame] | 206 | #endif |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 207 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 208 | if (fs->super->s_magic != EXT2_SUPER_MAGIC) { |
| 209 | retval = EXT2_ET_BAD_MAGIC; |
| 210 | goto cleanup; |
| 211 | } |
Theodore Ts'o | f3db356 | 1997-04-26 13:34:30 +0000 | [diff] [blame] | 212 | if (fs->super->s_rev_level > EXT2_LIB_CURRENT_REV) { |
| 213 | retval = EXT2_ET_REV_TOO_HIGH; |
| 214 | goto cleanup; |
| 215 | } |
Theodore Ts'o | e5b38a5 | 2001-01-01 16:17:12 +0000 | [diff] [blame] | 216 | |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 217 | /* |
| 218 | * Check for feature set incompatibility |
| 219 | */ |
| 220 | if (!(flags & EXT2_FLAG_FORCE)) { |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 221 | features = fs->super->s_feature_incompat; |
| 222 | #ifdef EXT2_LIB_SOFTSUPP_INCOMPAT |
| 223 | if (flags & EXT2_FLAG_SOFTSUPP_FEATURES) |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 224 | features &= ~EXT2_LIB_SOFTSUPP_INCOMPAT; |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 225 | #endif |
| 226 | if (features & ~EXT2_LIB_FEATURE_INCOMPAT_SUPP) { |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 227 | retval = EXT2_ET_UNSUPP_FEATURE; |
| 228 | goto cleanup; |
| 229 | } |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 230 | |
| 231 | features = fs->super->s_feature_ro_compat; |
| 232 | #ifdef EXT2_LIB_SOFTSUPP_RO_COMPAT |
| 233 | if (flags & EXT2_FLAG_SOFTSUPP_FEATURES) |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 234 | features &= ~EXT2_LIB_SOFTSUPP_RO_COMPAT; |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 235 | #endif |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 236 | if ((flags & EXT2_FLAG_RW) && |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 237 | (features & ~EXT2_LIB_FEATURE_RO_COMPAT_SUPP)) { |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 238 | retval = EXT2_ET_RO_UNSUPP_FEATURE; |
| 239 | goto cleanup; |
| 240 | } |
Theodore Ts'o | cf8272e | 2006-11-12 23:26:46 -0500 | [diff] [blame] | 241 | |
Theodore Ts'o | a112847 | 2001-01-16 06:56:14 +0000 | [diff] [blame] | 242 | if (!(flags & EXT2_FLAG_JOURNAL_DEV_OK) && |
| 243 | (fs->super->s_feature_incompat & |
| 244 | EXT3_FEATURE_INCOMPAT_JOURNAL_DEV)) { |
| 245 | retval = EXT2_ET_UNSUPP_FEATURE; |
| 246 | goto cleanup; |
| 247 | } |
Theodore Ts'o | 19c78dc | 1997-04-29 16:17:09 +0000 | [diff] [blame] | 248 | } |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 249 | |
Manish Katiyar | 49b4670 | 2008-07-11 17:45:07 -0400 | [diff] [blame] | 250 | if ((fs->super->s_log_block_size + EXT2_MIN_BLOCK_LOG_SIZE) > |
| 251 | EXT2_MAX_BLOCK_LOG_SIZE) { |
Theodore Ts'o | 1e3472c | 1997-04-29 14:53:37 +0000 | [diff] [blame] | 252 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 253 | goto cleanup; |
| 254 | } |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 255 | |
| 256 | /* |
| 257 | * bigalloc requires cluster-aware bitfield operations, which at the |
| 258 | * moment means we need EXT2_FLAG_64BITS. |
| 259 | */ |
| 260 | if (EXT2_HAS_RO_COMPAT_FEATURE(fs->super, |
| 261 | EXT4_FEATURE_RO_COMPAT_BIGALLOC) && |
| 262 | !(flags & EXT2_FLAG_64BITS)) { |
| 263 | retval = EXT2_ET_CANT_USE_LEGACY_BITMAPS; |
| 264 | goto cleanup; |
| 265 | } |
| 266 | |
| 267 | if (!EXT2_HAS_RO_COMPAT_FEATURE(fs->super, |
| 268 | EXT4_FEATURE_RO_COMPAT_BIGALLOC) && |
| 269 | (fs->super->s_log_block_size != fs->super->s_log_cluster_size)) { |
| 270 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 271 | goto cleanup; |
| 272 | } |
| 273 | fs->fragsize = fs->blocksize = EXT2_BLOCK_SIZE(fs->super); |
Theodore Ts'o | ba9d929 | 2007-09-07 16:40:25 -0400 | [diff] [blame] | 274 | if (EXT2_INODE_SIZE(fs->super) < EXT2_GOOD_OLD_INODE_SIZE) { |
| 275 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 276 | goto cleanup; |
| 277 | } |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 278 | fs->cluster_ratio_bits = fs->super->s_log_cluster_size - |
| 279 | fs->super->s_log_block_size; |
| 280 | if (EXT2_BLOCKS_PER_GROUP(fs->super) != |
| 281 | EXT2_CLUSTERS_PER_GROUP(fs->super) << fs->cluster_ratio_bits) { |
| 282 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 283 | goto cleanup; |
| 284 | } |
Thiemo Nagel | 79a9ab1 | 2009-01-19 23:16:10 -0500 | [diff] [blame] | 285 | fs->inode_blocks_per_group = ((EXT2_INODES_PER_GROUP(fs->super) * |
Theodore Ts'o | 7f88b04 | 1997-04-26 14:48:50 +0000 | [diff] [blame] | 286 | EXT2_INODE_SIZE(fs->super) + |
| 287 | EXT2_BLOCK_SIZE(fs->super) - 1) / |
| 288 | EXT2_BLOCK_SIZE(fs->super)); |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 289 | if (block_size) { |
| 290 | if (block_size != fs->blocksize) { |
| 291 | retval = EXT2_ET_UNEXPECTED_BLOCK_SIZE; |
| 292 | goto cleanup; |
| 293 | } |
| 294 | } |
| 295 | /* |
| 296 | * Set the blocksize to the filesystem's blocksize. |
| 297 | */ |
| 298 | io_channel_set_blksize(fs->io, fs->blocksize); |
Theodore Ts'o | a112847 | 2001-01-16 06:56:14 +0000 | [diff] [blame] | 299 | |
| 300 | /* |
| 301 | * If this is an external journal device, don't try to read |
| 302 | * the group descriptors, because they're not there. |
| 303 | */ |
| 304 | if (fs->super->s_feature_incompat & |
| 305 | EXT3_FEATURE_INCOMPAT_JOURNAL_DEV) { |
| 306 | fs->group_desc_count = 0; |
| 307 | *ret_fs = fs; |
| 308 | return 0; |
| 309 | } |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 310 | |
Theodore Ts'o | 341b52d | 2009-03-16 22:16:44 -0400 | [diff] [blame] | 311 | if (EXT2_INODES_PER_GROUP(fs->super) == 0) { |
| 312 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 313 | goto cleanup; |
| 314 | } |
| 315 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 316 | /* |
| 317 | * Read group descriptors |
| 318 | */ |
Andreas Dilger | b21bf26 | 2002-06-10 11:05:56 -0600 | [diff] [blame] | 319 | blocks_per_group = EXT2_BLOCKS_PER_GROUP(fs->super); |
| 320 | if (blocks_per_group == 0 || |
| 321 | blocks_per_group > EXT2_MAX_BLOCKS_PER_GROUP(fs->super) || |
Thiemo Nagel | 79a9ab1 | 2009-01-19 23:16:10 -0500 | [diff] [blame] | 322 | fs->inode_blocks_per_group > EXT2_MAX_INODES_PER_GROUP(fs->super) || |
| 323 | EXT2_DESC_PER_BLOCK(fs->super) == 0 || |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 324 | fs->super->s_first_data_block >= ext2fs_blocks_count(fs->super)) { |
Theodore Ts'o | f0687a5 | 1999-05-29 21:48:03 +0000 | [diff] [blame] | 325 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
| 326 | goto cleanup; |
| 327 | } |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 328 | fs->group_desc_count = ext2fs_div64_ceil(ext2fs_blocks_count(fs->super) - |
| 329 | fs->super->s_first_data_block, |
| 330 | blocks_per_group); |
| 331 | if (fs->group_desc_count * EXT2_INODES_PER_GROUP(fs->super) != |
| 332 | fs->super->s_inodes_count) { |
| 333 | retval = EXT2_ET_CORRUPT_SUPERBLOCK; |
Thiemo Nagel | 79a9ab1 | 2009-01-19 23:16:10 -0500 | [diff] [blame] | 334 | goto cleanup; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 335 | } |
Theodore Ts'o | 69022e0 | 2006-08-30 01:57:00 -0400 | [diff] [blame] | 336 | fs->desc_blocks = ext2fs_div_ceil(fs->group_desc_count, |
| 337 | EXT2_DESC_PER_BLOCK(fs->super)); |
Theodore Ts'o | ee01079 | 2007-11-09 19:01:06 -0500 | [diff] [blame] | 338 | retval = ext2fs_get_array(fs->desc_blocks, fs->blocksize, |
Theodore Ts'o | c4e3d3f | 2003-08-01 09:41:07 -0400 | [diff] [blame] | 339 | &fs->group_desc); |
Theodore Ts'o | 7b4e453 | 1997-10-26 03:41:24 +0000 | [diff] [blame] | 340 | if (retval) |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 341 | goto cleanup; |
Theodore Ts'o | f3db356 | 1997-04-26 13:34:30 +0000 | [diff] [blame] | 342 | if (!group_block) |
Theodore Ts'o | c046ac7 | 2002-10-20 00:38:57 -0400 | [diff] [blame] | 343 | group_block = fs->super->s_first_data_block; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 344 | if (group_block == 0 && fs->blocksize == 1024) |
| 345 | group_block = 1; /* Deal with 1024 blocksize && bigalloc */ |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 346 | dest = (char *) fs->group_desc; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 347 | #ifdef WORDS_BIGENDIAN |
Valerie Clement | f2de1d3 | 2007-08-30 17:38:13 +0200 | [diff] [blame] | 348 | groups_per_block = EXT2_DESC_PER_BLOCK(fs->super); |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 349 | #endif |
Theodore Ts'o | 8203fe5 | 2009-04-23 01:30:42 -0400 | [diff] [blame] | 350 | if (fs->super->s_feature_incompat & EXT2_FEATURE_INCOMPAT_META_BG) |
| 351 | first_meta_bg = fs->super->s_first_meta_bg; |
| 352 | else |
| 353 | first_meta_bg = fs->desc_blocks; |
| 354 | if (first_meta_bg) { |
| 355 | retval = io_channel_read_blk(fs->io, group_block+1, |
| 356 | first_meta_bg, dest); |
| 357 | if (retval) |
| 358 | goto cleanup; |
| 359 | #ifdef WORDS_BIGENDIAN |
| 360 | gdp = (struct ext2_group_desc *) dest; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 361 | for (j=0; j < groups_per_block*first_meta_bg; j++) { |
| 362 | gdp = ext2fs_group_desc(fs, fs->group_desc, j); |
| 363 | ext2fs_swap_group_desc2(fs, gdp); |
| 364 | } |
Theodore Ts'o | 8203fe5 | 2009-04-23 01:30:42 -0400 | [diff] [blame] | 365 | #endif |
| 366 | dest += fs->blocksize*first_meta_bg; |
| 367 | } |
| 368 | for (i=first_meta_bg ; i < fs->desc_blocks; i++) { |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 369 | blk = ext2fs_descriptor_block_loc2(fs, group_block, i); |
| 370 | retval = io_channel_read_blk64(fs->io, blk, 1, dest); |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 371 | if (retval) |
| 372 | goto cleanup; |
Theodore Ts'o | 126a291 | 2007-08-11 01:56:48 -0400 | [diff] [blame] | 373 | #ifdef WORDS_BIGENDIAN |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 374 | for (j=0; j < groups_per_block; j++) { |
| 375 | gdp = ext2fs_group_desc(fs, fs->group_desc, |
| 376 | i * groups_per_block + j); |
| 377 | ext2fs_swap_group_desc2(fs, gdp); |
| 378 | } |
Theodore Ts'o | 5df55d7 | 2001-06-11 07:00:04 +0000 | [diff] [blame] | 379 | #endif |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 380 | dest += fs->blocksize; |
| 381 | } |
| 382 | |
Theodore Ts'o | 96c6a3a | 2007-05-18 22:06:53 -0400 | [diff] [blame] | 383 | fs->stride = fs->super->s_raid_stride; |
| 384 | |
Jose R. Santos | d4f34d4 | 2007-10-21 21:03:25 -0500 | [diff] [blame] | 385 | /* |
| 386 | * If recovery is from backup superblock, Clear _UNININT flags & |
| 387 | * reset bg_itable_unused to zero |
| 388 | */ |
| 389 | if (superblock > 1 && EXT2_HAS_RO_COMPAT_FEATURE(fs->super, |
| 390 | EXT4_FEATURE_RO_COMPAT_GDT_CSUM)) { |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 391 | dgrp_t group; |
| 392 | |
| 393 | for (group = 0; group < fs->group_desc_count; group++) { |
| 394 | ext2fs_bg_flags_clear(fs, group, EXT2_BG_BLOCK_UNINIT); |
| 395 | ext2fs_bg_flags_clear(fs, group, EXT2_BG_INODE_UNINIT); |
| 396 | ext2fs_bg_itable_unused_set(fs, group, 0); |
| 397 | /* The checksum will be reset later, but fix it here |
| 398 | * anyway to avoid printing a lot of spurious errors. */ |
| 399 | ext2fs_group_desc_csum_set(fs, group); |
Jose R. Santos | d4f34d4 | 2007-10-21 21:03:25 -0500 | [diff] [blame] | 400 | } |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 401 | if (fs->flags & EXT2_FLAG_RW) |
| 402 | ext2fs_mark_super_dirty(fs); |
| 403 | } |
| 404 | |
| 405 | if ((fs->super->s_feature_incompat & EXT4_FEATURE_INCOMPAT_MMP) && |
| 406 | !(flags & EXT2_FLAG_SKIP_MMP) && |
| 407 | (flags & (EXT2_FLAG_RW | EXT2_FLAG_EXCLUSIVE))) { |
| 408 | retval = ext2fs_mmp_start(fs); |
| 409 | if (retval) { |
| 410 | fs->flags |= EXT2_FLAG_SKIP_MMP; /* just do cleanup */ |
| 411 | ext2fs_mmp_stop(fs); |
| 412 | goto cleanup; |
| 413 | } |
Andreas Dilger | 0f5eba7 | 2011-09-24 13:48:55 -0400 | [diff] [blame] | 414 | } |
| 415 | |
Theodore Ts'o | f0257d8 | 2013-01-08 20:47:11 -0500 | [diff] [blame] | 416 | fs->flags &= ~EXT2_FLAG_NOFREE_ON_ERROR; |
| 417 | *ret_fs = fs; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 418 | |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 419 | return 0; |
| 420 | cleanup: |
Theodore Ts'o | ab52e12 | 2008-02-26 20:45:36 -0500 | [diff] [blame] | 421 | if (flags & EXT2_FLAG_NOFREE_ON_ERROR) |
| 422 | *ret_fs = fs; |
| 423 | else |
| 424 | ext2fs_free(fs); |
Theodore Ts'o | 3839e65 | 1997-04-26 13:21:57 +0000 | [diff] [blame] | 425 | return retval; |
| 426 | } |
| 427 | |
Theodore Ts'o | 1ad54a9 | 2004-07-28 21:11:48 -0400 | [diff] [blame] | 428 | /* |
| 429 | * Set/get the filesystem data I/O channel. |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 430 | * |
Theodore Ts'o | 1ad54a9 | 2004-07-28 21:11:48 -0400 | [diff] [blame] | 431 | * These functions are only valid if EXT2_FLAG_IMAGE_FILE is true. |
| 432 | */ |
| 433 | errcode_t ext2fs_get_data_io(ext2_filsys fs, io_channel *old_io) |
| 434 | { |
| 435 | if ((fs->flags & EXT2_FLAG_IMAGE_FILE) == 0) |
| 436 | return EXT2_ET_NOT_IMAGE_FILE; |
| 437 | if (old_io) { |
| 438 | *old_io = (fs->image_io == fs->io) ? 0 : fs->io; |
| 439 | } |
| 440 | return 0; |
| 441 | } |
| 442 | |
| 443 | errcode_t ext2fs_set_data_io(ext2_filsys fs, io_channel new_io) |
| 444 | { |
| 445 | if ((fs->flags & EXT2_FLAG_IMAGE_FILE) == 0) |
| 446 | return EXT2_ET_NOT_IMAGE_FILE; |
| 447 | fs->io = new_io ? new_io : fs->image_io; |
| 448 | return 0; |
| 449 | } |
| 450 | |
| 451 | errcode_t ext2fs_rewrite_to_io(ext2_filsys fs, io_channel new_io) |
| 452 | { |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 453 | errcode_t err; |
| 454 | |
Theodore Ts'o | 1ad54a9 | 2004-07-28 21:11:48 -0400 | [diff] [blame] | 455 | if ((fs->flags & EXT2_FLAG_IMAGE_FILE) == 0) |
| 456 | return EXT2_ET_NOT_IMAGE_FILE; |
JP Abgrall | e0ed740 | 2014-03-19 19:08:39 -0700 | [diff] [blame] | 457 | err = io_channel_set_blksize(new_io, fs->blocksize); |
| 458 | if (err) |
| 459 | return err; |
| 460 | if ((new_io == fs->image_io) || (new_io == fs->io)) |
| 461 | return 0; |
| 462 | if ((fs->image_io != fs->io) && |
| 463 | fs->image_io) |
| 464 | io_channel_close(fs->image_io); |
| 465 | if (fs->io) |
| 466 | io_channel_close(fs->io); |
Theodore Ts'o | 1ad54a9 | 2004-07-28 21:11:48 -0400 | [diff] [blame] | 467 | fs->io = fs->image_io = new_io; |
Theodore Ts'o | efc6f62 | 2008-08-27 23:07:54 -0400 | [diff] [blame] | 468 | fs->flags |= EXT2_FLAG_DIRTY | EXT2_FLAG_RW | |
Theodore Ts'o | 1ad54a9 | 2004-07-28 21:11:48 -0400 | [diff] [blame] | 469 | EXT2_FLAG_BB_DIRTY | EXT2_FLAG_IB_DIRTY; |
| 470 | fs->flags &= ~EXT2_FLAG_IMAGE_FILE; |
| 471 | return 0; |
| 472 | } |