diff options
| -rw-r--r-- | fs/hfsplus/hfsplus_fs.h | 8 | ||||
| -rw-r--r-- | fs/hfsplus/super.c | 48 | ||||
| -rw-r--r-- | fs/hfsplus/wrapper.c | 163 |
3 files changed, 131 insertions, 88 deletions
diff --git a/fs/hfsplus/hfsplus_fs.h b/fs/hfsplus/hfsplus_fs.h index f07aa640c27d..276ddb0fd0fd 100644 --- a/fs/hfsplus/hfsplus_fs.h +++ b/fs/hfsplus/hfsplus_fs.h | |||
| @@ -107,8 +107,8 @@ struct hfsplus_vh; | |||
| 107 | struct hfs_btree; | 107 | struct hfs_btree; |
| 108 | 108 | ||
| 109 | struct hfsplus_sb_info { | 109 | struct hfsplus_sb_info { |
| 110 | struct buffer_head *s_vhbh; | ||
| 111 | struct hfsplus_vh *s_vhdr; | 110 | struct hfsplus_vh *s_vhdr; |
| 111 | struct hfsplus_vh *s_backup_vhdr; | ||
| 112 | struct hfs_btree *ext_tree; | 112 | struct hfs_btree *ext_tree; |
| 113 | struct hfs_btree *cat_tree; | 113 | struct hfs_btree *cat_tree; |
| 114 | struct hfs_btree *attr_tree; | 114 | struct hfs_btree *attr_tree; |
| @@ -118,7 +118,8 @@ struct hfsplus_sb_info { | |||
| 118 | 118 | ||
| 119 | /* Runtime variables */ | 119 | /* Runtime variables */ |
| 120 | u32 blockoffset; | 120 | u32 blockoffset; |
| 121 | u32 sect_count; | 121 | sector_t part_start; |
| 122 | sector_t sect_count; | ||
| 122 | int fs_shift; | 123 | int fs_shift; |
| 123 | 124 | ||
| 124 | /* immutable data from the volume header */ | 125 | /* immutable data from the volume header */ |
| @@ -385,8 +386,9 @@ int hfsplus_compare_dentry(struct dentry *dentry, struct qstr *s1, struct qstr * | |||
| 385 | 386 | ||
| 386 | /* wrapper.c */ | 387 | /* wrapper.c */ |
| 387 | int hfsplus_read_wrapper(struct super_block *); | 388 | int hfsplus_read_wrapper(struct super_block *); |
| 388 | |||
| 389 | int hfs_part_find(struct super_block *, sector_t *, sector_t *); | 389 | int hfs_part_find(struct super_block *, sector_t *, sector_t *); |
| 390 | int hfsplus_submit_bio(struct block_device *bdev, sector_t sector, | ||
| 391 | void *data, int rw); | ||
| 390 | 392 | ||
| 391 | /* access macros */ | 393 | /* access macros */ |
| 392 | static inline struct hfsplus_sb_info *HFSPLUS_SB(struct super_block *sb) | 394 | static inline struct hfsplus_sb_info *HFSPLUS_SB(struct super_block *sb) |
diff --git a/fs/hfsplus/super.c b/fs/hfsplus/super.c index 6a2349058618..fe8f7bffbea5 100644 --- a/fs/hfsplus/super.c +++ b/fs/hfsplus/super.c | |||
| @@ -157,45 +157,40 @@ int hfsplus_sync_fs(struct super_block *sb, int wait) | |||
| 157 | { | 157 | { |
| 158 | struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb); | 158 | struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb); |
| 159 | struct hfsplus_vh *vhdr = sbi->s_vhdr; | 159 | struct hfsplus_vh *vhdr = sbi->s_vhdr; |
| 160 | int write_backup = 0; | ||
| 161 | int error, error2; | ||
| 160 | 162 | ||
| 161 | dprint(DBG_SUPER, "hfsplus_write_super\n"); | 163 | dprint(DBG_SUPER, "hfsplus_write_super\n"); |
| 162 | 164 | ||
| 163 | mutex_lock(&sbi->vh_mutex); | ||
| 164 | mutex_lock(&sbi->alloc_mutex); | ||
| 165 | sb->s_dirt = 0; | 165 | sb->s_dirt = 0; |
| 166 | 166 | ||
| 167 | mutex_lock(&sbi->vh_mutex); | ||
| 168 | mutex_lock(&sbi->alloc_mutex); | ||
| 167 | vhdr->free_blocks = cpu_to_be32(sbi->free_blocks); | 169 | vhdr->free_blocks = cpu_to_be32(sbi->free_blocks); |
| 168 | vhdr->next_cnid = cpu_to_be32(sbi->next_cnid); | 170 | vhdr->next_cnid = cpu_to_be32(sbi->next_cnid); |
| 169 | vhdr->folder_count = cpu_to_be32(sbi->folder_count); | 171 | vhdr->folder_count = cpu_to_be32(sbi->folder_count); |
| 170 | vhdr->file_count = cpu_to_be32(sbi->file_count); | 172 | vhdr->file_count = cpu_to_be32(sbi->file_count); |
| 171 | 173 | ||
| 172 | mark_buffer_dirty(sbi->s_vhbh); | ||
| 173 | if (test_and_clear_bit(HFSPLUS_SB_WRITEBACKUP, &sbi->flags)) { | 174 | if (test_and_clear_bit(HFSPLUS_SB_WRITEBACKUP, &sbi->flags)) { |
| 174 | if (sbi->sect_count) { | 175 | memcpy(sbi->s_backup_vhdr, sbi->s_vhdr, sizeof(*sbi->s_vhdr)); |
| 175 | struct buffer_head *bh; | 176 | write_backup = 1; |
| 176 | u32 block, offset; | ||
| 177 | |||
| 178 | block = sbi->blockoffset; | ||
| 179 | block += (sbi->sect_count - 2) >> (sb->s_blocksize_bits - 9); | ||
| 180 | offset = ((sbi->sect_count - 2) << 9) & (sb->s_blocksize - 1); | ||
| 181 | printk(KERN_DEBUG "hfs: backup: %u,%u,%u,%u\n", | ||
| 182 | sbi->blockoffset, sbi->sect_count, | ||
| 183 | block, offset); | ||
| 184 | bh = sb_bread(sb, block); | ||
| 185 | if (bh) { | ||
| 186 | vhdr = (struct hfsplus_vh *)(bh->b_data + offset); | ||
| 187 | if (be16_to_cpu(vhdr->signature) == HFSPLUS_VOLHEAD_SIG) { | ||
| 188 | memcpy(vhdr, sbi->s_vhdr, sizeof(*vhdr)); | ||
| 189 | mark_buffer_dirty(bh); | ||
| 190 | brelse(bh); | ||
| 191 | } else | ||
| 192 | printk(KERN_WARNING "hfs: backup not found!\n"); | ||
| 193 | } | ||
| 194 | } | ||
| 195 | } | 177 | } |
| 178 | |||
| 179 | error = hfsplus_submit_bio(sb->s_bdev, | ||
| 180 | sbi->part_start + HFSPLUS_VOLHEAD_SECTOR, | ||
| 181 | sbi->s_vhdr, WRITE_SYNC); | ||
| 182 | if (!write_backup) | ||
| 183 | goto out; | ||
| 184 | |||
| 185 | error2 = hfsplus_submit_bio(sb->s_bdev, | ||
| 186 | sbi->part_start + sbi->sect_count - 2, | ||
| 187 | sbi->s_backup_vhdr, WRITE_SYNC); | ||
| 188 | if (!error) | ||
| 189 | error2 = error; | ||
| 190 | out: | ||
| 196 | mutex_unlock(&sbi->alloc_mutex); | 191 | mutex_unlock(&sbi->alloc_mutex); |
| 197 | mutex_unlock(&sbi->vh_mutex); | 192 | mutex_unlock(&sbi->vh_mutex); |
| 198 | return 0; | 193 | return error; |
| 199 | } | 194 | } |
| 200 | 195 | ||
| 201 | static void hfsplus_write_super(struct super_block *sb) | 196 | static void hfsplus_write_super(struct super_block *sb) |
| @@ -229,7 +224,8 @@ static void hfsplus_put_super(struct super_block *sb) | |||
| 229 | hfs_btree_close(sbi->ext_tree); | 224 | hfs_btree_close(sbi->ext_tree); |
| 230 | iput(sbi->alloc_file); | 225 | iput(sbi->alloc_file); |
| 231 | iput(sbi->hidden_dir); | 226 | iput(sbi->hidden_dir); |
| 232 | brelse(sbi->s_vhbh); | 227 | kfree(sbi->s_vhdr); |
| 228 | kfree(sbi->s_backup_vhdr); | ||
| 233 | unload_nls(sbi->nls); | 229 | unload_nls(sbi->nls); |
| 234 | kfree(sb->s_fs_info); | 230 | kfree(sb->s_fs_info); |
| 235 | sb->s_fs_info = NULL; | 231 | sb->s_fs_info = NULL; |
diff --git a/fs/hfsplus/wrapper.c b/fs/hfsplus/wrapper.c index 8972c20b3216..15e0eabb489e 100644 --- a/fs/hfsplus/wrapper.c +++ b/fs/hfsplus/wrapper.c | |||
| @@ -24,6 +24,40 @@ struct hfsplus_wd { | |||
| 24 | u16 embed_count; | 24 | u16 embed_count; |
| 25 | }; | 25 | }; |
| 26 | 26 | ||
| 27 | static void hfsplus_end_io_sync(struct bio *bio, int err) | ||
| 28 | { | ||
| 29 | if (err) | ||
| 30 | clear_bit(BIO_UPTODATE, &bio->bi_flags); | ||
| 31 | complete(bio->bi_private); | ||
| 32 | } | ||
| 33 | |||
| 34 | int hfsplus_submit_bio(struct block_device *bdev, sector_t sector, | ||
| 35 | void *data, int rw) | ||
| 36 | { | ||
| 37 | DECLARE_COMPLETION_ONSTACK(wait); | ||
| 38 | struct bio *bio; | ||
| 39 | |||
| 40 | bio = bio_alloc(GFP_NOIO, 1); | ||
| 41 | bio->bi_sector = sector; | ||
| 42 | bio->bi_bdev = bdev; | ||
| 43 | bio->bi_end_io = hfsplus_end_io_sync; | ||
| 44 | bio->bi_private = &wait; | ||
| 45 | |||
| 46 | /* | ||
| 47 | * We always submit one sector at a time, so bio_add_page must not fail. | ||
| 48 | */ | ||
| 49 | if (bio_add_page(bio, virt_to_page(data), HFSPLUS_SECTOR_SIZE, | ||
| 50 | offset_in_page(data)) != HFSPLUS_SECTOR_SIZE) | ||
| 51 | BUG(); | ||
| 52 | |||
| 53 | submit_bio(rw, bio); | ||
| 54 | wait_for_completion(&wait); | ||
| 55 | |||
| 56 | if (!bio_flagged(bio, BIO_UPTODATE)) | ||
| 57 | return -EIO; | ||
| 58 | return 0; | ||
| 59 | } | ||
| 60 | |||
| 27 | static int hfsplus_read_mdb(void *bufptr, struct hfsplus_wd *wd) | 61 | static int hfsplus_read_mdb(void *bufptr, struct hfsplus_wd *wd) |
| 28 | { | 62 | { |
| 29 | u32 extent; | 63 | u32 extent; |
| @@ -88,100 +122,111 @@ static int hfsplus_get_last_session(struct super_block *sb, | |||
| 88 | int hfsplus_read_wrapper(struct super_block *sb) | 122 | int hfsplus_read_wrapper(struct super_block *sb) |
| 89 | { | 123 | { |
| 90 | struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb); | 124 | struct hfsplus_sb_info *sbi = HFSPLUS_SB(sb); |
| 91 | struct buffer_head *bh; | ||
| 92 | struct hfsplus_vh *vhdr; | ||
| 93 | struct hfsplus_wd wd; | 125 | struct hfsplus_wd wd; |
| 94 | sector_t part_start, part_size; | 126 | sector_t part_start, part_size; |
| 95 | u32 blocksize; | 127 | u32 blocksize; |
| 128 | int error = 0; | ||
| 96 | 129 | ||
| 130 | error = -EINVAL; | ||
| 97 | blocksize = sb_min_blocksize(sb, HFSPLUS_SECTOR_SIZE); | 131 | blocksize = sb_min_blocksize(sb, HFSPLUS_SECTOR_SIZE); |
| 98 | if (!blocksize) | 132 | if (!blocksize) |
| 99 | return -EINVAL; | 133 | goto out; |
| 100 | 134 | ||
| 101 | if (hfsplus_get_last_session(sb, &part_start, &part_size)) | 135 | if (hfsplus_get_last_session(sb, &part_start, &part_size)) |
| 102 | return -EINVAL; | 136 | goto out; |
| 103 | if ((u64)part_start + part_size > 0x100000000ULL) { | 137 | if ((u64)part_start + part_size > 0x100000000ULL) { |
| 104 | pr_err("hfs: volumes larger than 2TB are not supported yet\n"); | 138 | pr_err("hfs: volumes larger than 2TB are not supported yet\n"); |
| 105 | return -EINVAL; | 139 | goto out; |
| 106 | } | 140 | } |
| 107 | while (1) { | ||
| 108 | bh = sb_bread512(sb, part_start + HFSPLUS_VOLHEAD_SECTOR, vhdr); | ||
| 109 | if (!bh) | ||
| 110 | return -EIO; | ||
| 111 | |||
| 112 | if (vhdr->signature == cpu_to_be16(HFSP_WRAP_MAGIC)) { | ||
| 113 | if (!hfsplus_read_mdb(vhdr, &wd)) | ||
| 114 | goto error; | ||
| 115 | wd.ablk_size >>= HFSPLUS_SECTOR_SHIFT; | ||
| 116 | part_start += wd.ablk_start + wd.embed_start * wd.ablk_size; | ||
| 117 | part_size = wd.embed_count * wd.ablk_size; | ||
| 118 | brelse(bh); | ||
| 119 | bh = sb_bread512(sb, part_start + HFSPLUS_VOLHEAD_SECTOR, vhdr); | ||
| 120 | if (!bh) | ||
| 121 | return -EIO; | ||
| 122 | } | ||
| 123 | if (vhdr->signature == cpu_to_be16(HFSPLUS_VOLHEAD_SIG)) | ||
| 124 | break; | ||
| 125 | if (vhdr->signature == cpu_to_be16(HFSPLUS_VOLHEAD_SIGX)) { | ||
| 126 | set_bit(HFSPLUS_SB_HFSX, &sbi->flags); | ||
| 127 | break; | ||
| 128 | } | ||
| 129 | brelse(bh); | ||
| 130 | 141 | ||
| 131 | /* check for a partition block | 142 | error = -ENOMEM; |
| 143 | sbi->s_vhdr = kmalloc(HFSPLUS_SECTOR_SIZE, GFP_KERNEL); | ||
| 144 | if (!sbi->s_vhdr) | ||
| 145 | goto out; | ||
| 146 | sbi->s_backup_vhdr = kmalloc(HFSPLUS_SECTOR_SIZE, GFP_KERNEL); | ||
| 147 | if (!sbi->s_backup_vhdr) | ||
| 148 | goto out_free_vhdr; | ||
| 149 | |||
| 150 | reread: | ||
| 151 | error = hfsplus_submit_bio(sb->s_bdev, | ||
| 152 | part_start + HFSPLUS_VOLHEAD_SECTOR, | ||
| 153 | sbi->s_vhdr, READ); | ||
| 154 | if (error) | ||
| 155 | goto out_free_backup_vhdr; | ||
| 156 | |||
| 157 | error = -EINVAL; | ||
| 158 | switch (sbi->s_vhdr->signature) { | ||
| 159 | case cpu_to_be16(HFSPLUS_VOLHEAD_SIGX): | ||
| 160 | set_bit(HFSPLUS_SB_HFSX, &sbi->flags); | ||
| 161 | /*FALLTHRU*/ | ||
| 162 | case cpu_to_be16(HFSPLUS_VOLHEAD_SIG): | ||
| 163 | break; | ||
| 164 | case cpu_to_be16(HFSP_WRAP_MAGIC): | ||
| 165 | if (!hfsplus_read_mdb(sbi->s_vhdr, &wd)) | ||
| 166 | goto out; | ||
| 167 | wd.ablk_size >>= HFSPLUS_SECTOR_SHIFT; | ||
| 168 | part_start += wd.ablk_start + wd.embed_start * wd.ablk_size; | ||
| 169 | part_size = wd.embed_count * wd.ablk_size; | ||
| 170 | goto reread; | ||
| 171 | default: | ||
| 172 | /* | ||
| 173 | * Check for a partition block. | ||
| 174 | * | ||
| 132 | * (should do this only for cdrom/loop though) | 175 | * (should do this only for cdrom/loop though) |
| 133 | */ | 176 | */ |
| 134 | if (hfs_part_find(sb, &part_start, &part_size)) | 177 | if (hfs_part_find(sb, &part_start, &part_size)) |
| 135 | return -EINVAL; | 178 | goto out; |
| 179 | goto reread; | ||
| 180 | } | ||
| 181 | |||
| 182 | error = hfsplus_submit_bio(sb->s_bdev, | ||
| 183 | part_start + part_size - 2, | ||
| 184 | sbi->s_backup_vhdr, READ); | ||
| 185 | if (error) | ||
| 186 | goto out_free_backup_vhdr; | ||
| 187 | |||
| 188 | error = -EINVAL; | ||
| 189 | if (sbi->s_backup_vhdr->signature != sbi->s_vhdr->signature) { | ||
| 190 | printk(KERN_WARNING | ||
| 191 | "hfs: invalid secondary volume header\n"); | ||
| 192 | goto out_free_backup_vhdr; | ||
| 136 | } | 193 | } |
| 137 | 194 | ||
| 138 | blocksize = be32_to_cpu(vhdr->blocksize); | 195 | blocksize = be32_to_cpu(sbi->s_vhdr->blocksize); |
| 139 | brelse(bh); | ||
| 140 | 196 | ||
| 141 | /* block size must be at least as large as a sector | 197 | /* |
| 142 | * and a multiple of 2 | 198 | * Block size must be at least as large as a sector and a multiple of 2. |
| 143 | */ | 199 | */ |
| 144 | if (blocksize < HFSPLUS_SECTOR_SIZE || | 200 | if (blocksize < HFSPLUS_SECTOR_SIZE || ((blocksize - 1) & blocksize)) |
| 145 | ((blocksize - 1) & blocksize)) | 201 | goto out_free_backup_vhdr; |
| 146 | return -EINVAL; | ||
| 147 | sbi->alloc_blksz = blocksize; | 202 | sbi->alloc_blksz = blocksize; |
| 148 | sbi->alloc_blksz_shift = 0; | 203 | sbi->alloc_blksz_shift = 0; |
| 149 | while ((blocksize >>= 1) != 0) | 204 | while ((blocksize >>= 1) != 0) |
| 150 | sbi->alloc_blksz_shift++; | 205 | sbi->alloc_blksz_shift++; |
| 151 | blocksize = min(sbi->alloc_blksz, (u32)PAGE_SIZE); | 206 | blocksize = min(sbi->alloc_blksz, (u32)PAGE_SIZE); |
| 152 | 207 | ||
| 153 | /* align block size to block offset */ | 208 | /* |
| 209 | * Align block size to block offset. | ||
| 210 | */ | ||
| 154 | while (part_start & ((blocksize >> HFSPLUS_SECTOR_SHIFT) - 1)) | 211 | while (part_start & ((blocksize >> HFSPLUS_SECTOR_SHIFT) - 1)) |
| 155 | blocksize >>= 1; | 212 | blocksize >>= 1; |
| 156 | 213 | ||
| 157 | if (sb_set_blocksize(sb, blocksize) != blocksize) { | 214 | if (sb_set_blocksize(sb, blocksize) != blocksize) { |
| 158 | printk(KERN_ERR "hfs: unable to set blocksize to %u!\n", blocksize); | 215 | printk(KERN_ERR "hfs: unable to set blocksize to %u!\n", blocksize); |
| 159 | return -EINVAL; | 216 | goto out_free_backup_vhdr; |
| 160 | } | 217 | } |
| 161 | 218 | ||
| 162 | sbi->blockoffset = | 219 | sbi->blockoffset = |
| 163 | part_start >> (sb->s_blocksize_bits - HFSPLUS_SECTOR_SHIFT); | 220 | part_start >> (sb->s_blocksize_bits - HFSPLUS_SECTOR_SHIFT); |
| 221 | sbi->part_start = part_start; | ||
| 164 | sbi->sect_count = part_size; | 222 | sbi->sect_count = part_size; |
| 165 | sbi->fs_shift = sbi->alloc_blksz_shift - sb->s_blocksize_bits; | 223 | sbi->fs_shift = sbi->alloc_blksz_shift - sb->s_blocksize_bits; |
| 166 | |||
| 167 | bh = sb_bread512(sb, part_start + HFSPLUS_VOLHEAD_SECTOR, vhdr); | ||
| 168 | if (!bh) | ||
| 169 | return -EIO; | ||
| 170 | |||
| 171 | /* should still be the same... */ | ||
| 172 | if (test_bit(HFSPLUS_SB_HFSX, &sbi->flags)) { | ||
| 173 | if (vhdr->signature != cpu_to_be16(HFSPLUS_VOLHEAD_SIGX)) | ||
| 174 | goto error; | ||
| 175 | } else { | ||
| 176 | if (vhdr->signature != cpu_to_be16(HFSPLUS_VOLHEAD_SIG)) | ||
| 177 | goto error; | ||
| 178 | } | ||
| 179 | |||
| 180 | sbi->s_vhbh = bh; | ||
| 181 | sbi->s_vhdr = vhdr; | ||
| 182 | |||
| 183 | return 0; | 224 | return 0; |
| 184 | error: | 225 | |
| 185 | brelse(bh); | 226 | out_free_backup_vhdr: |
| 186 | return -EINVAL; | 227 | kfree(sbi->s_backup_vhdr); |
| 228 | out_free_vhdr: | ||
| 229 | kfree(sbi->s_vhdr); | ||
| 230 | out: | ||
| 231 | return error; | ||
| 187 | } | 232 | } |
