1b1fc9befade6b5f916c185d38f113e3af4898c4
[platform/upstream/btrfs-progs.git] / dir-item.c
1 #include <stdio.h>
2 #include <stdlib.h>
3 #include "kerncompat.h"
4 #include "radix-tree.h"
5 #include "ctree.h"
6 #include "disk-io.h"
7 #include "hash.h"
8 #include "transaction.h"
9
10 int btrfs_insert_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root
11                           *root, char *name, int name_len, u64 dir, u64
12                           objectid, u8 type)
13 {
14         int ret = 0;
15         struct btrfs_path path;
16         struct btrfs_dir_item *dir_item;
17         char *name_ptr;
18         struct btrfs_key key;
19         u32 data_size;
20
21         key.objectid = dir;
22         key.flags = 0;
23         btrfs_set_key_type(&key, BTRFS_DIR_ITEM_KEY);
24         if (name_len == 1 && *name == '.')
25                 key.offset = 1;
26         else if (name_len == 2 && name[0] == '.' && name[1] == '.')
27                 key.offset = 2;
28         else
29                 ret = btrfs_name_hash(name, name_len, &key.offset);
30         BUG_ON(ret);
31         btrfs_init_path(&path);
32         data_size = sizeof(*dir_item) + name_len;
33         ret = btrfs_insert_empty_item(trans, root, &path, &key, data_size);
34         if (ret)
35                 goto out;
36
37         dir_item = btrfs_item_ptr(&path.nodes[0]->leaf, path.slots[0],
38                                   struct btrfs_dir_item);
39         btrfs_set_dir_objectid(dir_item, objectid);
40         btrfs_set_dir_type(dir_item, type);
41         btrfs_set_dir_flags(dir_item, 0);
42         btrfs_set_dir_name_len(dir_item, name_len);
43         name_ptr = (char *)(dir_item + 1);
44         memcpy(name_ptr, name, name_len);
45
46         // FIXME don't be stupid
47         if (key.offset == 2)
48                 goto out;
49         btrfs_release_path(root, &path);
50         btrfs_set_key_type(&key, BTRFS_DIR_INDEX_KEY);
51         key.offset = objectid;
52         ret = btrfs_insert_empty_item(trans, root, &path, &key, data_size);
53         if (ret)
54                 goto out;
55
56         dir_item = btrfs_item_ptr(&path.nodes[0]->leaf, path.slots[0],
57                                   struct btrfs_dir_item);
58         btrfs_set_dir_objectid(dir_item, objectid);
59         btrfs_set_dir_type(dir_item, type);
60         btrfs_set_dir_flags(dir_item, 0);
61         btrfs_set_dir_name_len(dir_item, name_len);
62         name_ptr = (char *)(dir_item + 1);
63         memcpy(name_ptr, name, name_len);
64 out:
65         btrfs_release_path(root, &path);
66         return ret;
67 }
68
69 int btrfs_lookup_dir_item(struct btrfs_trans_handle *trans, struct btrfs_root
70                           *root, struct btrfs_path *path, u64 dir, char *name,
71                           int name_len, int mod)
72 {
73         int ret;
74         struct btrfs_key key;
75         int ins_len = mod < 0 ? -1 : 0;
76         int cow = mod != 0;
77
78         key.objectid = dir;
79         key.flags = 0;
80         btrfs_set_key_type(&key, BTRFS_DIR_ITEM_KEY);
81         ret = btrfs_name_hash(name, name_len, &key.offset);
82         BUG_ON(ret);
83         ret = btrfs_search_slot(trans, root, &key, path, ins_len, cow);
84         return ret;
85 }
86
87 int btrfs_match_dir_item_name(struct btrfs_root *root,
88                               struct btrfs_path *path, char
89                               *name, int name_len)
90 {
91         struct btrfs_dir_item *dir_item;
92         char *name_ptr;
93
94         dir_item = btrfs_item_ptr(&path->nodes[0]->leaf, path->slots[0],
95                                   struct btrfs_dir_item);
96         if (btrfs_dir_name_len(dir_item) != name_len)
97                 return 0;
98         name_ptr = (char *)(dir_item + 1);
99         if (memcmp(name_ptr, name, name_len))
100                 return 0;
101         return 1;
102 }