btree.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:22k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * linux/fs/befs/btree.c
  3.  *
  4.  * Copyright (C) 2001-2002 Will Dyson <will_dyson@pobox.com>
  5.  *
  6.  * Licensed under the GNU GPL. See the file COPYING for details.
  7.  *
  8.  * 2002-02-05: Sergey S. Kostyliov added binary search withing
  9.  *  btree nodes.
  10.  *
  11.  * Many thanks to:
  12.  *
  13.  * Dominic Giampaolo, author of "Practical File System
  14.  * Design with the Be File System", for such a helpful book.
  15.  * 
  16.  * Marcus J. Ranum, author of the b+tree package in 
  17.  * comp.sources.misc volume 10. This code is not copied from that
  18.  * work, but it is partially based on it.
  19.  *
  20.  * Makoto Kato, author of the original BeFS for linux filesystem
  21.  * driver.
  22.  */
  23. #include <linux/kernel.h>
  24. #include <linux/string.h>
  25. #include <linux/slab.h>
  26. #include <linux/mm.h>
  27. #include "befs_fs.h"
  28. #include "endian.h"
  29. /*
  30.  * The btree functions in this file are built on top of the
  31.  * datastream.c interface, which is in turn built on top of the
  32.  * io.c interface.
  33.  */
  34. /* Befs B+tree structure:
  35.  * 
  36.  * The first thing in the tree is the tree superblock. It tells you
  37.  * all kinds of usefull things about the tree, like where the rootnode
  38.  * is located, and the size of the nodes (always 1024 with current version
  39.  * of BeOS).
  40.  *
  41.  * The rest of the tree consists of a series of nodes. Nodes contain a header
  42.  * (struct befs_btree_nodehead), the packed key data, an array of shorts 
  43.  * containing the ending offsets for each of the keys, and an array of
  44.  * befs_off_t values. In interior nodes, the keys are the ending keys for 
  45.  * the childnode they point to, and the values are offsets into the 
  46.  * datastream containing the tree. 
  47.  */
  48. /* Note:
  49.  * 
  50.  * The book states 2 confusing things about befs b+trees. First, 
  51.  * it states that the overflow feild of node headers is used by internal nodes 
  52.  * to point to another node that "effectivly continues this one". Here is what 
  53.  * I belive that means. Each key in internal nodes points to another node that
  54.  * contains key values less than itself. Inspection reveals that the last key 
  55.  * in the internal node is not the last key in the index. Keys that are 
  56.  * greater than the last key in the internal node go into the overflow node. 
  57.  * I imagine there is a performance reason for this.
  58.  *
  59.  * Second, it states that the header of a btree node is sufficient to 
  60.  * distinguish internal nodes from leaf nodes. Without saying exactly how. 
  61.  * After figuring out the first, it becomes obvious that internal nodes have
  62.  * overflow nodes and leafnodes do not.
  63.  */
  64. /* 
  65.  * Currently, this code is only good for directory B+trees.
  66.  * In order to be used for other BFS indexes, it needs to be extended to handle
  67.  * duplicate keys and non-string keytypes (int32, int64, float, double).
  68.  */
  69. /*
  70.  * In memory structure of each btree node
  71.  */
  72. typedef struct {
  73. befs_btree_nodehead head; /* head of node converted to cpu byteorder */
  74. struct buffer_head *bh;
  75. befs_btree_nodehead *od_node; /* on disk node */
  76. } befs_btree_node;
  77. /* local constants */
  78. const static befs_off_t befs_bt_inval = 0xffffffffffffffff;
  79. /* local functions */
  80. static int befs_btree_seekleaf(struct super_block *sb, befs_data_stream * ds,
  81.        befs_btree_super * bt_super,
  82.        befs_btree_node * this_node,
  83.        befs_off_t * node_off);
  84. static int befs_bt_read_super(struct super_block *sb, befs_data_stream * ds,
  85.       befs_btree_super * sup);
  86. static int befs_bt_read_node(struct super_block *sb, befs_data_stream * ds,
  87.      befs_btree_node * node, befs_off_t node_off);
  88. static int befs_leafnode(befs_btree_node * node);
  89. static u16 *befs_bt_keylen_index(befs_btree_node * node);
  90. static befs_off_t *befs_bt_valarray(befs_btree_node * node);
  91. static char *befs_bt_keydata(befs_btree_node * node);
  92. static int befs_find_key(struct super_block *sb, befs_btree_node * node,
  93.  const char *findkey, befs_off_t * value);
  94. static char *befs_bt_get_key(struct super_block *sb, befs_btree_node * node,
  95.      int index, u16 * keylen);
  96. static int befs_compare_strings(const void *key1, int keylen1,
  97. const void *key2, int keylen2);
  98. /**
  99.  * befs_bt_read_super - read in btree superblock convert to cpu byteorder
  100.  * @sb: Filesystem superblock
  101.  * @ds: Datastream to read from
  102.  * @sup: Buffer in which to place the btree superblock
  103.  *
  104.  * Calls befs_read_datastream to read in the btree superblock and
  105.  * makes sure it is in cpu byteorder, byteswapping if nessisary.
  106.  *
  107.  * On success, returns BEFS_OK and *@sup contains the btree superblock,
  108.  * in cpu byte order.
  109.  *
  110.  * On failure, BEFS_ERR is returned.
  111.  */
  112. static int
  113. befs_bt_read_super(struct super_block *sb, befs_data_stream * ds,
  114.    befs_btree_super * sup)
  115. {
  116. struct buffer_head *bh = NULL;
  117. befs_btree_super *od_sup = NULL;
  118. befs_debug(sb, "---> befs_btree_read_super()");
  119. bh = befs_read_datastream(sb, ds, 0, NULL);
  120. if (!bh) {
  121. befs_error(sb, "Couldn't read index header.");
  122. goto error;
  123. }
  124. od_sup = (befs_btree_super *) bh->b_data;
  125. befs_dump_index_entry(sb, od_sup);
  126. sup->magic = fs32_to_cpu(sb, od_sup->magic);
  127. sup->node_size = fs32_to_cpu(sb, od_sup->node_size);
  128. sup->max_depth = fs32_to_cpu(sb, od_sup->max_depth);
  129. sup->data_type = fs32_to_cpu(sb, od_sup->data_type);
  130. sup->root_node_ptr = fs64_to_cpu(sb, od_sup->root_node_ptr);
  131. sup->free_node_ptr = fs64_to_cpu(sb, od_sup->free_node_ptr);
  132. sup->max_size = fs64_to_cpu(sb, od_sup->max_size);
  133. brelse(bh);
  134. if (sup->magic != BEFS_BTREE_MAGIC) {
  135. befs_error(sb, "Index header has bad magic.");
  136. goto error;
  137. }
  138. befs_debug(sb, "<--- befs_btree_read_super()");
  139. return BEFS_OK;
  140.       error:
  141. befs_debug(sb, "<--- befs_btree_read_super() ERROR");
  142. return BEFS_ERR;
  143. }
  144. /**
  145.  * befs_bt_read_node - read in btree node and convert to cpu byteorder
  146.  * @sb: Filesystem superblock
  147.  * @ds: Datastream to read from
  148.  * @node: Buffer in which to place the btree node
  149.  * @node_off: Starting offset (in bytes) of the node in @ds
  150.  *
  151.  * Calls befs_read_datastream to read in the indicated btree node and
  152.  * makes sure its header feilds are in cpu byteorder, byteswapping if 
  153.  * nessisary.
  154.  * Note: node->bh must be NULL when this function called first
  155.  * time. Don't forget brelse(node->bh) after last call.
  156.  *
  157.  * On success, returns BEFS_OK and *@node contains the btree node that
  158.  * starts at @node_off, with the node->head fields in cpu byte order.
  159.  *
  160.  * On failure, BEFS_ERR is returned.
  161.  */
  162. static int
  163. befs_bt_read_node(struct super_block *sb, befs_data_stream * ds,
  164.   befs_btree_node * node, befs_off_t node_off)
  165. {
  166. uint off = 0;
  167. befs_debug(sb, "---> befs_bt_read_node()");
  168. if (node->bh)
  169. brelse(node->bh);
  170. node->bh = befs_read_datastream(sb, ds, node_off, &off);
  171. if (!node->bh) {
  172. befs_error(sb, "befs_bt_read_node() failed to read "
  173.    "node at %Lu", node_off);
  174. befs_debug(sb, "<--- befs_bt_read_node() ERROR");
  175. return BEFS_ERR;
  176. }
  177. node->od_node =
  178.     (befs_btree_nodehead *) ((void *) node->bh->b_data + off);
  179. befs_dump_index_node(sb, node->od_node);
  180. node->head.left = fs64_to_cpu(sb, node->od_node->left);
  181. node->head.right = fs64_to_cpu(sb, node->od_node->right);
  182. node->head.overflow = fs64_to_cpu(sb, node->od_node->overflow);
  183. node->head.all_key_count =
  184.     fs16_to_cpu(sb, node->od_node->all_key_count);
  185. node->head.all_key_length =
  186.     fs16_to_cpu(sb, node->od_node->all_key_length);
  187. befs_debug(sb, "<--- befs_btree_read_node()");
  188. return BEFS_OK;
  189. }
  190. /**
  191.  * befs_btree_find - Find a key in a befs B+tree
  192.  * @sb: Filesystem superblock
  193.  * @ds: Datastream containing btree
  194.  * @key: Key string to lookup in btree
  195.  * @value: Value stored with @key
  196.  *
  197.  * On sucess, returns BEFS_OK and sets *@value to the value stored
  198.  * with @key (usually the disk block number of an inode).
  199.  *
  200.  * On failure, returns BEFS_ERR or BEFS_BT_NOT_FOUND.
  201.  * 
  202.  * Algorithm: 
  203.  *   Read the superblock and rootnode of the b+tree.
  204.  *   Drill down through the interior nodes using befs_find_key().
  205.  *   Once at the correct leaf node, use befs_find_key() again to get the
  206.  *   actuall value stored with the key.
  207.  */
  208. int
  209. befs_btree_find(struct super_block *sb, befs_data_stream * ds,
  210. const char *key, befs_off_t * value)
  211. {
  212. befs_btree_node *this_node = NULL;
  213. befs_btree_super bt_super;
  214. befs_off_t node_off;
  215. int res;
  216. befs_debug(sb, "---> befs_btree_find() Key: %s", key);
  217. if (befs_bt_read_super(sb, ds, &bt_super) != BEFS_OK) {
  218. befs_error(sb,
  219.    "befs_btree_find() failed to read index superblock");
  220. goto error;
  221. }
  222. this_node = (befs_btree_node *) kmalloc(sizeof (befs_btree_node),
  223. GFP_NOFS);
  224. if (!this_node) {
  225. befs_error(sb, "befs_btree_find() failed to allocate %u "
  226.    "bytes of memory", sizeof (befs_btree_node));
  227. goto error;
  228. }
  229. this_node->bh = NULL;
  230. /* read in root node */
  231. node_off = bt_super.root_node_ptr;
  232. if (befs_bt_read_node(sb, ds, this_node, node_off) != BEFS_OK) {
  233. befs_error(sb, "befs_btree_find() failed to read "
  234.    "node at %Lu", node_off);
  235. goto error_alloc;
  236. }
  237. while (!befs_leafnode(this_node)) {
  238. res = befs_find_key(sb, this_node, key, &node_off);
  239. if (res == BEFS_BT_NOT_FOUND)
  240. node_off = this_node->head.overflow;
  241. /* if no match, go to overflow node */
  242. if (befs_bt_read_node(sb, ds, this_node, node_off) != BEFS_OK) {
  243. befs_error(sb, "befs_btree_find() failed to read "
  244.    "node at %Lu", node_off);
  245. goto error_alloc;
  246. }
  247. }
  248. /* at the correct leaf node now */
  249. res = befs_find_key(sb, this_node, key, value);
  250. brelse(this_node->bh);
  251. kfree(this_node);
  252. if (res != BEFS_BT_MATCH) {
  253. befs_debug(sb, "<--- befs_btree_find() Key %s not found", key);
  254. *value = 0;
  255. return BEFS_BT_NOT_FOUND;
  256. }
  257. befs_debug(sb, "<--- befs_btree_find() Found key %s, value %Lu",
  258.    key, *value);
  259. return BEFS_OK;
  260.       error_alloc:
  261. kfree(this_node);
  262.       error:
  263. *value = 0;
  264. befs_debug(sb, "<--- befs_btree_find() ERROR");
  265. return BEFS_ERR;
  266. }
  267. /**
  268.  * befs_find_key - Search for a key within a node
  269.  * @sb: Filesystem superblock
  270.  * @node: Node to find the key within
  271.  * @key: Keystring to search for
  272.  * @value: If key is found, the value stored with the key is put here
  273.  *
  274.  * finds exact match if one exists, and returns BEFS_BT_MATCH
  275.  * If no exact match, finds first key in node that is greater
  276.  * (alpabeticly) than the search key and returns BEFS_BT_PARMATCH
  277.  * (for partial match, I guess). Can you think of something better to
  278.  * call it?
  279.  *
  280.  * If no key was a match or greater than the search key, return
  281.  * BEFS_BT_NOT_FOUND.
  282.  *
  283.  * Use binary search instead of a linear.
  284.  */
  285. static int
  286. befs_find_key(struct super_block *sb, befs_btree_node * node,
  287.       const char *findkey, befs_off_t * value)
  288. {
  289. int first, last, mid;
  290. int eq;
  291. u16 keylen;
  292. int findkey_len;
  293. char *thiskey;
  294. befs_off_t *valarray;
  295. befs_debug(sb, "---> befs_find_key() %s", findkey);
  296. *value = 0;
  297. findkey_len = strlen(findkey);
  298. /* if node can not contain key, just skeep this node */
  299. last = node->head.all_key_count - 1;
  300. thiskey = befs_bt_get_key(sb, node, last, &keylen);
  301. eq = befs_compare_strings(thiskey, keylen, findkey, findkey_len);
  302. if (eq < 0) {
  303. befs_debug(sb, "<--- befs_find_key() %s not found", findkey);
  304. return BEFS_BT_NOT_FOUND;
  305. }
  306. valarray = befs_bt_valarray(node);
  307. /* simple binary search */
  308. first = 0;
  309. mid = 0;
  310. while (last >= first) {
  311. mid = (last + first) / 2;
  312. befs_debug(sb, "first: %d, last: %d, mid: %d", first, last,
  313.    mid);
  314. thiskey = befs_bt_get_key(sb, node, mid, &keylen);
  315. eq = befs_compare_strings(thiskey, keylen, findkey,
  316.   findkey_len);
  317. *value = fs64_to_cpu(sb, valarray[mid]);
  318. if (eq == 0) {
  319. befs_debug(sb, "<--- befs_find_key() found %s at %d",
  320.    thiskey, mid);
  321. return BEFS_BT_MATCH;
  322. }
  323. if (eq > 0)
  324. last = mid - 1;
  325. else
  326. first = mid + 1;
  327. }
  328. if (eq < 0)
  329. *value = fs64_to_cpu(sb, valarray[mid + 1]);
  330. befs_debug(sb, "<--- befs_find_key() found %s at %d", thiskey, mid);
  331. return BEFS_BT_PARMATCH;
  332. }
  333. /**
  334.  * befs_btree_read - Traverse leafnodes of a btree
  335.  * @sb: Filesystem superblock
  336.  * @ds: Datastream containing btree
  337.  * @key_no: Key number (alphabetical order) of key to read
  338.  * @bufsize: Size of the buffer to return key in
  339.  * @keybuf: Pointer to a buffer to put the key in
  340.  * @keysize: Length of the returned key
  341.  * @value: Value stored with the returned key
  342.  *
  343.  * Heres how it works: Key_no is the index of the key/value pair to 
  344.  * retun in keybuf/value.
  345.  * Bufsize is the size of keybuf (BEFS_NAME_LEN+1 is a good size). Keysize is 
  346.  * the number of charecters in the key (just a convience).
  347.  *
  348.  * Algorithm:
  349.  *   Get the first leafnode of the tree. See if the requested key is in that
  350.  *   node. If not, follow the node->right link to the next leafnode. Repeat 
  351.  *   until the (key_no)th key is found or the tree is out of keys.
  352.  */
  353. int
  354. befs_btree_read(struct super_block *sb, befs_data_stream * ds,
  355. loff_t key_no, size_t bufsize, char *keybuf, size_t * keysize,
  356. befs_off_t * value)
  357. {
  358. befs_btree_node *this_node;
  359. befs_btree_super bt_super;
  360. befs_off_t node_off = 0;
  361. int cur_key;
  362. befs_off_t *valarray;
  363. char *keystart;
  364. u16 keylen;
  365. int res;
  366. uint key_sum = 0;
  367. befs_debug(sb, "---> befs_btree_read()");
  368. if (befs_bt_read_super(sb, ds, &bt_super) != BEFS_OK) {
  369. befs_error(sb,
  370.    "befs_btree_read() failed to read index superblock");
  371. goto error;
  372. }
  373. if ((this_node = (befs_btree_node *)
  374.      kmalloc(sizeof (befs_btree_node), GFP_NOFS)) == NULL) {
  375. befs_error(sb, "befs_btree_read() failed to allocate %u "
  376.    "bytes of memory", sizeof (befs_btree_node));
  377. goto error;
  378. }
  379. node_off = bt_super.root_node_ptr;
  380. this_node->bh = NULL;
  381. /* seeks down to first leafnode, reads it into this_node */
  382. res = befs_btree_seekleaf(sb, ds, &bt_super, this_node, &node_off);
  383. if (res == BEFS_BT_EMPTY) {
  384. brelse(this_node->bh);
  385. kfree(this_node);
  386. *value = 0;
  387. *keysize = 0;
  388. befs_debug(sb, "<--- befs_btree_read() Tree is EMPTY");
  389. return BEFS_BT_EMPTY;
  390. } else if (res == BEFS_ERR) {
  391. goto error_alloc;
  392. }
  393. /* find the leaf node containing the key_no key */
  394. while (key_sum + this_node->head.all_key_count <= key_no) {
  395. /* no more nodes to look in: key_no is too large */
  396. if (this_node->head.right == befs_bt_inval) {
  397. *keysize = 0;
  398. *value = 0;
  399. befs_debug(sb,
  400.    "<--- befs_btree_read() END of keys at %Lu",
  401.    key_sum + this_node->head.all_key_count);
  402. brelse(this_node->bh);
  403. kfree(this_node);
  404. return BEFS_BT_END;
  405. }
  406. key_sum += this_node->head.all_key_count;
  407. node_off = this_node->head.right;
  408. if (befs_bt_read_node(sb, ds, this_node, node_off) != BEFS_OK) {
  409. befs_error(sb, "befs_btree_read() failed to read "
  410.    "node at %Lu", node_off);
  411. goto error_alloc;
  412. }
  413. }
  414. /* how many keys into this_node is key_no */
  415. cur_key = key_no - key_sum;
  416. /* get pointers to datastructures within the node body */
  417. valarray = befs_bt_valarray(this_node);
  418. keystart = befs_bt_get_key(sb, this_node, cur_key, &keylen);
  419. befs_debug(sb, "Read [%Lu,%d]: keysize %d", node_off, cur_key, keylen);
  420. if (bufsize < keylen + 1) {
  421. befs_error(sb, "befs_btree_read() keybuf too small (%u) "
  422.    "for key of size %d", bufsize, keylen);
  423. brelse(this_node->bh);
  424. goto error_alloc;
  425. };
  426. strncpy(keybuf, keystart, keylen);
  427. *value = fs64_to_cpu(sb, valarray[cur_key]);
  428. *keysize = keylen;
  429. keybuf[keylen] = '';
  430. befs_debug(sb, "Read [%Lu,%d]: Key "%.*s", Value %Lu", node_off,
  431.    cur_key, keylen, keybuf, *value);
  432. brelse(this_node->bh);
  433. kfree(this_node);
  434. befs_debug(sb, "<--- befs_btree_read()");
  435. return BEFS_OK;
  436.       error_alloc:
  437. kfree(this_node);
  438.       error:
  439. *keysize = 0;
  440. *value = 0;
  441. befs_debug(sb, "<--- befs_btree_read() ERROR");
  442. return BEFS_ERR;
  443. }
  444. /**
  445.  * befs_btree_seekleaf - Find the first leafnode in the btree
  446.  * @sb: Filesystem superblock
  447.  * @ds: Datastream containing btree
  448.  * @bt_super: Pointer to the uperblock of the btree
  449.  * @this_node: Buffer to return the leafnode in
  450.  * @node_off: Pointer to offset of current node within datastream. Modified
  451.  *  by the function.
  452.  *
  453.  *
  454.  * Helper function for btree traverse. Moves the current position to the 
  455.  * start of the first leaf node.
  456.  *
  457.  * Also checks for an empty tree. If there are no keys, returns BEFS_BT_EMPTY.
  458.  */
  459. static int
  460. befs_btree_seekleaf(struct super_block *sb, befs_data_stream * ds,
  461.     befs_btree_super * bt_super, befs_btree_node * this_node,
  462.     befs_off_t * node_off)
  463. {
  464. befs_debug(sb, "---> befs_btree_seekleaf()");
  465. if (befs_bt_read_node(sb, ds, this_node, *node_off) != BEFS_OK) {
  466. befs_error(sb, "befs_btree_seekleaf() failed to read "
  467.    "node at %Lu", *node_off);
  468. goto error;
  469. }
  470. befs_debug(sb, "Seekleaf to root node %Lu", *node_off);
  471. if (this_node->head.all_key_count == 0 && befs_leafnode(this_node)) {
  472. befs_debug(sb, "<--- befs_btree_seekleaf() Tree is EMPTY");
  473. return BEFS_BT_EMPTY;
  474. }
  475. while (!befs_leafnode(this_node)) {
  476. if (this_node->head.all_key_count == 0) {
  477. befs_debug(sb, "befs_btree_seekleaf() encountered "
  478.    "an empty interior node: %Lu. Using Overflow "
  479.    "node: %Lu", *node_off,
  480.    this_node->head.overflow);
  481. *node_off = this_node->head.overflow;
  482. } else {
  483. befs_off_t *valarray = befs_bt_valarray(this_node);
  484. *node_off = fs64_to_cpu(sb, valarray[0]);
  485. }
  486. if (befs_bt_read_node(sb, ds, this_node, *node_off) != BEFS_OK) {
  487. befs_error(sb, "befs_btree_seekleaf() failed to read "
  488.    "node at %Lu", *node_off);
  489. goto error;
  490. }
  491. befs_debug(sb, "Seekleaf to child node %Lu", *node_off);
  492. }
  493. befs_debug(sb, "Node %Lu is a leaf node", *node_off);
  494. return BEFS_OK;
  495.       error:
  496. befs_debug(sb, "<--- befs_btree_seekleaf() ERROR");
  497. return BEFS_ERR;
  498. }
  499. /**
  500.  * befs_leafnode - Determine if the btree node is a leaf node or an 
  501.  * interior node
  502.  * @node: Pointer to node structure to test
  503.  * 
  504.  * Return 1 if leaf, 0 if interior
  505.  */
  506. static int
  507. befs_leafnode(befs_btree_node * node)
  508. {
  509. /* all interior nodes (and only interior nodes) have an overflow node */
  510. if (node->head.overflow == befs_bt_inval)
  511. return 1;
  512. else
  513. return 0;
  514. }
  515. /**
  516.  * befs_bt_keylen_index - Finds start of keylen index in a node
  517.  * @node: Pointer to the node structure to find the keylen index within
  518.  *
  519.  * Returns a pointer to the start of the key length index array
  520.  * of the B+tree node *@node
  521.  *
  522.  * "The length of all the keys in the node is added to the size of the
  523.  * header and then rounded up to a multiple of four to get the begining
  524.  * of the key length index" (p.88, practical filesystem design).
  525.  *
  526.  * Exept that rounding up to 8 works, and rounding up to 4 doesn't.
  527.  */
  528. static u16 *
  529. befs_bt_keylen_index(befs_btree_node * node)
  530. {
  531. const int keylen_align = 8;
  532. unsigned long int off =
  533.     (sizeof (befs_btree_nodehead) + node->head.all_key_length);
  534. ulong tmp = off % keylen_align;
  535. if (tmp)
  536. off += keylen_align - tmp;
  537. return (u16 *) ((void *) node->od_node + off);
  538. }
  539. /**
  540.  * befs_bt_valarray - Finds the start of value array in a node
  541.  * @node: Pointer to the node structure to find the value array within
  542.  *
  543.  * Returns a pointer to the start of the value array
  544.  * of the node pointed to by the node header
  545.  */
  546. static befs_off_t *
  547. befs_bt_valarray(befs_btree_node * node)
  548. {
  549. void *keylen_index_start = (void *) befs_bt_keylen_index(node);
  550. size_t keylen_index_size = node->head.all_key_count * sizeof (u16);
  551. return (befs_off_t *) (keylen_index_start + keylen_index_size);
  552. }
  553. /**
  554.  * befs_bt_keydata - Finds start of keydata array in a node
  555.  * @node: Pointer to the node structure to find the keydata array within
  556.  *
  557.  * Returns a pointer to the start of the keydata array
  558.  * of the node pointed to by the node header 
  559.  */
  560. static char *
  561. befs_bt_keydata(befs_btree_node * node)
  562. {
  563. return (char *) ((void *) node->od_node + sizeof (befs_btree_nodehead));
  564. }
  565. /**
  566.  * befs_bt_get_key - returns a pointer to the start of a key
  567.  * @sb: filesystem superblock
  568.  * @node: node in which to look for the key
  569.  * @index: the index of the key to get
  570.  * @keylen: modified to be the length of the key at @index
  571.  *
  572.  * Returns a valid pointer into @node on success.
  573.  * Returns NULL on failure (bad input) and sets *@keylen = 0
  574.  */
  575. static char *
  576. befs_bt_get_key(struct super_block *sb, befs_btree_node * node,
  577. int index, u16 * keylen)
  578. {
  579. int prev_key_end;
  580. char *keystart;
  581. u16 *keylen_index;
  582. if (index < 0 || index > node->head.all_key_count) {
  583. *keylen = 0;
  584. return NULL;
  585. }
  586. keystart = befs_bt_keydata(node);
  587. keylen_index = befs_bt_keylen_index(node);
  588. if (index == 0)
  589. prev_key_end = 0;
  590. else
  591. prev_key_end = fs16_to_cpu(sb, keylen_index[index - 1]);
  592. *keylen = fs16_to_cpu(sb, keylen_index[index]) - prev_key_end;
  593. return keystart + prev_key_end;
  594. }
  595. /**
  596.  * befs_compare_strings - compare two strings
  597.  * @key1: pointer to the first key to be compared 
  598.  * @keylen1: length in bytes of key1
  599.  * @key2: pointer to the second key to be compared
  600.  * @kelen2: lenght in bytes of key2
  601.  *
  602.  * Returns 0 if @key1 and @key2 are equal.
  603.  * Returns >0 if @key1 is greater.
  604.  * Returns <0 if @key2 is greater..
  605.  */
  606. static int
  607. befs_compare_strings(const void *key1, int keylen1,
  608.      const void *key2, int keylen2)
  609. {
  610. int len = min_t(int, keylen1, keylen2);
  611. int result = strncmp(key1, key2, len);
  612. if (result == 0)
  613. result = keylen1 - keylen2;
  614. return result;
  615. }
  616. /* These will be used for non-string keyed btrees */
  617. #if 0
  618. static int
  619. btree_compare_int32(cont void *key1, int keylen1, const void *key2, int keylen2)
  620. {
  621. return *(int32_t *) key1 - *(int32_t *) key2;
  622. }
  623. static int
  624. btree_compare_uint32(cont void *key1, int keylen1,
  625.      const void *key2, int keylen2)
  626. {
  627. if (*(u_int32_t *) key1 == *(u_int32_t *) key2)
  628. return 0;
  629. else if (*(u_int32_t *) key1 > *(u_int32_t *) key2)
  630. return 1;
  631. return -1;
  632. }
  633. static int
  634. btree_compare_int64(cont void *key1, int keylen1, const void *key2, int keylen2)
  635. {
  636. if (*(int64_t *) key1 == *(int64_t *) key2)
  637. return 0;
  638. else if (*(int64_t *) key1 > *(int64_t *) key2)
  639. return 1;
  640. return -1;
  641. }
  642. static int
  643. btree_compare_uint64(cont void *key1, int keylen1,
  644.      const void *key2, int keylen2)
  645. {
  646. if (*(u_int64_t *) key1 == *(u_int64_t *) key2)
  647. return 0;
  648. else if (*(u_int64_t *) key1 > *(u_int64_t *) key2)
  649. return 1;
  650. return -1;
  651. }
  652. static int
  653. btree_compare_float(cont void *key1, int keylen1, const void *key2, int keylen2)
  654. {
  655. float result = *(float *) key1 - *(float *) key2;
  656. if (result == 0.0f)
  657. return 0;
  658. return (result < 0.0f) ? -1 : 1;
  659. }
  660. static int
  661. btree_compare_double(cont void *key1, int keylen1,
  662.      const void *key2, int keylen2)
  663. {
  664. double result = *(double *) key1 - *(double *) key2;
  665. if (result == 0.0)
  666. return 0;
  667. return (result < 0.0) ? -1 : 1;
  668. }
  669. #endif //0