inode.c
来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 2,057 行 · 第 1/5 页
C
2,057 行
} if (ni->itype.index.block_size > PAGE_CACHE_SIZE) { ntfs_error(vi->i_sb, "Index block size (%u) > " "PAGE_CACHE_SIZE (%ld) is not " "supported. Sorry.", ni->itype.index.block_size, PAGE_CACHE_SIZE); err = -EOPNOTSUPP; goto unm_err_out; } if (ni->itype.index.block_size < NTFS_BLOCK_SIZE) { ntfs_error(vi->i_sb, "Index block size (%u) < " "NTFS_BLOCK_SIZE (%i) is not " "supported. Sorry.", ni->itype.index.block_size, NTFS_BLOCK_SIZE); err = -EOPNOTSUPP; goto unm_err_out; } ni->itype.index.block_size_bits = ffs(ni->itype.index.block_size) - 1; /* Determine the size of a vcn in the directory index. */ if (vol->cluster_size <= ni->itype.index.block_size) { ni->itype.index.vcn_size = vol->cluster_size; ni->itype.index.vcn_size_bits = vol->cluster_size_bits; } else { ni->itype.index.vcn_size = vol->sector_size; ni->itype.index.vcn_size_bits = vol->sector_size_bits; } /* Setup the index allocation attribute, even if not present. */ NInoSetMstProtected(ni); ni->type = AT_INDEX_ALLOCATION; ni->name = I30; ni->name_len = 4; if (!(ir->index.flags & LARGE_INDEX)) { /* No index allocation. */ vi->i_size = ni->initialized_size = ni->allocated_size = 0; /* We are done with the mft record, so we release it. */ ntfs_attr_put_search_ctx(ctx); unmap_mft_record(ni); m = NULL; ctx = NULL; goto skip_large_dir_stuff; } /* LARGE_INDEX: Index allocation present. Setup state. */ NInoSetIndexAllocPresent(ni); /* Find index allocation attribute. */ ntfs_attr_reinit_search_ctx(ctx); err = ntfs_attr_lookup(AT_INDEX_ALLOCATION, I30, 4, CASE_SENSITIVE, 0, NULL, 0, ctx); if (unlikely(err)) { if (err == -ENOENT) ntfs_error(vi->i_sb, "$INDEX_ALLOCATION " "attribute is not present but " "$INDEX_ROOT indicated it " "is."); else ntfs_error(vi->i_sb, "Failed to lookup " "$INDEX_ALLOCATION " "attribute."); goto unm_err_out; } if (!ctx->attr->non_resident) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute " "is resident."); goto unm_err_out; } if (ctx->attr->flags & ATTR_IS_ENCRYPTED) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute " "is encrypted."); goto unm_err_out; } if (ctx->attr->flags & ATTR_IS_SPARSE) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute " "is sparse."); goto unm_err_out; } if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { ntfs_error(vi->i_sb, "$INDEX_ALLOCATION attribute " "is compressed."); goto unm_err_out; } if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of " "$INDEX_ALLOCATION attribute has non " "zero lowest_vcn. Inode is corrupt. " "You should run chkdsk."); goto unm_err_out; } vi->i_size = sle64_to_cpu( ctx->attr->data.non_resident.data_size); ni->initialized_size = sle64_to_cpu( ctx->attr->data.non_resident.initialized_size); ni->allocated_size = sle64_to_cpu( ctx->attr->data.non_resident.allocated_size); /* * We are done with the mft record, so we release it. Otherwise * we would deadlock in ntfs_attr_iget(). */ ntfs_attr_put_search_ctx(ctx); unmap_mft_record(ni); m = NULL; ctx = NULL; /* Get the index bitmap attribute inode. */ bvi = ntfs_attr_iget(vi, AT_BITMAP, I30, 4); if (IS_ERR(bvi)) { ntfs_error(vi->i_sb, "Failed to get bitmap attribute."); err = PTR_ERR(bvi); goto unm_err_out; } ni->itype.index.bmp_ino = bvi; bni = NTFS_I(bvi); if (NInoCompressed(bni) || NInoEncrypted(bni) || NInoSparse(bni)) { ntfs_error(vi->i_sb, "$BITMAP attribute is compressed " "and/or encrypted and/or sparse."); goto unm_err_out; } /* Consistency check bitmap size vs. index allocation size. */ if ((bvi->i_size << 3) < (vi->i_size >> ni->itype.index.block_size_bits)) { ntfs_error(vi->i_sb, "Index bitmap too small (0x%llx) " "for index allocation (0x%llx).", bvi->i_size << 3, vi->i_size); goto unm_err_out; }skip_large_dir_stuff: /* Everyone gets read and scan permissions. */ vi->i_mode |= S_IRUGO | S_IXUGO; /* If not read-only, set write permissions. */ if (!IS_RDONLY(vi)) vi->i_mode |= S_IWUGO; /* * Apply the directory permissions mask set in the mount * options. */ vi->i_mode &= ~vol->dmask; /* Setup the operations for this inode. */ vi->i_op = &ntfs_dir_inode_ops; vi->i_fop = &ntfs_dir_ops; vi->i_mapping->a_ops = &ntfs_mst_aops; } else { /* It is a file. */ ntfs_attr_reinit_search_ctx(ctx); /* Setup the data attribute, even if not present. */ ni->type = AT_DATA; ni->name = NULL; ni->name_len = 0; /* Find first extent of the unnamed data attribute. */ err = ntfs_attr_lookup(AT_DATA, NULL, 0, 0, 0, NULL, 0, ctx); if (unlikely(err)) { vi->i_size = ni->initialized_size = ni->allocated_size = 0; if (err != -ENOENT) { ntfs_error(vi->i_sb, "Failed to lookup $DATA " "attribute."); goto unm_err_out; } /* * FILE_Secure does not have an unnamed $DATA * attribute, so we special case it here. */ if (vi->i_ino == FILE_Secure) goto no_data_attr_special_case; /* * Most if not all the system files in the $Extend * system directory do not have unnamed data * attributes so we need to check if the parent * directory of the file is FILE_Extend and if it is * ignore this error. To do this we need to get the * name of this inode from the mft record as the name * contains the back reference to the parent directory. */ if (ntfs_is_extended_system_file(ctx) > 0) goto no_data_attr_special_case; // FIXME: File is corrupt! Hot-fix with empty data // attribute if recovery option is set. ntfs_error(vi->i_sb, "$DATA attribute is " "missing."); goto unm_err_out; } /* Setup the state. */ if (ctx->attr->non_resident) { NInoSetNonResident(ni); if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { NInoSetCompressed(ni); if (vol->cluster_size > 4096) { ntfs_error(vi->i_sb, "Found " "compressed data but " "compression is disabled due " "to cluster size (%i) > 4kiB.", vol->cluster_size); goto unm_err_out; } if ((ctx->attr->flags & ATTR_COMPRESSION_MASK) != ATTR_IS_COMPRESSED) { ntfs_error(vi->i_sb, "Found " "unknown compression method or " "corrupt file."); goto unm_err_out; } ni->itype.compressed.block_clusters = 1U << ctx->attr->data.non_resident. compression_unit; if (ctx->attr->data.non_resident. compression_unit != 4) { ntfs_error(vi->i_sb, "Found " "nonstandard compression unit " "(%u instead of 4). Cannot " "handle this. This might " "indicate corruption so you " "should run chkdsk.", ctx->attr->data.non_resident. compression_unit); err = -EOPNOTSUPP; goto unm_err_out; } ni->itype.compressed.block_size = 1U << ( ctx->attr->data.non_resident. compression_unit + vol->cluster_size_bits); ni->itype.compressed.block_size_bits = ffs( ni->itype.compressed.block_size) - 1; } if (ctx->attr->flags & ATTR_IS_ENCRYPTED) { if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { ntfs_error(vi->i_sb, "Found encrypted " "and compressed data."); goto unm_err_out; } NInoSetEncrypted(ni); } if (ctx->attr->flags & ATTR_IS_SPARSE) NInoSetSparse(ni); if (ctx->attr->data.non_resident.lowest_vcn) { ntfs_error(vi->i_sb, "First extent of $DATA " "attribute has non zero " "lowest_vcn. Inode is corrupt. " "You should run chkdsk."); goto unm_err_out; } /* Setup all the sizes. */ vi->i_size = sle64_to_cpu( ctx->attr->data.non_resident.data_size); ni->initialized_size = sle64_to_cpu( ctx->attr->data.non_resident. initialized_size); ni->allocated_size = sle64_to_cpu( ctx->attr->data.non_resident. allocated_size); if (NInoCompressed(ni)) { ni->itype.compressed.size = sle64_to_cpu( ctx->attr->data.non_resident. compressed_size); } } else { /* Resident attribute. */ /* * Make all sizes equal for simplicity in read code * paths. FIXME: Need to keep this in mind when * converting to non-resident attribute in write code * path. (Probably only affects truncate().) */ vi->i_size = ni->initialized_size = ni->allocated_size = le32_to_cpu( ctx->attr->data.resident.value_length); }no_data_attr_special_case: /* We are done with the mft record, so we release it. */ ntfs_attr_put_search_ctx(ctx); unmap_mft_record(ni); m = NULL; ctx = NULL; /* Everyone gets all permissions. */ vi->i_mode |= S_IRWXUGO; /* If read-only, noone gets write permissions. */ if (IS_RDONLY(vi)) vi->i_mode &= ~S_IWUGO; /* Apply the file permissions mask set in the mount options. */ vi->i_mode &= ~vol->fmask; /* Setup the operations for this inode. */ vi->i_op = &ntfs_file_inode_ops; vi->i_fop = &ntfs_file_ops; vi->i_mapping->a_ops = &ntfs_aops; } /* * The number of 512-byte blocks used on disk (for stat). This is in so * far inaccurate as it doesn't account for any named streams or other * special non-resident attributes, but that is how Windows works, too, * so we are at least consistent with Windows, if not entirely * consistent with the Linux Way. Doing it the Linux Way would cause a * significant slowdown as it would involve iterating over all * attributes in the mft record and adding the allocated/compressed * sizes of all non-resident attributes present to give us the Linux * correct size that should go into i_blocks (after division by 512). */ if (S_ISDIR(vi->i_mode) || !NInoCompressed(ni)) vi->i_blocks = ni->allocated_size >> 9; else vi->i_blocks = ni->itype.compressed.size >> 9; ntfs_debug("Done."); return 0;unm_err_out: if (!err) err = -EIO; if (ctx) ntfs_attr_put_search_ctx(ctx); if (m) unmap_mft_record(ni);err_out: ntfs_error(vi->i_sb, "Failed with error code %i. Marking inode 0x%lx " "as bad.", -err, vi->i_ino); make_bad_inode(vi); return err;}/** * ntfs_read_locked_attr_inode - read an attribute inode from its base inode * @base_vi: base inode * @vi: attribute inode to read * * ntfs_read_locked_attr_inode() is called from ntfs_attr_iget() to read the * attribute inode described by @vi into memory from the base mft record * described by @base_ni. * * ntfs_read_locked_attr_inode() maps, pins and locks the base inode for * reading and looks up the attribute described by @vi before setting up the * necessary fields in @vi as well as initializing the ntfs inode. * * Q: What locks are held when the function is called? * A: i_state has I_LOCK set, hence the inode is locked, also * i_count is set to 1, so it is not going to go away * * Return 0 on success and -errno on error. In the error case, the inode will * have had make_bad_inode() executed on it. */static int ntfs_read_locked_attr_inode(struct inode *base_vi, struct inode *vi){ ntfs_volume *vol = NTFS_SB(vi->i_sb); ntfs_inode *ni, *base_ni; MFT_RECORD *m; ntfs_attr_search_ctx *ctx; int err = 0; ntfs_debug("Entering for i_ino 0x%lx.", vi->i_ino); ntfs_init_big_inode(vi); ni = NTFS_I(vi); base_ni = NTFS_I(base_vi); /* Just mirror the values from the base inode. */ vi->i_blksize = base_vi->i_blksize; vi->i_version = base_vi->i_version; vi->i_uid = base_vi->i_uid; vi->i_gid = base_vi->i_gid; vi->i_nlink = base_vi->i_nlink; vi->i_mtime = base_vi->i_mtime; vi->i_ctime = base_vi->i_ctime; vi->i_atime = base_vi->i_atime; vi->i_generation = ni->seq_no = base_ni->seq_no; /* Set inode type to zero but preserve permissions. */ vi->i_mode = base_vi->i_mode & ~S_IFMT; m = map_mft_record(base_ni); if (IS_ERR(m)) { err = PTR_ERR(m); goto err_out; } ctx = ntfs_attr_get_search_ctx(base_ni, m); if (!ctx) { err = -ENOMEM; goto unm_err_out; } /* Find the attribute. */ err = ntfs_attr_lookup(ni->type, ni->name, ni->name_len, CASE_SENSITIVE, 0, NULL, 0, ctx); if (unlikely(err)) goto unm_err_out; if (!ctx->attr->non_resident) { if (NInoMstProtected(ni) || ctx->attr->flags) { ntfs_error(vi->i_sb, "Found mst protected attribute " "or attribute with non-zero flags but " "the attribute is resident (mft_no " "0x%lx, type 0x%x, name_len %i). " "Please report you saw this message " "to linux-ntfs-dev@lists." "sourceforge.net", vi->i_ino, ni->type, ni->name_len); goto unm_err_out; } /* * Resident attribute. Make all sizes equal for simplicity in * read code paths. */ vi->i_size = ni->initialized_size = ni->allocated_size = le32_to_cpu(ctx->attr->data.resident.value_length); } else { NInoSetNonResident(ni); if (ctx->attr->flags & ATTR_COMPRESSION_MASK) { if (NInoMstProtected(ni)) { ntfs_error(vi->i_sb, "Found mst protected " "attribute but the attribute " "is compressed (mft_no 0x%lx, "
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?