X-Git-Url: https://git.whamcloud.com/?a=blobdiff_plain;f=debugfs%2Fhtree.c;h=a9f9211ba563108a9e2d04946004a1d277328bce;hb=cfe767c083ffc375203257175d187f6bb5f779e4;hp=1659b63f083ad00510fc04fb385790a25f90e52a;hpb=5dd77dbe5a0ac6d78c1c6441fae4087be56d9088;p=tools%2Fe2fsprogs.git diff --git a/debugfs/htree.c b/debugfs/htree.c index 1659b63..a9f9211 100644 --- a/debugfs/htree.c +++ b/debugfs/htree.c @@ -1,10 +1,11 @@ /* * htree.c --- hash tree routines - * + * * Copyright (C) 2002 Theodore Ts'o. This file may be redistributed * under the terms of the GNU Public License. */ +#include "config.h" #include #include #include @@ -17,42 +18,53 @@ #include #ifdef HAVE_GETOPT_H #include -#else +#else extern int optind; extern char *optarg; #endif #include "debugfs.h" +#include "uuid/uuid.h" +#include "e2p/e2p.h" static FILE *pager; static void htree_dump_leaf_node(ext2_filsys fs, ext2_ino_t ino, struct ext2_inode *inode, struct ext2_dx_root_info * rootnode, - blk_t blk, char *buf) + blk64_t blk, char *buf) { errcode_t errcode; struct ext2_dir_entry *dirent; int thislen, col = 0; unsigned int offset = 0; char name[EXT2_NAME_LEN + 1]; - char tmp[EXT2_NAME_LEN + 16]; - blk_t pblk; + char tmp[EXT2_NAME_LEN + 64]; + blk64_t pblk; ext2_dirhash_t hash, minor_hash; - int rec_len, hash_alg; - - errcode = ext2fs_bmap(fs, ino, inode, buf, 0, blk, &pblk); + unsigned int rec_len; + int hash_alg; + int hash_flags = inode->i_flags & EXT4_CASEFOLD_FL; + int csum_size = 0; + + if (ext2fs_has_feature_metadata_csum(fs->super)) + csum_size = sizeof(struct ext2_dir_entry_tail); + + errcode = ext2fs_bmap2(fs, ino, inode, buf, 0, blk, 0, &pblk); if (errcode) { com_err("htree_dump_leaf_node", errcode, - "while mapping logical block %u\n", blk); + "while mapping logical block %llu\n", + (unsigned long long) blk); return; } - printf("Reading directory block %lu, phys %lu\n", blk, pblk); - errcode = ext2fs_read_dir_block2(current_fs, pblk, buf, 0); + fprintf(pager, "Reading directory block %llu, phys %llu\n", + (unsigned long long) blk, (unsigned long long) pblk); + errcode = ext2fs_read_dir_block4(current_fs, pblk, buf, 0, ino); if (errcode) { com_err("htree_dump_leaf_node", errcode, - "while reading block %lu (%lu)\n", blk, pblk); + "while reading block %llu (%llu)\n", + (unsigned long long) blk, (unsigned long long) pblk); return; } hash_alg = rootnode->hash_version; @@ -62,27 +74,48 @@ static void htree_dump_leaf_node(ext2_filsys fs, ext2_ino_t ino, while (offset < fs->blocksize) { dirent = (struct ext2_dir_entry *) (buf + offset); - rec_len = (dirent->rec_len || fs->blocksize < 65536) ? - dirent->rec_len : 65536; + errcode = ext2fs_get_rec_len(fs, dirent, &rec_len); + if (errcode) { + com_err("htree_dump_leaf_inode", errcode, + "while getting rec_len for block %lu", + (unsigned long) blk); + return; + } + thislen = ext2fs_dirent_name_len(dirent); if (((offset + rec_len) > fs->blocksize) || (rec_len < 8) || ((rec_len % 4) != 0) || - (((dirent->name_len & 0xFF)+8) > rec_len)) { - fprintf(pager, "Corrupted directory block (%u)!\n", blk); + ((unsigned) thislen + 8 > rec_len)) { + fprintf(pager, "Corrupted directory block (%llu)!\n", + (unsigned long long) blk); break; } - thislen = ((dirent->name_len & 0xFF) < EXT2_NAME_LEN) ? - (dirent->name_len & 0xFF) : EXT2_NAME_LEN; strncpy(name, dirent->name, thislen); name[thislen] = '\0'; - errcode = ext2fs_dirhash(hash_alg, name, - thislen, fs->super->s_hash_seed, - &hash, &minor_hash); + errcode = ext2fs_dirhash2(hash_alg, name, thislen, + fs->encoding, hash_flags, + fs->super->s_hash_seed, + &hash, &minor_hash); if (errcode) com_err("htree_dump_leaf_node", errcode, "while calculating hash"); - sprintf(tmp, "%u 0x%08x-%08x (%d) %s ", dirent->inode, - hash, minor_hash, rec_len, name); + if ((offset == fs->blocksize - csum_size) && + (dirent->inode == 0) && + (dirent->rec_len == csum_size) && + (dirent->name_len == EXT2_DIR_NAME_LEN_CSUM)) { + struct ext2_dir_entry_tail *t; + + t = (struct ext2_dir_entry_tail *) dirent; + + snprintf(tmp, EXT2_NAME_LEN + 64, + "leaf block checksum: 0x%08x ", + t->det_checksum); + } else { + snprintf(tmp, EXT2_NAME_LEN + 64, + "%u 0x%08x-%08x (%d) %s ", + dirent->inode, hash, minor_hash, + rec_len, name); + } thislen = strlen(tmp); if (col + thislen > 80) { fprintf(pager, "\n"); @@ -99,28 +132,44 @@ static void htree_dump_leaf_node(ext2_filsys fs, ext2_ino_t ino, static void htree_dump_int_block(ext2_filsys fs, ext2_ino_t ino, struct ext2_inode *inode, struct ext2_dx_root_info * rootnode, - blk_t blk, char *buf, int level); + blk64_t blk, char *buf, int level); static void htree_dump_int_node(ext2_filsys fs, ext2_ino_t ino, struct ext2_inode *inode, struct ext2_dx_root_info * rootnode, - struct ext2_dx_entry *ent, + struct ext2_dx_entry *ent, __u32 crc, char *buf, int level) { - struct ext2_dx_countlimit limit; - struct ext2_dx_entry e; + struct ext2_dx_countlimit dx_countlimit; + struct ext2_dx_tail *tail; int hash, i; - + int limit, count; + int remainder; + + dx_countlimit = *((struct ext2_dx_countlimit *) ent); + count = ext2fs_le16_to_cpu(dx_countlimit.count); + limit = ext2fs_le16_to_cpu(dx_countlimit.limit); - limit = *((struct ext2_dx_countlimit *) ent); - limit.count = ext2fs_le16_to_cpu(limit.count); - limit.limit = ext2fs_le16_to_cpu(limit.limit); + fprintf(pager, "Number of entries (count): %d\n", count); + fprintf(pager, "Number of entries (limit): %d\n", limit); - fprintf(pager, "Number of entries (count): %d\n", limit.count); - fprintf(pager, "Number of entries (limit): %d\n", limit.limit); + remainder = fs->blocksize - (limit * sizeof(struct ext2_dx_entry)); + if (ent == (struct ext2_dx_entry *)(rootnode + 1)) + remainder -= sizeof(struct ext2_dx_root_info) + 24; + else + remainder -= 8; + if (ext2fs_has_feature_metadata_csum(fs->super) && + remainder == sizeof(struct ext2_dx_tail)) { + tail = (struct ext2_dx_tail *)(ent + limit); + fprintf(pager, "Checksum: 0x%08x", + ext2fs_le32_to_cpu(tail->dt_checksum)); + if (tail->dt_checksum != crc) + fprintf(pager, " --- EXPECTED: 0x%08x", crc); + fputc('\n', pager); + } - for (i=0; i < limit.count; i++) { + for (i=0; i < count; i++) { hash = i ? ext2fs_le32_to_cpu(ent[i].hash) : 0; fprintf(pager, "Entry #%d: Hash 0x%08x%s, block %u\n", i, hash, (hash & 1) ? " (**)" : "", @@ -129,17 +178,19 @@ static void htree_dump_int_node(ext2_filsys fs, ext2_ino_t ino, fprintf(pager, "\n"); - for (i=0; i < limit.count; i++) { - e.hash = ext2fs_le32_to_cpu(ent[i].hash); - e.block = ext2fs_le32_to_cpu(ent[i].block); + for (i=0; i < count; i++) { + unsigned int hashval, block; + + hashval = ext2fs_le32_to_cpu(ent[i].hash); + block = ext2fs_le32_to_cpu(ent[i].block); fprintf(pager, "Entry #%d: Hash 0x%08x, block %u\n", i, - i ? e.hash : 0, e.block); + i ? hashval : 0, block); if (level) htree_dump_int_block(fs, ino, inode, rootnode, - e.block, buf, level-1); + block, buf, level-1); else htree_dump_leaf_node(fs, ino, inode, rootnode, - e.block, buf); + block, buf); } fprintf(pager, "---------------------\n"); @@ -148,81 +199,70 @@ static void htree_dump_int_node(ext2_filsys fs, ext2_ino_t ino, static void htree_dump_int_block(ext2_filsys fs, ext2_ino_t ino, struct ext2_inode *inode, struct ext2_dx_root_info * rootnode, - blk_t blk, char *buf, int level) + blk64_t blk, char *buf, int level) { char *cbuf; errcode_t errcode; - blk_t pblk; + blk64_t pblk; + __u32 crc; cbuf = malloc(fs->blocksize); if (!cbuf) { fprintf(pager, "Couldn't allocate child block.\n"); return; } - - errcode = ext2fs_bmap(fs, ino, inode, buf, 0, blk, &pblk); + + errcode = ext2fs_bmap2(fs, ino, inode, buf, 0, blk, 0, &pblk); if (errcode) { com_err("htree_dump_int_block", errcode, - "while mapping logical block %u\n", blk); + "while mapping logical block %llu\n", + (unsigned long long) blk); goto errout; } - errcode = io_channel_read_blk(current_fs->io, pblk, 1, buf); + errcode = io_channel_read_blk64(current_fs->io, pblk, 1, buf); if (errcode) { com_err("htree_dump_int_block", errcode, - "while reading block %u\n", blk); + "while reading block %llu\n", + (unsigned long long) blk); goto errout; } + errcode = ext2fs_dx_csum(current_fs, ino, + (struct ext2_dir_entry *) buf, &crc, NULL); + if (errcode) { + com_err("htree_dump_int_block", errcode, + "while calculating checksum for logical block %llu\n", + (unsigned long long) blk); + crc = (unsigned int) -1; + } htree_dump_int_node(fs, ino, inode, rootnode, (struct ext2_dx_entry *) (buf+8), - cbuf, level); + crc, cbuf, level); errout: free(cbuf); } -void do_htree_dump(int argc, char *argv[]) +void do_htree_dump(int argc, char *argv[], int sci_idx EXT2FS_ATTR((unused)), + void *infop EXT2FS_ATTR((unused))) { ext2_ino_t ino; struct ext2_inode inode; - int c; - int long_opt = 0; - blk_t blk; + blk64_t blk; char *buf = NULL; struct ext2_dx_root_info *rootnode; struct ext2_dx_entry *ent; - struct ext2_dx_countlimit *limit; errcode_t errcode; + __u32 crc; if (check_fs_open(argv[0])) return; pager = open_pager(); - reset_getopt(); - while ((c = getopt (argc, argv, "l")) != EOF) { - switch (c) { - case 'l': - long_opt++; - break; - default: - goto print_usage; - } - } - - if (argc > optind+1) { - print_usage: - com_err(0, 0, "Usage: htree_dump [-l] file"); - goto errout; - } - - if (argc == optind) - ino = cwd; - else - ino = string_to_inode(argv[optind]); - if (!ino) + if (common_inode_args_process(argc, argv, &ino, 0)) goto errout; if (debugfs_read_inode(ino, &inode, argv[1])) @@ -232,7 +272,7 @@ void do_htree_dump(int argc, char *argv[]) com_err(argv[0], 0, "Not a directory"); goto errout; } - + if ((inode.i_flags & EXT2_BTREE_FL) == 0) { com_err(argv[0], 0, "Not a hash-indexed directory"); goto errout; @@ -244,15 +284,15 @@ void do_htree_dump(int argc, char *argv[]) goto errout; } - errcode = ext2fs_bmap(current_fs, ino, &inode, buf, 0, 0, &blk); + errcode = ext2fs_bmap2(current_fs, ino, &inode, buf, 0, 0, 0, &blk); if (errcode) { com_err("do_htree_block", errcode, "while mapping logical block 0\n"); goto errout; } - errcode = io_channel_read_blk(current_fs->io, blk, - 1, buf); + errcode = io_channel_read_blk64(current_fs->io, blk, + 1, buf); if (errcode) { com_err(argv[0], errcode, "Error reading root node"); goto errout; @@ -265,39 +305,68 @@ void do_htree_dump(int argc, char *argv[]) fprintf(pager, "\t Hash Version: %d\n", rootnode->hash_version); fprintf(pager, "\t Info length: %d\n", rootnode->info_length); fprintf(pager, "\t Indirect levels: %d\n", rootnode->indirect_levels); - fprintf(pager, "\t Flags: %d\n", rootnode->unused_flags); + fprintf(pager, "\t Flags: %#x\n", rootnode->unused_flags); - ent = (struct ext2_dx_entry *) (buf + 24 + rootnode->info_length); - limit = (struct ext2_dx_countlimit *) ent; + ent = (struct ext2_dx_entry *) + ((char *)rootnode + rootnode->info_length); - htree_dump_int_node(current_fs, ino, &inode, rootnode, ent, + errcode = ext2fs_dx_csum(current_fs, ino, + (struct ext2_dir_entry *) buf, &crc, NULL); + if (errcode) { + com_err("htree_dump_int_block", errcode, + "while calculating checksum for htree root\n"); + crc = (unsigned int) -1; + } + htree_dump_int_node(current_fs, ino, &inode, rootnode, ent, crc, buf + current_fs->blocksize, rootnode->indirect_levels); errout: - if (buf) - free(buf); + free(buf); close_pager(pager); } /* * This function prints the hash of a given file. */ -void do_dx_hash(int argc, char *argv[]) +void do_dx_hash(int argc, char *argv[], int sci_idx EXT2FS_ATTR((unused)), + void *infop EXT2FS_ATTR((unused))) { ext2_dirhash_t hash, minor_hash; errcode_t err; int c; int hash_version = 0; __u32 hash_seed[4]; - + int hash_flags = 0; + const struct ext2fs_nls_table *encoding = NULL; + hash_seed[0] = hash_seed[1] = hash_seed[2] = hash_seed[3] = 0; reset_getopt(); - while ((c = getopt (argc, argv, "h:")) != EOF) { + while ((c = getopt(argc, argv, "h:s:ce:")) != EOF) { switch (c) { case 'h': - hash_version = atoi(optarg); + hash_version = e2p_string2hash(optarg); + if (hash_version < 0) + hash_version = atoi(optarg); + break; + case 's': + if (uuid_parse(optarg, (unsigned char *) hash_seed)) { + fprintf(stderr, "Invalid UUID format: %s\n", + optarg); + return; + } + break; + case 'c': + hash_flags |= EXT4_CASEFOLD_FL; + break; + case 'e': + encoding = ext2fs_load_nls_table(e2p_str2encoding(optarg)); + if (!encoding) { + fprintf(stderr, "Invalid encoding: %s\n", + optarg); + return; + } break; default: goto print_usage; @@ -305,13 +374,16 @@ void do_dx_hash(int argc, char *argv[]) } if (optind != argc-1) { print_usage: - com_err(argv[0], 0, "usage: dx_hash filename"); + com_err(argv[0], 0, "usage: dx_hash [-h hash_alg] " + "[-s hash_seed] [-c] [-e encoding] filename"); return; } - err = ext2fs_dirhash(hash_version, argv[optind], strlen(argv[optind]), - hash_seed, &hash, &minor_hash); + err = ext2fs_dirhash2(hash_version, argv[optind], + strlen(argv[optind]), encoding, hash_flags, + hash_seed, &hash, &minor_hash); + if (err) { - com_err(argv[0], err, "while caclulating hash"); + com_err(argv[0], err, "while calculating hash"); return; } printf("Hash of %s is 0x%0x (minor 0x%0x)\n", argv[optind], @@ -329,15 +401,16 @@ struct process_block_struct { int len; }; -static int search_dir_block(ext2_filsys fs, blk_t *blocknr, - e2_blkcnt_t blockcnt, blk_t ref_blk, +static int search_dir_block(ext2_filsys fs, blk64_t *blocknr, + e2_blkcnt_t blockcnt, blk64_t ref_blk, int ref_offset, void *priv_data); -void do_dirsearch(int argc, char *argv[]) +void do_dirsearch(int argc, char *argv[], int sci_idx EXT2FS_ATTR((unused)), + void *infop EXT2FS_ATTR((unused))) { ext2_ino_t inode; struct process_block_struct pb; - + if (check_fs_open(argv[0])) return; @@ -357,17 +430,17 @@ void do_dirsearch(int argc, char *argv[]) } pb.search_name = argv[2]; pb.len = strlen(pb.search_name); - - ext2fs_block_iterate2(current_fs, inode, BLOCK_FLAG_READ_ONLY, 0, + + ext2fs_block_iterate3(current_fs, inode, BLOCK_FLAG_READ_ONLY, 0, search_dir_block, &pb); free(pb.buf); } -static int search_dir_block(ext2_filsys fs, blk_t *blocknr, - e2_blkcnt_t blockcnt, - blk_t ref_blk EXT2FS_ATTR((unused)), +static int search_dir_block(ext2_filsys fs, blk64_t *blocknr, + e2_blkcnt_t blockcnt, + blk64_t ref_blk EXT2FS_ATTR((unused)), int ref_offset EXT2FS_ATTR((unused)), void *priv_data) { @@ -375,14 +448,14 @@ static int search_dir_block(ext2_filsys fs, blk_t *blocknr, struct ext2_dir_entry *dirent; errcode_t errcode; unsigned int offset = 0; - int rec_len; + unsigned int rec_len; if (blockcnt < 0) return 0; p = (struct process_block_struct *) priv_data; - errcode = io_channel_read_blk(current_fs->io, *blocknr, 1, p->buf); + errcode = io_channel_read_blk64(current_fs->io, *blocknr, 1, p->buf); if (errcode) { com_err("search_dir_block", errcode, "while reading block %lu", (unsigned long) *blocknr); @@ -391,15 +464,20 @@ static int search_dir_block(ext2_filsys fs, blk_t *blocknr, while (offset < fs->blocksize) { dirent = (struct ext2_dir_entry *) (p->buf + offset); - rec_len = (dirent->rec_len || fs->blocksize < 65536) ? - dirent->rec_len : 65536; + errcode = ext2fs_get_rec_len(fs, dirent, &rec_len); + if (errcode) { + com_err("htree_dump_leaf_inode", errcode, + "while getting rec_len for block %lu", + (unsigned long) *blocknr); + return BLOCK_ABORT; + } if (dirent->inode && - p->len == (dirent->name_len & 0xFF) && + p->len == ext2fs_dirent_name_len(dirent) && strncmp(p->search_name, dirent->name, p->len) == 0) { printf("Entry found at logical block %lld, " - "phys %u, offset %u\n", (long long)blockcnt, - *blocknr, offset); + "phys %llu, offset %u\n", (long long)blockcnt, + (unsigned long long) *blocknr, offset); printf("offset %u\n", offset); return BLOCK_ABORT; }