summaryrefslogtreecommitdiff
path: root/src/findkey.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/findkey.c')
-rw-r--r--src/findkey.c145
1 files changed, 145 insertions, 0 deletions
diff --git a/src/findkey.c b/src/findkey.c
new file mode 100644
index 0000000..c2a0653
--- /dev/null
+++ b/src/findkey.c
@@ -0,0 +1,145 @@
+/* findkey.c - The routine that finds a key entry in the file. */
+
+/* This file is part of GDBM, the GNU data base manager.
+ Copyright (C) 1990, 1991, 1993, 2007, 2011, 2013 Free Software Foundation,
+ Inc.
+
+ GDBM is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 3, or (at your option)
+ any later version.
+
+ GDBM 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 GDBM. If not, see <http://www.gnu.org/licenses/>. */
+
+/* Include system configuration before all else. */
+#include "autoconf.h"
+
+#include "gdbmdefs.h"
+
+
+/* Read the data found in bucket entry ELEM_LOC in file DBF and
+ return a pointer to it. Also, cache the read value. */
+
+char *
+_gdbm_read_entry (GDBM_FILE dbf, int elem_loc)
+{
+ int rc;
+ int key_size;
+ int data_size;
+ off_t file_pos;
+ data_cache_elem *data_ca;
+
+ /* Is it already in the cache? */
+ if (dbf->cache_entry->ca_data.elem_loc == elem_loc)
+ return dbf->cache_entry->ca_data.dptr;
+
+ /* Set sizes and pointers. */
+ key_size = dbf->bucket->h_table[elem_loc].key_size;
+ data_size = dbf->bucket->h_table[elem_loc].data_size;
+ data_ca = &dbf->cache_entry->ca_data;
+
+ /* Set up the cache. */
+ if (data_ca->dptr != NULL) free (data_ca->dptr);
+ data_ca->key_size = key_size;
+ data_ca->data_size = data_size;
+ data_ca->elem_loc = elem_loc;
+ data_ca->hash_val = dbf->bucket->h_table[elem_loc].hash_value;
+ if (key_size+data_size == 0)
+ data_ca->dptr = (char *) malloc (1);
+ else
+ data_ca->dptr = (char *) malloc (key_size+data_size);
+ if (data_ca->dptr == NULL) _gdbm_fatal (dbf, _("malloc error"));
+
+
+ /* Read into the cache. */
+ file_pos = __lseek (dbf, dbf->bucket->h_table[elem_loc].data_pointer,
+ SEEK_SET);
+ if (file_pos != dbf->bucket->h_table[elem_loc].data_pointer)
+ _gdbm_fatal (dbf, _("lseek error"));
+ rc = _gdbm_full_read (dbf, data_ca->dptr, key_size+data_size);
+ if (rc)
+ _gdbm_fatal (dbf, gdbm_strerror (rc));
+
+ return data_ca->dptr;
+}
+
+
+
+/* Find the KEY in the file and get ready to read the associated data. The
+ return value is the location in the current hash bucket of the KEY's
+ entry. If it is found, a pointer to the data and the key are returned
+ in DPTR. If it is not found, the value -1 is returned. Since find
+ key computes the hash value of key, that value */
+int
+_gdbm_findkey (GDBM_FILE dbf, datum key, char **dptr, int *new_hash_val)
+{
+ int bucket_hash_val; /* The hash value from the bucket. */
+ char *file_key; /* The complete key as stored in the file. */
+ int elem_loc; /* The location in the bucket. */
+ int home_loc; /* The home location in the bucket. */
+ int key_size; /* Size of the key on the file. */
+
+ /* Compute hash value and load proper bucket. */
+ *new_hash_val = _gdbm_hash (key);
+ _gdbm_get_bucket (dbf, *new_hash_val>> (31-dbf->header->dir_bits));
+
+ /* Is the element the last one found for this bucket? */
+ if (dbf->cache_entry->ca_data.elem_loc != -1
+ && *new_hash_val == dbf->cache_entry->ca_data.hash_val
+ && dbf->cache_entry->ca_data.key_size == key.dsize
+ && dbf->cache_entry->ca_data.dptr != NULL
+ && memcmp (dbf->cache_entry->ca_data.dptr, key.dptr, key.dsize) == 0)
+ {
+ /* This is it. Return the cache pointer. */
+ *dptr = dbf->cache_entry->ca_data.dptr+key.dsize;
+ return dbf->cache_entry->ca_data.elem_loc;
+ }
+
+ /* It is not the cached value, search for element in the bucket. */
+ elem_loc = *new_hash_val % dbf->header->bucket_elems;
+ home_loc = elem_loc;
+ bucket_hash_val = dbf->bucket->h_table[elem_loc].hash_value;
+ while (bucket_hash_val != -1)
+ {
+ key_size = dbf->bucket->h_table[elem_loc].key_size;
+ if (bucket_hash_val != *new_hash_val
+ || key_size != key.dsize
+ || memcmp (dbf->bucket->h_table[elem_loc].key_start, key.dptr,
+ (SMALL < key_size ? SMALL : key_size)) != 0)
+ {
+ /* Current elem_loc is not the item, go to next item. */
+ elem_loc = (elem_loc + 1) % dbf->header->bucket_elems;
+ if (elem_loc == home_loc) return -1;
+ bucket_hash_val = dbf->bucket->h_table[elem_loc].hash_value;
+ }
+ else
+ {
+ /* This may be the one we want.
+ The only way to tell is to read it. */
+ file_key = _gdbm_read_entry (dbf, elem_loc);
+ if (memcmp (file_key, key.dptr, key_size) == 0)
+ {
+ /* This is the item. */
+ *dptr = file_key+key.dsize;
+ return elem_loc;
+ }
+ else
+ {
+ /* Not the item, try the next one. Return if not found. */
+ elem_loc = (elem_loc + 1) % dbf->header->bucket_elems;
+ if (elem_loc == home_loc) return -1;
+ bucket_hash_val = dbf->bucket->h_table[elem_loc].hash_value;
+ }
+ }
+ }
+
+ /* If we get here, we never found the key. */
+ return -1;
+
+}