X-Git-Url: http://ftp.safe.ca/?a=blobdiff_plain;f=fs%2Fbtrfs%2Finode-map.c;h=c56eb5909172956da6354eef29ea36f7f698a8fc;hb=9bad145ee2ef43ac36d397190a7d8b1a7306d3fb;hp=c45aec258bd57c89e9164cb403466d9b455b138a;hpb=e20d96d64f9cf9288ffecc9ad4714e91c3b97ca8;p=safe%2Fjmp%2Flinux-2.6 diff --git a/fs/btrfs/inode-map.c b/fs/btrfs/inode-map.c index c45aec2..c56eb59 100644 --- a/fs/btrfs/inode-map.c +++ b/fs/btrfs/inode-map.c @@ -1,133 +1,80 @@ -#include +/* + * Copyright (C) 2007 Oracle. All rights reserved. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public + * License v2 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 + * License along with this program; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 021110-1307, USA. + */ + #include "ctree.h" #include "disk-io.h" #include "transaction.h" -/* - * walks the btree of allocated inodes and find a hole. - */ -int btrfs_find_free_objectid(struct btrfs_trans_handle *trans, - struct btrfs_root *fs_root, - u64 dirid, u64 *objectid) +int btrfs_find_highest_inode(struct btrfs_root *root, u64 *objectid) { - struct btrfs_path path; - struct btrfs_key key; + struct btrfs_path *path; int ret; - u64 hole_size = 0; - int slot = 0; - u64 last_ino = 0; - int start_found; - struct btrfs_leaf *l; - struct btrfs_root *root = fs_root->fs_info->inode_root; + struct extent_buffer *l; struct btrfs_key search_key; - u64 search_start = dirid; - - if (fs_root->fs_info->last_inode_alloc_dirid == dirid) - search_start = fs_root->fs_info->last_inode_alloc; + struct btrfs_key found_key; + int slot; - search_key.objectid = search_start; - search_key.flags = 0; - btrfs_set_key_type(&search_key, BTRFS_INODE_MAP_ITEM_KEY); - search_key.offset = 0; + path = btrfs_alloc_path(); + BUG_ON(!path); - btrfs_init_path(&path); - start_found = 0; - ret = btrfs_search_slot(trans, root, &search_key, &path, 0, 0); + search_key.objectid = BTRFS_LAST_FREE_OBJECTID; + search_key.type = -1; + search_key.offset = (u64)-1; + ret = btrfs_search_slot(NULL, root, &search_key, path, 0, 0); if (ret < 0) goto error; - - if (path.slots[0] > 0) - path.slots[0]--; - - while (1) { - l = btrfs_buffer_leaf(path.nodes[0]); - slot = path.slots[0]; - if (slot >= btrfs_header_nritems(&l->header)) { - ret = btrfs_next_leaf(root, &path); - if (ret == 0) - continue; - if (ret < 0) - goto error; - if (!start_found) { - *objectid = search_start; - start_found = 1; - goto found; - } - *objectid = last_ino > search_start ? - last_ino : search_start; - goto found; - } - btrfs_disk_key_to_cpu(&key, &l->items[slot].key); - if (key.objectid >= search_start) { - if (start_found) { - if (last_ino < search_start) - last_ino = search_start; - hole_size = key.objectid - last_ino; - if (hole_size > 0) { - *objectid = last_ino; - goto found; - } - } - } - start_found = 1; - last_ino = key.objectid + 1; - path.slots[0]++; + BUG_ON(ret == 0); + if (path->slots[0] > 0) { + slot = path->slots[0] - 1; + l = path->nodes[0]; + btrfs_item_key_to_cpu(l, &found_key, slot); + *objectid = max_t(u64, found_key.objectid, + BTRFS_FIRST_FREE_OBJECTID - 1); + } else { + *objectid = BTRFS_FIRST_FREE_OBJECTID - 1; } - // FIXME -ENOSPC -found: - root->fs_info->last_inode_alloc = *objectid; - root->fs_info->last_inode_alloc_dirid = dirid; - btrfs_release_path(root, &path); - BUG_ON(*objectid < search_start); - return 0; + ret = 0; error: - btrfs_release_path(root, &path); + btrfs_free_path(path); return ret; } -int btrfs_insert_inode_map(struct btrfs_trans_handle *trans, - struct btrfs_root *fs_root, - u64 objectid, struct btrfs_key *location) +int btrfs_find_free_objectid(struct btrfs_trans_handle *trans, + struct btrfs_root *root, + u64 dirid, u64 *objectid) { - int ret = 0; - struct btrfs_path path; - struct btrfs_inode_map_item *inode_item; - struct btrfs_key key; - struct btrfs_root *inode_root = fs_root->fs_info->inode_root; + int ret; + mutex_lock(&root->objectid_mutex); + + if (unlikely(root->highest_objectid < BTRFS_FIRST_FREE_OBJECTID)) { + ret = btrfs_find_highest_inode(root, &root->highest_objectid); + if (ret) + goto out; + } - key.objectid = objectid; - key.flags = 0; - btrfs_set_key_type(&key, BTRFS_INODE_MAP_ITEM_KEY); - key.offset = 0; - btrfs_init_path(&path); - ret = btrfs_insert_empty_item(trans, inode_root, &path, &key, - sizeof(struct btrfs_inode_map_item)); - if (ret) + if (unlikely(root->highest_objectid >= BTRFS_LAST_FREE_OBJECTID)) { + ret = -ENOSPC; goto out; + } - inode_item = btrfs_item_ptr(btrfs_buffer_leaf(path.nodes[0]), - path.slots[0], struct btrfs_inode_map_item); - btrfs_cpu_key_to_disk(&inode_item->key, location); + *objectid = ++root->highest_objectid; + ret = 0; out: - btrfs_release_path(inode_root, &path); + mutex_unlock(&root->objectid_mutex); return ret; } - -int btrfs_lookup_inode_map(struct btrfs_trans_handle *trans, - struct btrfs_root *fs_root, struct btrfs_path *path, - u64 objectid, int mod) -{ - int ret; - struct btrfs_key key; - int ins_len = mod < 0 ? -1 : 0; - int cow = mod != 0; - struct btrfs_root *inode_root = fs_root->fs_info->inode_root; - - key.objectid = objectid; - key.flags = 0; - key.offset = 0; - btrfs_set_key_type(&key, BTRFS_INODE_MAP_ITEM_KEY); - ret = btrfs_search_slot(trans, inode_root, &key, path, ins_len, cow); - return ret; -} -