/* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public Licens * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111- */ #include #include #include static ext4_fsblk_t ext4_new_meta_blocks(void *icb, struct inode *inode, ext4_fsblk_t goal, unsigned int flags, unsigned long *count, int *errp) { NTSTATUS status; ULONG blockcnt = (count) ? *count : 1; ULONG block = 0; status = Ext2NewBlock((PEXT2_IRP_CONTEXT)icb, inode->i_sb->s_priv, 0, (ULONG)goal, &block, &blockcnt); if (count) *count = blockcnt; if (!NT_SUCCESS(status)) { *errp = Ext2LinuxError(status); return 0; } inode->i_blocks += (blockcnt * (inode->i_sb->s_blocksize >> 9)); return block; } static void ext4_free_blocks(void *icb, struct inode *inode, ext4_fsblk_t block, int count, int flags) { Ext2FreeBlock((PEXT2_IRP_CONTEXT)icb, inode->i_sb->s_priv, (ULONG)block, count); inode->i_blocks -= count * (inode->i_sb->s_blocksize >> 9); return; } static inline ext4_fsblk_t ext4_inode_to_goal_block(struct inode *inode) { PEXT2_VCB Vcb; Vcb = inode->i_sb->s_priv; return (inode->i_ino - 1) / BLOCKS_PER_GROUP; } #define NAME_HASH_SHIFT 5 #define VALUE_HASH_SHIFT 16 static inline void ext4_xattr_compute_hash(struct ext4_xattr_header *header, struct ext4_xattr_entry *entry) { __u32 hash = 0; char *name = EXT4_XATTR_NAME(entry); int n; for (n = 0; n < entry->e_name_len; n++) { hash = (hash << NAME_HASH_SHIFT) ^ (hash >> (8 * sizeof(hash) - NAME_HASH_SHIFT)) ^ *name++; } if (entry->e_value_block == 0 && entry->e_value_size != 0) { __le32 *value = (__le32 *)((char *)header + le16_to_cpu(entry->e_value_offs)); for (n = (le32_to_cpu(entry->e_value_size) + EXT4_XATTR_ROUND) >> EXT4_XATTR_PAD_BITS; n; n--) { hash = (hash << VALUE_HASH_SHIFT) ^ (hash >> (8 * sizeof(hash) - VALUE_HASH_SHIFT)) ^ le32_to_cpu(*value++); } } entry->e_hash = cpu_to_le32(hash); } #define BLOCK_HASH_SHIFT 16 /* * ext4_xattr_rehash() * * Re-compute the extended attribute hash value after an entry has changed. */ static void ext4_xattr_rehash(struct ext4_xattr_header *header, struct ext4_xattr_entry *entry) { struct ext4_xattr_entry *here; __u32 hash = 0; ext4_xattr_compute_hash(header, entry); here = EXT4_XATTR_ENTRY(header + 1); while (!EXT4_XATTR_IS_LAST_ENTRY(here)) { if (!here->e_hash) { /* Block is not shared if an entry's hash value == 0 */ hash = 0; break; } hash = (hash << BLOCK_HASH_SHIFT) ^ (hash >> (8 * sizeof(hash) - BLOCK_HASH_SHIFT)) ^ le32_to_cpu(here->e_hash); here = EXT4_XATTR_NEXT(here); } header->h_hash = cpu_to_le32(hash); } #if CONFIG_META_CSUM_ENABLE static __u32 ext4_xattr_block_checksum(PEXT2_MCB inode_ref, ext4_fsblk_t blocknr, struct ext4_xattr_header *header) { __u32 checksum = 0; __u64 le64_blocknr = blocknr; struct ext4_sblock *sb = &inode_ref->fs->sb; if (ext4_sb_feature_ro_com(sb, EXT4_FRO_COM_METADATA_CSUM)) { __u32 orig_checksum; /* Preparation: temporarily set bg checksum to 0 */ orig_checksum = header->h_checksum; header->h_checksum = 0; /* First calculate crc32 checksum against fs uuid */ checksum = ext4_crc32c(EXT4_CRC32_INIT, sb->uuid, sizeof(sb->uuid)); /* Then calculate crc32 checksum block number */ checksum = ext4_crc32c(checksum, &le64_blocknr, sizeof(le64_blocknr)); /* Finally calculate crc32 checksum against * the entire xattr block */ checksum = ext4_crc32c(checksum, header, ext4_sb_get_block_size(sb)); header->h_checksum = orig_checksum; } return checksum; } #else #define ext4_xattr_block_checksum(...) 0 #endif static void ext4_xattr_set_block_checksum(PEXT2_MCB inode_ref, ext4_fsblk_t blocknr, struct ext4_xattr_header *header) { /* TODO: Need METADATA_CSUM supports. */ header->h_checksum = 0; } static int ext4_xattr_item_cmp(struct rb_node *_a, struct rb_node *_b) { int result; struct ext4_xattr_item *a, *b; a = container_of(_a, struct ext4_xattr_item, node); a = container_of(_a, struct ext4_xattr_item, node); b = container_of(_b, struct ext4_xattr_item, node); if (a->is_data && !b->is_data) return -1; if (!a->is_data && b->is_data) return 1; result = a->name_index - b->name_index; if (result) return result; if (a->name_len < b->name_len) return -1; if (a->name_len > b->name_len) return 1; return memcmp(a->name, b->name, a->name_len); } // // Red-black tree insert routine. // static struct ext4_xattr_item * ext4_xattr_item_search(struct ext4_xattr_ref *xattr_ref, struct ext4_xattr_item *name) { struct rb_node *new = xattr_ref->root.rb_node; while (new) { struct ext4_xattr_item *node = container_of(new, struct ext4_xattr_item, node); int result = ext4_xattr_item_cmp(&name->node, new); if (result < 0) new = new->rb_left; else if (result > 0) new = new->rb_right; else return node; } return NULL; } static void ext4_xattr_item_insert(struct ext4_xattr_ref *xattr_ref, struct ext4_xattr_item *item) { rb_insert(&xattr_ref->root, &item->node, ext4_xattr_item_cmp); list_add_tail(&item->list_node, &xattr_ref->ordered_list); } static void ext4_xattr_item_remove(struct ext4_xattr_ref *xattr_ref, struct ext4_xattr_item *item) { rb_erase(&item->node, &xattr_ref->root); list_del_init(&item->list_node); } static struct ext4_xattr_item * ext4_xattr_item_alloc(__u8 name_index, const char *name, size_t name_len) { struct ext4_xattr_item *item; item = kzalloc(sizeof(struct ext4_xattr_item) + name_len, GFP_NOFS); if (!item) return NULL; item->name_index = name_index; item->name = (char *)(item + 1); item->name_len = name_len; item->data = NULL; item->data_size = 0; INIT_LIST_HEAD(&item->list_node); memcpy(item->name, name, name_len); if (name_index == EXT4_XATTR_INDEX_SYSTEM && name_len == 4 && !memcmp(name, "data", 4)) item->is_data = TRUE; else item->is_data = FALSE; return item; } static int ext4_xattr_item_alloc_data(struct ext4_xattr_item *item, const void *orig_data, size_t data_size) { void *data = NULL; ASSERT(!item->data); data = kmalloc(data_size, GFP_NOFS); if (!data) return -ENOMEM; if (orig_data) memcpy(data, orig_data, data_size); item->data = data; item->data_size = data_size; return 0; } static void ext4_xattr_item_free_data(struct ext4_xattr_item *item) { ASSERT(item->data); kfree(item->data); item->data = NULL; item->data_size = 0; } static int ext4_xattr_item_resize_data(struct ext4_xattr_item *item, size_t new_data_size) { if (new_data_size != item->data_size) { void *new_data; new_data = kmalloc(new_data_size, GFP_NOFS); if (!new_data) return -ENOMEM; memcpy(new_data, item->data, item->data_size); kfree(item->data); item->data = new_data; item->data_size = new_data_size; } return 0; } static void ext4_xattr_item_free(struct ext4_xattr_item *item) { if (item->data) ext4_xattr_item_free_data(item); kfree(item); } static void *ext4_xattr_entry_data(struct ext4_xattr_ref *xattr_ref, struct ext4_xattr_entry *entry, BOOL in_inode) { char *ret; int block_size; if (in_inode) { struct ext4_xattr_ibody_header *header; struct ext4_xattr_entry *first_entry; int inode_size = xattr_ref->fs->InodeSize; header = EXT4_XATTR_IHDR(xattr_ref->OnDiskInode); first_entry = EXT4_XATTR_IFIRST(header); ret = ((char *)first_entry + le16_to_cpu(entry->e_value_offs)); if (ret + EXT4_XATTR_SIZE(le32_to_cpu(entry->e_value_size)) - (char *)xattr_ref->OnDiskInode > inode_size) ret = NULL; return ret; } block_size = xattr_ref->fs->BlockSize; ret = ((char *)xattr_ref->block_bh->b_data + le16_to_cpu(entry->e_value_offs)); if (ret + EXT4_XATTR_SIZE(le32_to_cpu(entry->e_value_size)) - (char *)xattr_ref->block_bh->b_data > block_size) ret = NULL; return ret; } static int ext4_xattr_block_fetch(struct ext4_xattr_ref *xattr_ref) { int ret = 0; size_t size_rem; void *data; struct ext4_xattr_entry *entry = NULL; ASSERT(xattr_ref->block_bh->b_data); entry = EXT4_XATTR_BFIRST(xattr_ref->block_bh); size_rem = xattr_ref->fs->BlockSize; for (; size_rem > 0 && !EXT4_XATTR_IS_LAST_ENTRY(entry); entry = EXT4_XATTR_NEXT(entry), size_rem -= EXT4_XATTR_LEN(entry->e_name_len)) { struct ext4_xattr_item *item; char *e_name = EXT4_XATTR_NAME(entry); data = ext4_xattr_entry_data(xattr_ref, entry, FALSE); if (!data) { ret = -EIO; goto Finish; } item = ext4_xattr_item_alloc(entry->e_name_index, e_name, (size_t)entry->e_name_len); if (!item) { ret = -ENOMEM; goto Finish; } if (ext4_xattr_item_alloc_data( item, data, le32_to_cpu(entry->e_value_size)) != 0) { ext4_xattr_item_free(item); ret = -ENOMEM; goto Finish; } ext4_xattr_item_insert(xattr_ref, item); xattr_ref->block_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); xattr_ref->ea_size += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } Finish: return ret; } static int ext4_xattr_inode_fetch(struct ext4_xattr_ref *xattr_ref) { void *data; size_t size_rem; int ret = 0; struct ext4_xattr_ibody_header *header = NULL; struct ext4_xattr_entry *entry = NULL; int inode_size = xattr_ref->fs->InodeSize; header = EXT4_XATTR_IHDR(xattr_ref->OnDiskInode); entry = EXT4_XATTR_IFIRST(header); size_rem = inode_size - EXT4_GOOD_OLD_INODE_SIZE - xattr_ref->OnDiskInode->i_extra_isize; for (; size_rem > 0 && !EXT4_XATTR_IS_LAST_ENTRY(entry); entry = EXT4_XATTR_NEXT(entry), size_rem -= EXT4_XATTR_LEN(entry->e_name_len)) { struct ext4_xattr_item *item; char *e_name = EXT4_XATTR_NAME(entry); data = ext4_xattr_entry_data(xattr_ref, entry, TRUE); if (!data) { ret = -EIO; goto Finish; } item = ext4_xattr_item_alloc(entry->e_name_index, e_name, (size_t)entry->e_name_len); if (!item) { ret = -ENOMEM; goto Finish; } if (ext4_xattr_item_alloc_data( item, data, le32_to_cpu(entry->e_value_size)) != 0) { ext4_xattr_item_free(item); ret = -ENOMEM; goto Finish; } item->in_inode = TRUE; ext4_xattr_item_insert(xattr_ref, item); xattr_ref->inode_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); xattr_ref->ea_size += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } Finish: return ret; } static __s32 ext4_xattr_inode_space(struct ext4_xattr_ref *xattr_ref) { int inode_size = xattr_ref->fs->InodeSize; int size_rem = inode_size - EXT4_GOOD_OLD_INODE_SIZE - xattr_ref->OnDiskInode->i_extra_isize; return size_rem; } static __s32 ext4_xattr_block_space(struct ext4_xattr_ref *xattr_ref) { return xattr_ref->fs->BlockSize; } static int ext4_xattr_fetch(struct ext4_xattr_ref *xattr_ref) { int ret = 0; int inode_size = xattr_ref->fs->InodeSize; if (inode_size > EXT4_GOOD_OLD_INODE_SIZE) { ret = ext4_xattr_inode_fetch(xattr_ref); if (ret != 0) return ret; } if (xattr_ref->block_loaded) ret = ext4_xattr_block_fetch(xattr_ref); xattr_ref->dirty = FALSE; return ret; } static struct ext4_xattr_item * ext4_xattr_lookup_item(struct ext4_xattr_ref *xattr_ref, __u8 name_index, const char *name, size_t name_len) { struct ext4_xattr_item tmp = { FALSE, FALSE, name_index, (char *)name, /*won't touch this string*/ name_len, }; if (name_index == EXT4_XATTR_INDEX_SYSTEM && name_len == 4 && !memcmp(name, "data", 4)) tmp.is_data = TRUE; return ext4_xattr_item_search(xattr_ref, &tmp); } static struct ext4_xattr_item * ext4_xattr_insert_item(struct ext4_xattr_ref *xattr_ref, __u8 name_index, const char *name, size_t name_len, const void *data, size_t data_size, int *err) { struct ext4_xattr_item *item; item = ext4_xattr_item_alloc(name_index, name, name_len); if (!item) { if (err) *err = -ENOMEM; return NULL; } item->in_inode = TRUE; if (xattr_ref->inode_size_rem < EXT4_XATTR_SIZE(data_size) + EXT4_XATTR_LEN(item->name_len)) { if (xattr_ref->block_size_rem < EXT4_XATTR_SIZE(data_size) + EXT4_XATTR_LEN(item->name_len)) { if (err) *err = -ENOSPC; return NULL; } item->in_inode = FALSE; } if (ext4_xattr_item_alloc_data(item, data, data_size) != 0) { ext4_xattr_item_free(item); if (err) *err = -ENOMEM; return NULL; } ext4_xattr_item_insert(xattr_ref, item); xattr_ref->ea_size += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); if (item->in_inode) { xattr_ref->inode_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } else { xattr_ref->block_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } xattr_ref->dirty = TRUE; if (err) *err = 0; return item; } static struct ext4_xattr_item * ext4_xattr_insert_item_ordered(struct ext4_xattr_ref *xattr_ref, __u8 name_index, const char *name, size_t name_len, const void *data, size_t data_size, int *err) { struct ext4_xattr_item *item, *last_item = NULL; item = ext4_xattr_item_alloc(name_index, name, name_len); if (!item) { if (err) *err = -ENOMEM; return NULL; } if (!list_empty(&xattr_ref->ordered_list)) last_item = list_entry(xattr_ref->ordered_list.prev, struct ext4_xattr_item, list_node); item->in_inode = TRUE; if ((xattr_ref->inode_size_rem < EXT4_XATTR_SIZE(data_size) + EXT4_XATTR_LEN(item->name_len)) || (last_item && !last_item->in_inode)) { if (xattr_ref->block_size_rem < EXT4_XATTR_SIZE(data_size) + EXT4_XATTR_LEN(item->name_len)) { if (err) *err = -ENOSPC; return NULL; } item->in_inode = FALSE; } if (ext4_xattr_item_alloc_data(item, data, data_size) != 0) { ext4_xattr_item_free(item); if (err) *err = -ENOMEM; return NULL; } ext4_xattr_item_insert(xattr_ref, item); xattr_ref->ea_size += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); if (item->in_inode) { xattr_ref->inode_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } else { xattr_ref->block_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } xattr_ref->dirty = TRUE; if (err) *err = 0; return item; } static int ext4_xattr_remove_item(struct ext4_xattr_ref *xattr_ref, __u8 name_index, const char *name, size_t name_len) { int ret = -ENOENT; struct ext4_xattr_item *item = ext4_xattr_lookup_item(xattr_ref, name_index, name, name_len); if (item) { if (item == xattr_ref->iter_from) { struct rb_node *next_node; next_node = rb_next(&item->node); if (next_node) xattr_ref->iter_from = container_of(next_node, struct ext4_xattr_item, node); else xattr_ref->iter_from = NULL; } xattr_ref->ea_size -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); if (item->in_inode) { xattr_ref->inode_size_rem += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } else { xattr_ref->block_size_rem += EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); } ext4_xattr_item_remove(xattr_ref, item); ext4_xattr_item_free(item); xattr_ref->dirty = TRUE; ret = 0; } return ret; } static int ext4_xattr_resize_item(struct ext4_xattr_ref *xattr_ref, struct ext4_xattr_item *item, size_t new_data_size) { int ret = 0; BOOL to_inode = FALSE, to_block = FALSE; size_t old_data_size = item->data_size; size_t orig_room_size = item->in_inode ? xattr_ref->inode_size_rem : xattr_ref->block_size_rem; /* * Check if we can hold this entry in both in-inode and * on-block form * * More complicated case: we do not allow entries stucking in * the middle between in-inode space and on-block space, so * the entry has to stay in either inode space or block space. */ if (item->in_inode) { if (xattr_ref->inode_size_rem + EXT4_XATTR_SIZE(old_data_size) < EXT4_XATTR_SIZE(new_data_size)) { if (xattr_ref->block_size_rem < EXT4_XATTR_SIZE(new_data_size) + EXT4_XATTR_LEN(item->name_len)) return -ENOSPC; to_block = TRUE; } } else { if (xattr_ref->block_size_rem + EXT4_XATTR_SIZE(old_data_size) < EXT4_XATTR_SIZE(new_data_size)) { if (xattr_ref->inode_size_rem < EXT4_XATTR_SIZE(new_data_size) + EXT4_XATTR_LEN(item->name_len)) return -ENOSPC; to_inode = TRUE; } } ret = ext4_xattr_item_resize_data(item, new_data_size); if (ret) return ret; xattr_ref->ea_size = xattr_ref->ea_size - EXT4_XATTR_SIZE(old_data_size) + EXT4_XATTR_SIZE(new_data_size); /* * This entry may originally lie in inode space or block space, * and it is going to be transferred to another place. */ if (to_block) { xattr_ref->inode_size_rem += EXT4_XATTR_SIZE(old_data_size) + EXT4_XATTR_LEN(item->name_len); xattr_ref->block_size_rem -= EXT4_XATTR_SIZE(new_data_size) + EXT4_XATTR_LEN(item->name_len); item->in_inode = FALSE; } else if (to_inode) { xattr_ref->block_size_rem += EXT4_XATTR_SIZE(old_data_size) + EXT4_XATTR_LEN(item->name_len); xattr_ref->inode_size_rem -= EXT4_XATTR_SIZE(new_data_size) + EXT4_XATTR_LEN(item->name_len); item->in_inode = TRUE; } else { /* * No need to transfer as there is enough space for the entry * to stay in inode space or block space it used to be. */ orig_room_size += EXT4_XATTR_SIZE(old_data_size); orig_room_size -= EXT4_XATTR_SIZE(new_data_size); if (item->in_inode) xattr_ref->inode_size_rem = orig_room_size; else xattr_ref->block_size_rem = orig_room_size; } xattr_ref->dirty = TRUE; return ret; } void ext4_xattr_purge_items(struct ext4_xattr_ref *xattr_ref) { struct rb_node *first_node; struct ext4_xattr_item *item = NULL; first_node = rb_first(&xattr_ref->root); if (first_node) item = container_of(first_node, struct ext4_xattr_item, node); while (item) { struct rb_node *next_node; struct ext4_xattr_item *next_item = NULL; next_node = rb_next(&item->node); if (next_node) next_item = container_of(next_node, struct ext4_xattr_item, node); else next_item = NULL; ext4_xattr_item_remove(xattr_ref, item); ext4_xattr_item_free(item); item = next_item; } xattr_ref->ea_size = 0; if (ext4_xattr_inode_space(xattr_ref) < sizeof(struct ext4_xattr_ibody_header)) xattr_ref->inode_size_rem = 0; else xattr_ref->inode_size_rem = ext4_xattr_inode_space(xattr_ref) - sizeof(struct ext4_xattr_ibody_header); xattr_ref->block_size_rem = ext4_xattr_block_space(xattr_ref) - sizeof(struct ext4_xattr_header); } static int ext4_xattr_try_alloc_block(struct ext4_xattr_ref *xattr_ref) { int ret = 0; ext4_fsblk_t xattr_block = 0; xattr_block = xattr_ref->inode_ref->Inode.i_file_acl; if (!xattr_block) { ext4_fsblk_t goal = ext4_inode_to_goal_block(&xattr_ref->inode_ref->Inode); xattr_block = ext4_new_meta_blocks(xattr_ref->IrpContext, &xattr_ref->inode_ref->Inode, goal, 0, NULL, &ret); if (ret != 0) goto Finish; xattr_ref->block_bh = extents_bwrite(&xattr_ref->fs->sb, xattr_block); if (!xattr_ref->block_bh) { ext4_free_blocks(xattr_ref->IrpContext, &xattr_ref->inode_ref->Inode, xattr_block, 1, 0); ret = -ENOMEM; goto Finish; } xattr_ref->inode_ref->Inode.i_file_acl = xattr_block; xattr_ref->IsOnDiskInodeDirty = TRUE; xattr_ref->block_loaded = TRUE; } Finish: return ret; } static void ext4_xattr_try_free_block(struct ext4_xattr_ref *xattr_ref) { ext4_fsblk_t xattr_block; xattr_block = xattr_ref->inode_ref->Inode.i_file_acl; xattr_ref->inode_ref->Inode.i_file_acl = 0; extents_brelse(xattr_ref->block_bh); xattr_ref->block_bh = NULL; ext4_free_blocks(xattr_ref->IrpContext, &xattr_ref->inode_ref->Inode, xattr_block, 1, 0); xattr_ref->IsOnDiskInodeDirty = TRUE; xattr_ref->block_loaded = FALSE; } static void ext4_xattr_set_block_header(struct ext4_xattr_ref *xattr_ref) { struct ext4_xattr_header *block_header = NULL; block_header = EXT4_XATTR_BHDR(xattr_ref->block_bh); memset(block_header, 0, sizeof(struct ext4_xattr_header)); block_header->h_magic = EXT4_XATTR_MAGIC; block_header->h_refcount = cpu_to_le32(1); block_header->h_blocks = cpu_to_le32(1); } static void ext4_xattr_set_inode_entry(struct ext4_xattr_item *item, struct ext4_xattr_ibody_header *ibody_header, struct ext4_xattr_entry *entry, void *ibody_data_ptr) { entry->e_name_len = (__u8)item->name_len; entry->e_name_index = item->name_index; entry->e_value_offs = cpu_to_le16((char *)ibody_data_ptr - (char *)EXT4_XATTR_IFIRST(ibody_header)); entry->e_value_block = 0; entry->e_value_size = cpu_to_le32(item->data_size); } static void ext4_xattr_set_block_entry(struct ext4_xattr_item *item, struct ext4_xattr_header *block_header, struct ext4_xattr_entry *block_entry, void *block_data_ptr) { block_entry->e_name_len = (__u8)item->name_len; block_entry->e_name_index = item->name_index; block_entry->e_value_offs = cpu_to_le16((char *)block_data_ptr - (char *)block_header); block_entry->e_value_block = 0; block_entry->e_value_size = cpu_to_le32(item->data_size); } static int ext4_xattr_write_to_disk(struct ext4_xattr_ref *xattr_ref) { int ret = 0; BOOL block_modified = FALSE; void *ibody_data = NULL; void *block_data = NULL; size_t inode_size_rem, block_size_rem; struct ext4_xattr_ibody_header *ibody_header = NULL; struct ext4_xattr_header *block_header = NULL; struct ext4_xattr_entry *entry = NULL; struct ext4_xattr_entry *block_entry = NULL; struct ext4_xattr_item *item = NULL; inode_size_rem = ext4_xattr_inode_space(xattr_ref); block_size_rem = ext4_xattr_block_space(xattr_ref); if (inode_size_rem > sizeof(struct ext4_xattr_ibody_header)) { ibody_header = EXT4_XATTR_IHDR(xattr_ref->OnDiskInode); entry = EXT4_XATTR_IFIRST(ibody_header); } if (!xattr_ref->dirty) goto Finish; /* If there are enough spaces in the ibody EA table.*/ if (inode_size_rem > sizeof(struct ext4_xattr_ibody_header)) { memset(ibody_header, 0, inode_size_rem); ibody_header->h_magic = EXT4_XATTR_MAGIC; ibody_data = (char *)ibody_header + inode_size_rem; inode_size_rem -= sizeof(struct ext4_xattr_ibody_header); xattr_ref->IsOnDiskInodeDirty = TRUE; } /* If we need an extra block to hold the EA entries*/ if (xattr_ref->ea_size > inode_size_rem) { if (!xattr_ref->block_loaded) { ret = ext4_xattr_try_alloc_block(xattr_ref); if (ret != 0) goto Finish; } memset(xattr_ref->block_bh->b_data, 0, xattr_ref->fs->BlockSize); block_header = EXT4_XATTR_BHDR(xattr_ref->block_bh); block_entry = EXT4_XATTR_BFIRST(xattr_ref->block_bh); ext4_xattr_set_block_header(xattr_ref); block_data = (char *)block_header + block_size_rem; block_size_rem -= sizeof(struct ext4_xattr_header); extents_mark_buffer_dirty(xattr_ref->block_bh); } else { /* We don't need an extra block.*/ if (xattr_ref->block_loaded) { block_header = EXT4_XATTR_BHDR(xattr_ref->block_bh); le32_add_cpu(&block_header->h_refcount, -1); if (!block_header->h_refcount) { ext4_xattr_try_free_block(xattr_ref); block_header = NULL; } else { block_entry = EXT4_XATTR_BFIRST(xattr_ref->block_bh); block_data = (char *)block_header + block_size_rem; block_size_rem -= sizeof(struct ext4_xattr_header); xattr_ref->inode_ref->Inode.i_file_acl = 0; xattr_ref->IsOnDiskInodeDirty = TRUE; extents_mark_buffer_dirty(xattr_ref->block_bh); } } } list_for_each_entry(item, &xattr_ref->ordered_list, struct ext4_xattr_item, list_node) { if (item->in_inode) { ibody_data = (char *)ibody_data - EXT4_XATTR_SIZE(item->data_size); ext4_xattr_set_inode_entry(item, ibody_header, entry, ibody_data); memcpy(EXT4_XATTR_NAME(entry), item->name, item->name_len); memcpy(ibody_data, item->data, item->data_size); entry = EXT4_XATTR_NEXT(entry); inode_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); xattr_ref->IsOnDiskInodeDirty = TRUE; continue; } if (EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len) > block_size_rem) { ret = -ENOSPC; DbgPrint("ext4_xattr.c: IMPOSSIBLE -ENOSPC AS WE DID INSPECTION!\n"); ASSERT(0); } block_data = (char *)block_data - EXT4_XATTR_SIZE(item->data_size); ext4_xattr_set_block_entry(item, block_header, block_entry, block_data); memcpy(EXT4_XATTR_NAME(block_entry), item->name, item->name_len); memcpy(block_data, item->data, item->data_size); ext4_xattr_compute_hash(block_header, block_entry); block_entry = EXT4_XATTR_NEXT(block_entry); block_size_rem -= EXT4_XATTR_SIZE(item->data_size) + EXT4_XATTR_LEN(item->name_len); block_modified = TRUE; } xattr_ref->dirty = FALSE; if (block_modified) { ext4_xattr_rehash(block_header, EXT4_XATTR_BFIRST(xattr_ref->block_bh)); ext4_xattr_set_block_checksum(xattr_ref->inode_ref, xattr_ref->block_bh->b_blocknr, block_header); extents_mark_buffer_dirty(xattr_ref->block_bh); } Finish: return ret; } void ext4_fs_xattr_iterate(struct ext4_xattr_ref *ref, int (*iter)(struct ext4_xattr_ref *ref, struct ext4_xattr_item *item, BOOL is_last)) { struct ext4_xattr_item *item; if (!ref->iter_from) { struct list_head *first_node; first_node = ref->ordered_list.next; if (first_node && first_node != &ref->ordered_list) { ref->iter_from = list_entry(first_node, struct ext4_xattr_item, list_node); } } item = ref->iter_from; while (item) { struct list_head *next_node; struct ext4_xattr_item *next_item; int ret = EXT4_XATTR_ITERATE_CONT; next_node = item->list_node.next; if (next_node && next_node != &ref->ordered_list) next_item = list_entry(next_node, struct ext4_xattr_item, list_node); else next_item = NULL; if (iter) ret = iter(ref, item, !next_item); if (ret != EXT4_XATTR_ITERATE_CONT) { if (ret == EXT4_XATTR_ITERATE_STOP) ref->iter_from = NULL; break; } item = next_item; } } void ext4_fs_xattr_iterate_reset(struct ext4_xattr_ref *ref) { ref->iter_from = NULL; } int ext4_fs_set_xattr(struct ext4_xattr_ref *ref, __u8 name_index, const char *name, size_t name_len, const void *data, size_t data_size, BOOL replace) { int ret = 0; struct ext4_xattr_item *item = ext4_xattr_lookup_item(ref, name_index, name, name_len); if (replace) { if (!item) { ret = -ENODATA; goto Finish; } if (item->data_size != data_size) ret = ext4_xattr_resize_item(ref, item, data_size); if (ret != 0) { goto Finish; } memcpy(item->data, data, data_size); } else { if (item) { ret = -EEXIST; goto Finish; } item = ext4_xattr_insert_item(ref, name_index, name, name_len, data, data_size, &ret); } Finish: return ret; } int ext4_fs_set_xattr_ordered(struct ext4_xattr_ref *ref, __u8 name_index, const char *name, size_t name_len, const void *data, size_t data_size) { int ret = 0; struct ext4_xattr_item *item = ext4_xattr_lookup_item(ref, name_index, name, name_len); if (item) { ret = -EEXIST; goto Finish; } item = ext4_xattr_insert_item_ordered(ref, name_index, name, name_len, data, data_size, &ret); Finish: return ret; } int ext4_fs_remove_xattr(struct ext4_xattr_ref *ref, __u8 name_index, const char *name, size_t name_len) { return ext4_xattr_remove_item(ref, name_index, name, name_len); } int ext4_fs_get_xattr(struct ext4_xattr_ref *ref, __u8 name_index, const char *name, size_t name_len, void *buf, size_t buf_size, size_t *data_size) { int ret = 0; size_t item_size = 0; struct ext4_xattr_item *item = ext4_xattr_lookup_item(ref, name_index, name, name_len); if (!item) { ret = -ENODATA; goto Finish; } item_size = item->data_size; if (buf_size > item_size) buf_size = item_size; if (buf) memcpy(buf, item->data, buf_size); Finish: if (data_size) *data_size = item_size; return ret; } int ext4_fs_get_xattr_ref(PEXT2_IRP_CONTEXT IrpContext, PEXT2_VCB fs, PEXT2_MCB inode_ref, struct ext4_xattr_ref *ref) { int rc; ext4_fsblk_t xattr_block; xattr_block = inode_ref->Inode.i_file_acl; memset(&ref->root, 0, sizeof(struct rb_root)); ref->ea_size = 0; ref->iter_from = NULL; if (xattr_block) { ref->block_bh = extents_bread(&fs->sb, xattr_block); if (!ref->block_bh) return -EIO; ref->block_loaded = TRUE; } else ref->block_loaded = FALSE; ref->inode_ref = inode_ref; ref->fs = fs; INIT_LIST_HEAD(&ref->ordered_list); ref->OnDiskInode = Ext2AllocateInode(fs); if (!ref->OnDiskInode) { if (xattr_block) { extents_brelse(ref->block_bh); ref->block_bh = NULL; } return -ENOMEM; } if (!Ext2LoadInodeXattr(fs, &inode_ref->Inode, ref->OnDiskInode)) { if (xattr_block) { extents_brelse(ref->block_bh); ref->block_bh = NULL; } Ext2DestroyInode(fs, ref->OnDiskInode); return -EIO; } ref->IsOnDiskInodeDirty = FALSE; if (ext4_xattr_inode_space(ref) < sizeof(struct ext4_xattr_ibody_header) + sizeof(__u32)) ref->inode_size_rem = 0; else { ref->inode_size_rem = ext4_xattr_inode_space(ref) - sizeof(struct ext4_xattr_ibody_header); } ref->block_size_rem = ext4_xattr_block_space(ref) - sizeof(struct ext4_xattr_header) - sizeof(__u32); rc = ext4_xattr_fetch(ref); if (rc != 0) { ext4_xattr_purge_items(ref); if (xattr_block) { extents_brelse(ref->block_bh); ref->block_bh = NULL; } Ext2DestroyInode(fs, ref->OnDiskInode); return rc; } ref->IrpContext = IrpContext; return 0; } int ext4_fs_put_xattr_ref(struct ext4_xattr_ref *ref) { int ret; sector_t orig_file_acl = ref->inode_ref->Inode.i_file_acl; ret = ext4_xattr_write_to_disk(ref); if (ref->IsOnDiskInodeDirty) { ASSERT(ref->fs->InodeSize > EXT4_GOOD_OLD_INODE_SIZE); /* As we may do block allocation in ext4_xattr_write_to_disk */ if (ret) ref->inode_ref->Inode.i_file_acl = orig_file_acl; if (!ret) { ret = Ext2SaveInode(ref->IrpContext, ref->fs, &ref->inode_ref->Inode) ? 0 : -EIO; if (!ret) { ret = Ext2SaveInodeXattr(ref->IrpContext, ref->fs, &ref->inode_ref->Inode, ref->OnDiskInode) ? 0 : -EIO; } } ref->IsOnDiskInodeDirty = FALSE; } if (ref->block_loaded) { if (!ret) extents_brelse(ref->block_bh); else extents_bforget(ref->block_bh); ref->block_bh = NULL; ref->block_loaded = FALSE; } ext4_xattr_purge_items(ref); Ext2DestroyInode(ref->fs, ref->OnDiskInode); ref->OnDiskInode = NULL; ref->inode_ref = NULL; ref->fs = NULL; return ret; } struct xattr_prefix { const char *prefix; __u8 name_index; }; static const struct xattr_prefix prefix_tbl[] = { {"user.", EXT4_XATTR_INDEX_USER}, {"system.posix_acl_access", EXT4_XATTR_INDEX_POSIX_ACL_ACCESS}, {"system.posix_acl_default", EXT4_XATTR_INDEX_POSIX_ACL_DEFAULT}, {"trusted.", EXT4_XATTR_INDEX_TRUSTED}, {"security.", EXT4_XATTR_INDEX_SECURITY}, {"system.", EXT4_XATTR_INDEX_SYSTEM}, {"system.richacl", EXT4_XATTR_INDEX_RICHACL}, {NULL, 0}, }; const char *ext4_extract_xattr_name(const char *full_name, size_t full_name_len, __u8 *name_index, size_t *name_len, BOOL *found) { int i; ASSERT(name_index); ASSERT(found); *found = FALSE; if (!full_name_len) { if (name_len) *name_len = 0; return NULL; } for (i = 0; prefix_tbl[i].prefix; i++) { size_t prefix_len = strlen(prefix_tbl[i].prefix); if (full_name_len >= prefix_len && !memcmp(full_name, prefix_tbl[i].prefix, prefix_len)) { BOOL require_name = prefix_tbl[i].prefix[prefix_len - 1] == '.'; *name_index = prefix_tbl[i].name_index; if (name_len) *name_len = full_name_len - prefix_len; if (!(full_name_len - prefix_len) && require_name) return NULL; *found = TRUE; if (require_name) return full_name + prefix_len; return NULL; } } if (name_len) *name_len = 0; return NULL; } const char *ext4_get_xattr_name_prefix(__u8 name_index, size_t *ret_prefix_len) { int i; for (i = 0; prefix_tbl[i].prefix; i++) { size_t prefix_len = strlen(prefix_tbl[i].prefix); if (prefix_tbl[i].name_index == name_index) { if (ret_prefix_len) *ret_prefix_len = prefix_len; return prefix_tbl[i].prefix; } } if (ret_prefix_len) *ret_prefix_len = 0; return NULL; }