| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #define _FILE_OFFSET_BITS 64 |
| #define _LARGEFILE64_SOURCE 1 |
| |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <sys/types.h> |
| #include <sys/mman.h> |
| #include <fcntl.h> |
| #include <inttypes.h> |
| #include <libgen.h> |
| #include <unistd.h> |
| |
| #include <sparse/sparse.h> |
| |
| #include "ext4_utils.h" |
| #include "make_ext4fs.h" |
| #include "allocate.h" |
| |
| #if defined(__APPLE__) && defined(__MACH__) |
| #define off64_t off_t |
| #endif |
| |
| #ifndef USE_MINGW /* O_BINARY is windows-specific flag */ |
| #define O_BINARY 0 |
| #endif |
| |
| extern struct fs_info info; |
| |
| static int verbose = 0; |
| |
| static void usage(char *path) |
| { |
| fprintf(stderr, "%s [ options ] <image or block device> <output image>\n", path); |
| fprintf(stderr, "\n"); |
| fprintf(stderr, " -c include CRC block\n"); |
| fprintf(stderr, " -v verbose output\n"); |
| fprintf(stderr, " -z gzip output\n"); |
| fprintf(stderr, " -S don't use sparse output format\n"); |
| } |
| |
| static int read_ext(int fd) |
| { |
| off64_t ret; |
| struct ext4_super_block sb; |
| |
| ret = lseek64(fd, 1024, SEEK_SET); |
| if (ret < 0) |
| critical_error_errno("failed to seek to superblock"); |
| |
| ret = read(fd, &sb, sizeof(sb)); |
| if (ret < 0) |
| critical_error_errno("failed to read superblock"); |
| if (ret != sizeof(sb)) |
| critical_error("failed to read all of superblock"); |
| |
| ext4_parse_sb_info(&sb); |
| |
| ret = lseek64(fd, info.len, SEEK_SET); |
| if (ret < 0) |
| critical_error_errno("failed to seek to end of input image"); |
| |
| ret = lseek64(fd, info.block_size * (aux_info.first_data_block + 1), SEEK_SET); |
| if (ret < 0) |
| critical_error_errno("failed to seek to block group descriptors"); |
| |
| ret = read(fd, aux_info.bg_desc, info.block_size * aux_info.bg_desc_blocks); |
| if (ret < 0) |
| critical_error_errno("failed to read block group descriptors"); |
| if (ret != (int)info.block_size * (int)aux_info.bg_desc_blocks) |
| critical_error("failed to read all of block group descriptors"); |
| |
| if (verbose) { |
| printf("Found filesystem with parameters:\n"); |
| printf(" Size: %"PRIu64"\n", info.len); |
| printf(" Block size: %d\n", info.block_size); |
| printf(" Blocks per group: %d\n", info.blocks_per_group); |
| printf(" Inodes per group: %d\n", info.inodes_per_group); |
| printf(" Inode size: %d\n", info.inode_size); |
| printf(" Label: %s\n", info.label); |
| printf(" Blocks: %"PRIu64"\n", aux_info.len_blocks); |
| printf(" Block groups: %d\n", aux_info.groups); |
| printf(" Reserved block group size: %d\n", info.bg_desc_reserve_blocks); |
| printf(" Used %d/%d inodes and %d/%d blocks\n", |
| aux_info.sb->s_inodes_count - aux_info.sb->s_free_inodes_count, |
| aux_info.sb->s_inodes_count, |
| aux_info.sb->s_blocks_count_lo - aux_info.sb->s_free_blocks_count_lo, |
| aux_info.sb->s_blocks_count_lo); |
| } |
| |
| return 0; |
| } |
| |
| static int bitmap_get_bit(u8 *bitmap, u32 bit) |
| { |
| if (bitmap[bit / 8] & 1 << (bit % 8)) |
| return 1; |
| |
| return 0; |
| } |
| |
| static int build_sparse_ext(int fd, const char *filename) |
| { |
| unsigned int i; |
| unsigned int block; |
| int start_contiguous_block; |
| u8 *block_bitmap; |
| off64_t ret; |
| |
| block_bitmap = malloc(info.block_size); |
| if (!block_bitmap) |
| critical_error("failed to allocate block bitmap"); |
| |
| if (aux_info.first_data_block > 0) |
| sparse_file_add_file(ext4_sparse_file, filename, 0, |
| info.block_size * aux_info.first_data_block, 0); |
| |
| for (i = 0; i < aux_info.groups; i++) { |
| u32 first_block = aux_info.first_data_block + i * info.blocks_per_group; |
| u32 last_block = min(info.blocks_per_group, aux_info.len_blocks - first_block); |
| |
| ret = lseek64(fd, (u64)info.block_size * aux_info.bg_desc[i].bg_block_bitmap, |
| SEEK_SET); |
| if (ret < 0) |
| critical_error_errno("failed to seek to block group bitmap %d", i); |
| |
| ret = read(fd, block_bitmap, info.block_size); |
| if (ret < 0) |
| critical_error_errno("failed to read block group bitmap %d", i); |
| if (ret != (int)info.block_size) |
| critical_error("failed to read all of block group bitmap %d", i); |
| |
| start_contiguous_block = -1; |
| for (block = 0; block < last_block; block++) { |
| if (start_contiguous_block >= 0) { |
| if (!bitmap_get_bit(block_bitmap, block)) { |
| u32 start_block = first_block + start_contiguous_block; |
| u32 len_blocks = block - start_contiguous_block; |
| |
| sparse_file_add_file(ext4_sparse_file, filename, |
| (u64)info.block_size * start_block, |
| info.block_size * len_blocks, start_block); |
| start_contiguous_block = -1; |
| } |
| } else { |
| if (bitmap_get_bit(block_bitmap, block)) |
| start_contiguous_block = block; |
| } |
| } |
| |
| if (start_contiguous_block >= 0) { |
| u32 start_block = first_block + start_contiguous_block; |
| u32 len_blocks = last_block - start_contiguous_block; |
| sparse_file_add_file(ext4_sparse_file, filename, |
| (u64)info.block_size * start_block, |
| info.block_size * len_blocks, start_block); |
| } |
| } |
| |
| return 0; |
| } |
| |
| int main(int argc, char **argv) |
| { |
| int opt; |
| const char *in = NULL; |
| const char *out = NULL; |
| int gzip = 0; |
| int sparse = 1; |
| int infd, outfd; |
| int crc = 0; |
| |
| while ((opt = getopt(argc, argv, "cvzS")) != -1) { |
| switch (opt) { |
| case 'c': |
| crc = 1; |
| break; |
| case 'v': |
| verbose = 1; |
| break; |
| case 'z': |
| gzip = 1; |
| break; |
| case 'S': |
| sparse = 0; |
| break; |
| } |
| } |
| |
| if (optind >= argc) { |
| fprintf(stderr, "Expected image or block device after options\n"); |
| usage(argv[0]); |
| exit(EXIT_FAILURE); |
| } |
| |
| in = argv[optind++]; |
| |
| if (optind >= argc) { |
| fprintf(stderr, "Expected output image after input image\n"); |
| usage(argv[0]); |
| exit(EXIT_FAILURE); |
| } |
| |
| out = argv[optind++]; |
| |
| if (optind < argc) { |
| fprintf(stderr, "Unexpected argument: %s\n", argv[optind]); |
| usage(argv[0]); |
| exit(EXIT_FAILURE); |
| } |
| |
| infd = open(in, O_RDONLY); |
| |
| if (infd < 0) |
| critical_error_errno("failed to open input image"); |
| |
| read_ext(infd); |
| |
| ext4_sparse_file = sparse_file_new(info.block_size, info.len); |
| |
| build_sparse_ext(infd, in); |
| |
| close(infd); |
| |
| if (strcmp(out, "-")) { |
| outfd = open(out, O_WRONLY | O_CREAT | O_TRUNC | O_BINARY, 0644); |
| if (outfd < 0) { |
| error_errno("open"); |
| return EXIT_FAILURE; |
| } |
| } else { |
| outfd = STDOUT_FILENO; |
| } |
| |
| write_ext4_image(outfd, gzip, sparse, crc); |
| close(outfd); |
| |
| sparse_file_destroy(ext4_sparse_file); |
| |
| return 0; |
| } |