Better error handling:
authorMartin Pool <mbp@samba.org>
Tue, 4 Dec 2001 07:40:25 +0000 (07:40 +0000)
committerMartin Pool <mbp@samba.org>
Tue, 4 Dec 2001 07:40:25 +0000 (07:40 +0000)
 - tdb_open api changed so that you now pass an error handling
   callback when opening the file, so that errors detected during
   opening have somewhere to go.  (All calls from the body of Samba to
   this function go through a wrapper in tdbutil, which has been
   updated.)

 - Clean up logic for deciding how to open tdb.  Emit log messages if
   something goes wrong (e.g. bad magic.)

 - tdbtool now logs errors to stderr.
(This used to be commit 0aa800618eab1043d802c04fb1d125cd07936769)

source3/tdb/tdb.c
source3/tdb/tdb.h
source3/tdb/tdbtest.c
source3/tdb/tdbtool.c
source3/tdb/tdbtorture.c
source3/tdb/tdbutil.c

index 31733e9ece8a9642fa17518c6215e4125876d906..b18d869f4ec1dfee4b79ef8aff0735401c83ef3a 100644 (file)
@@ -35,6 +35,7 @@
 #include <errno.h>
 #include <sys/mman.h>
 #include <sys/stat.h>
+#include <stdarg.h>
 #include "tdb.h"
 #include "spinlock.h"
 #else
@@ -152,7 +153,9 @@ struct list_struct {
    this functions locks/unlocks 1 byte at the specified offset.
 
    On error, errno is also set so that errors are passed back properly
-   through tdb_open(). */
+   through tdb_open().
+
+   @param probe True to not emit errors if the lock fails. */
 static int tdb_brlock(TDB_CONTEXT *tdb, tdb_off offset, 
                      int rw_type, int lck_type, int probe)
 {
@@ -173,8 +176,8 @@ static int tdb_brlock(TDB_CONTEXT *tdb, tdb_off offset,
 
        if (fcntl(tdb->fd,lck_type,&fl)) {
                if (!probe) {
-                       TDB_LOG((tdb, 5,"tdb_brlock failed at offset %d rw_type=%d lck_type=%d\n", 
-                                offset, rw_type, lck_type));
+                       TDB_LOG((tdb, 5,"tdb_brlock failed at offset %d rw_type=%d lck_type=%d: %s\n", 
+                                offset, rw_type, lck_type, strerror(errno)));
                }
                /* errno set by fcntl */
                return TDB_ERRCODE(TDB_ERR_LOCK, -1);
@@ -1369,6 +1372,33 @@ int tdb_store(TDB_CONTEXT *tdb, TDB_DATA key, TDB_DATA dbuf, int flag)
        return ret;
 }
 
+
+static int tdb_read_header(TDB_CONTEXT *tdb)
+{
+       int readlen = read(tdb->fd, &tdb->header, sizeof(tdb->header));
+       
+       if (readlen == 0) {
+               /* file is just empty, still needs to be created */
+       } else if (readlen == -1) {
+               TDB_LOG((tdb, 0, "tdb_read_header: failed to read tdb header of %s: %s\n",
+                        tdb->name, strerror(errno)));
+       } else if (readlen != sizeof(tdb->header)) {
+               TDB_LOG((tdb, 0, "tdb_read_header: tdb header in %s is truncated at byte %d\n",
+                        tdb->name, readlen));
+       } else if (strcmp(tdb->header.magic_food, TDB_MAGIC_FOOD) != 0) {
+               TDB_LOG((tdb, 0, "tdb_read_header: rotten food in %s\n", tdb->name));
+               errno = EIO;
+       } else if (tdb->header.version != TDB_VERSION
+                  && tdb->header.version != TDB_BYTEREV(TDB_VERSION)) {
+               TDB_LOG((tdb, 0, "tdb_read_header: bad version in header of %s: %#x\n",
+                        tdb->name, tdb->header.version));
+       } else {
+               return 1;
+       }
+       return 0;
+}
+
+
 /* open the database, creating it if necessary 
 
    The open_flags and mode are passed straight to the open call on the
@@ -1378,21 +1408,24 @@ int tdb_store(TDB_CONTEXT *tdb, TDB_DATA key, TDB_DATA dbuf, int flag)
    Return is NULL on error, in which case errno is also set.  Don't 
    try to call tdb_error or tdb_errname, just do strerror(errno).  */
 TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
-                     int open_flags, mode_t mode)
+                     int open_flags, mode_t mode,
+                     tdb_log_func log_func)
 {
-       TDB_CONTEXT tdb, *ret, *i;
+       TDB_CONTEXT tdb[1], *ret, *i;
        struct stat st;
-       int rev = 0, locked;
+       int rev, locked;
 
-       memset(&tdb, 0, sizeof(tdb));
-       tdb.fd = -1;
-       tdb.name = NULL;
-       tdb.map_ptr = NULL;
-       tdb.lockedkeys = NULL;
-       tdb.flags = tdb_flags;
-       tdb.open_flags = open_flags;
+       memset(tdb, 0, sizeof(tdb));
+       tdb->fd = -1;
+       tdb->name = NULL;
+       tdb->map_ptr = NULL;
+       tdb->lockedkeys = NULL;
+       tdb->flags = tdb_flags;
+       tdb->open_flags = open_flags;
+       tdb_logging_function(tdb, log_func);
 
        if ((open_flags & O_ACCMODE) == O_WRONLY) {
+               TDB_LOG((tdb, 0, "tdb_open: tdb cannot be opened O_WRONLY\n"));
                errno = EINVAL;
                goto fail;
        }
@@ -1400,87 +1433,102 @@ TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
        if (hash_size == 0)
                hash_size = DEFAULT_HASH_SIZE;
        if ((open_flags & O_ACCMODE) == O_RDONLY) {
-               tdb.read_only = 1;
+               tdb->read_only = 1;
                /* read only databases don't do locking or clear if first */
-               tdb.flags |= TDB_NOLOCK;
-               tdb.flags &= ~TDB_CLEAR_IF_FIRST;
+               tdb->flags |= TDB_NOLOCK;
+               tdb->flags &= ~TDB_CLEAR_IF_FIRST;
        }
 
        /* internal databases don't mmap or lock, and start off cleared */
-       if (tdb.flags & TDB_INTERNAL) {
-               tdb.flags |= (TDB_NOLOCK | TDB_NOMMAP);
-               tdb.flags &= ~TDB_CLEAR_IF_FIRST;
-               tdb_new_database(&tdb, hash_size);
+       if (tdb->flags & TDB_INTERNAL) {
+               tdb->flags |= (TDB_NOLOCK | TDB_NOMMAP);
+               tdb->flags &= ~TDB_CLEAR_IF_FIRST;
+               tdb_new_database(tdb, hash_size);
                goto internal;
        }
 
-       if ((tdb.fd = open(name, open_flags, mode)) == -1)
+       tdb->name = (char *)strdup(name);
+       if (!tdb->name) {
+               TDB_LOG((tdb, 0, "tdb_open: couldn't allocate memory for name\n"));
+               errno = ENOMEM;
+               goto fail;
+       }
+
+       if ((tdb->fd = open(name, open_flags, mode)) == -1) {
+               TDB_LOG((tdb, 0, "tdb_open: failed to open %s: %s\n",
+                        name, strerror(errno)));
                goto fail;      /* errno set by open(2) */
+       }
 
        /* ensure there is only one process initialising at once */
-       if (tdb_brlock(&tdb, GLOBAL_LOCK, F_WRLCK, F_SETLKW, 0) == -1)
+       if (tdb_brlock(tdb, GLOBAL_LOCK, F_WRLCK, F_SETLKW, 0) == -1)
                goto fail;      /* errno set by tdb_brlock */
 
-       /* we need to zero database if we are the only one with it open */
-       if ((locked = (tdb_brlock(&tdb, ACTIVE_LOCK, F_WRLCK, F_SETLK, 0) == 0))
+       /* We need to zero database if we are the only one with it
+        * open.  Be quiet if the lock fails. */
+       if ((locked = (tdb_brlock(tdb, ACTIVE_LOCK, F_WRLCK, F_SETLK, 1) == 0))
            && (tdb_flags & TDB_CLEAR_IF_FIRST)) {
                open_flags |= O_CREAT;
-               if (ftruncate(tdb.fd, 0) == -1)
+               if (ftruncate(tdb->fd, 0) == -1) {
+                       TDB_LOG((tdb, 0, "tdb_open: couldn't truncate %s: %s\n",
+                                name, strerror(errno)));
                        goto fail; /* errno set by ftruncate */
+               }
        }
-
-       if (read(tdb.fd, &tdb.header, sizeof(tdb.header)) != sizeof(tdb.header)
-           || strcmp(tdb.header.magic_food, TDB_MAGIC_FOOD) != 0
-           || (tdb.header.version != TDB_VERSION
-               && !(rev = (tdb.header.version==TDB_BYTEREV(TDB_VERSION))))) {
+       
+       if (tdb_read_header(tdb)) {
+               rev = tdb->header.version = TDB_BYTEREV(TDB_VERSION);
+       } else {
                /* its not a valid database - possibly initialise it */
-               if (!(open_flags & O_CREAT) || tdb_new_database(&tdb, hash_size) == -1) {
+               if (!(open_flags & O_CREAT)) {
+                       TDB_LOG((tdb, 0, "tdb_open: database header not OK and O_CREAT not specified\n"));
                        errno = EIO; /* ie bad format or something */
                        goto fail;
+               } else if (tdb_new_database(tdb, hash_size) == -1) {
+                       goto fail;
                }
-               rev = (tdb.flags & TDB_CONVERT);
+
+               rev = (tdb->flags & TDB_CONVERT);
        }
+
        if (!rev)
-               tdb.flags &= ~TDB_CONVERT;
+               tdb->flags &= ~TDB_CONVERT;
        else {
-               tdb.flags |= TDB_CONVERT;
-               convert(&tdb.header, sizeof(tdb.header));
+               tdb->flags |= TDB_CONVERT;
+               convert(&tdb->header, sizeof(tdb->header));
        }
-       if (fstat(tdb.fd, &st) == -1)
+       if (fstat(tdb->fd, &st) == -1) {
+               TDB_LOG((tdb, 0, "tdb_open: fstat of %s failed: %s\n",
+                        name, strerror(errno)));
                goto fail;
+       }
 
        /* Is it already in the open list?  If so, fail. */
        for (i = tdbs; i; i = i->next) {
                if (i->device == st.st_dev && i->inode == st.st_ino) {
                        errno = EBUSY;
-                       close(tdb.fd);
+                       close(tdb->fd);
                        return NULL;
                }
        }
 
-       /* map the database and fill in the return structure */
-       tdb.name = (char *)strdup(name);
-       if (!tdb.name) {
-               errno = ENOMEM;
-               goto fail;
-       }
-       tdb.map_size = st.st_size;
-       tdb.device = st.st_dev;
-       tdb.inode = st.st_ino;
-       tdb.locked = calloc(tdb.header.hash_size+1, sizeof(tdb.locked[0]));
-       if (!tdb.locked) {
+       tdb->map_size = st.st_size;
+       tdb->device = st.st_dev;
+       tdb->inode = st.st_ino;
+       tdb->locked = calloc(tdb->header.hash_size+1, sizeof(tdb->locked[0]));
+       if (!tdb->locked) {
                errno = ENOMEM;
                goto fail;
        }
-       tdb_mmap(&tdb);
+       tdb_mmap(tdb);
        if (locked) {
-               if (!tdb.read_only)
-                       tdb_clear_spinlocks(&tdb);
-               if (tdb_brlock(&tdb, ACTIVE_LOCK, F_UNLCK, F_SETLK, 0) == -1)
+               if (!tdb->read_only)
+                       tdb_clear_spinlocks(tdb);
+               if (tdb_brlock(tdb, ACTIVE_LOCK, F_UNLCK, F_SETLK, 0) == -1)
                        goto fail;
        }
        /* leave this lock in place to indicate it's in use */
-       if (tdb_brlock(&tdb, ACTIVE_LOCK, F_RDLCK, F_SETLKW, 0) == -1)
+       if (tdb_brlock(tdb, ACTIVE_LOCK, F_RDLCK, F_SETLKW, 0) == -1)
                goto fail;
 
  internal:
@@ -1488,8 +1536,8 @@ TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
                errno = ENOMEM;
                goto fail;
        }
-       *ret = tdb;
-       if (tdb_brlock(&tdb, GLOBAL_LOCK, F_UNLCK, F_SETLKW, 0) == -1)
+       *ret = *tdb;
+       if (tdb_brlock(tdb, GLOBAL_LOCK, F_UNLCK, F_SETLKW, 0) == -1)
                goto fail;
        ret->next = tdbs;
        tdbs = ret;
@@ -1498,18 +1546,18 @@ TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
  fail:
        { int save_errno = errno;
        
-       if (tdb.map_ptr) {
-               if (tdb.flags & TDB_INTERNAL)
-                       free(tdb.map_ptr);
+       if (tdb->map_ptr) {
+               if (tdb->flags & TDB_INTERNAL)
+                       free(tdb->map_ptr);
                else
-                       tdb_munmap(&tdb);
-       }
-       if (tdb.name)
-               free(tdb.name);
-       if (tdb.fd != -1)
-               close(tdb.fd);
-       if (tdb.locked)
-               free(tdb.locked);
+                       tdb_munmap(tdb);
+       }
+       if (tdb->name)
+               free(tdb->name);
+       if (tdb->fd != -1)
+               close(tdb->fd);
+       if (tdb->locked)
+               free(tdb->locked);
        errno = save_errno;
        return NULL;
        }
@@ -1639,6 +1687,16 @@ void tdb_chainunlock(TDB_CONTEXT *tdb, TDB_DATA key)
 }
 
 
+void tdb_log_to_stderr(TDB_CONTEXT *tdb, int level, const char *format, ...)
+{
+       va_list ap;
+    
+       va_start(ap, format);
+       vfprintf(stderr, format, ap);
+       va_end(ap);
+       fflush(stderr);
+}
+
 /* register a loging function */
 void tdb_logging_function(TDB_CONTEXT *tdb, void (*fn)(TDB_CONTEXT *, int , const char *, ...))
 {
index 9989474a941f0e327ce3e8572084f061e6445b96..991343a67cfb19204517319878e8cd907b5b4075 100644 (file)
@@ -99,12 +99,15 @@ typedef struct tdb_context {
 } TDB_CONTEXT;
 
 typedef int (*tdb_traverse_func)(TDB_CONTEXT *, TDB_DATA, TDB_DATA, void *);
+typedef void (*tdb_log_func)(TDB_CONTEXT *, int , const char *, ...);
 
-TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
-                     int open_flags, mode_t mode);
 int tdb_reopen(TDB_CONTEXT *tdb);
 int tdb_reopen_all(void);
-void tdb_logging_function(TDB_CONTEXT *tdb, void (*fn)(TDB_CONTEXT *, int , const char *, ...));
+void tdb_log_to_stderr(TDB_CONTEXT *tdb, int level, const char *format, ...);
+void tdb_logging_function(TDB_CONTEXT *tdb, tdb_log_func fn);
+TDB_CONTEXT *tdb_open(char *name, int hash_size, int tdb_flags,
+                     int open_flags, mode_t mode,
+                     tdb_log_func log_func);
 enum TDB_ERROR tdb_error(TDB_CONTEXT *tdb);
 const char *tdb_errorstr(TDB_CONTEXT *tdb);
 TDB_DATA tdb_fetch(TDB_CONTEXT *tdb, TDB_DATA key);
index cfc6135a0ae7a776a5af470c14ed65a992866d65..ca4ca5414266f885601944a385a181f8b142b617 100644 (file)
@@ -189,7 +189,6 @@ static int traverse_fn(TDB_CONTEXT *db, TDB_DATA key, TDB_DATA dbuf, void *state
 
 static void merge_test()
 {
-       int klen, dlen;
        int i;
        char keys[5][2];
        TDB_DATA key, data;
@@ -227,7 +226,8 @@ int main(int argc, char *argv[])
        unlink("test.gdbm");
 
        db = tdb_open("test.tdb", 0, TDB_CLEAR_IF_FIRST, 
-                     O_RDWR | O_CREAT | O_TRUNC, 0600);
+                     O_RDWR | O_CREAT | O_TRUNC, 0600,
+                     tdb_log_to_stderr);
        gdbm = gdbm_open("test.gdbm", 512, GDBM_WRITER|GDBM_NEWDB|GDBM_FAST, 
                         0600, NULL);
 
index 31ecd17a0c100ce471e75d6b1133a40d497a6e2f..ecde37a5eab9ff55acc2607590c0012504d80e5f 100644 (file)
@@ -29,6 +29,7 @@
 #include <unistd.h>
 #include <string.h>
 #include <fcntl.h>
+#include <time.h>
 #include <sys/mman.h>
 #include <sys/stat.h>
 #include <sys/time.h>
@@ -174,7 +175,8 @@ static void create_tdb(void)
        }
        if (tdb) tdb_close(tdb);
        tdb = tdb_open(tok, 0, TDB_CLEAR_IF_FIRST,
-                      O_RDWR | O_CREAT | O_TRUNC, 0600);
+                      O_RDWR | O_CREAT | O_TRUNC, 0600,
+                      tdb_log_to_stderr);
        if (!tdb) {
                printf("Could not create %s: %s\n", tok, strerror(errno));
        }
@@ -188,7 +190,7 @@ static void open_tdb(void)
                return;
        }
        if (tdb) tdb_close(tdb);
-       tdb = tdb_open(tok, 0, 0, O_RDWR, 0600);
+       tdb = tdb_open(tok, 0, 0, O_RDWR, 0600, tdb_log_to_stderr);
        if (!tdb) {
                printf("Could not open %s: %s\n", tok, strerror(errno));
        }
@@ -383,6 +385,7 @@ static void next_record(TDB_CONTEXT *tdb, TDB_DATA *pkey)
                print_rec(tdb, *pkey, dbuf, NULL);
 }
 
+
 int main(int argc, char *argv[])
 {
     int bIterate = 0;
index c76446b4fd212a8009270d069e69d8bfa9f027c2..4e12386d737348296b0aa39d79016128605002f0 100644 (file)
@@ -178,7 +178,7 @@ int main(int argc, char *argv[])
        }
 
        db = tdb_open("torture.tdb", 2, TDB_CLEAR_IF_FIRST, 
-                     O_RDWR | O_CREAT, 0600);
+                     O_RDWR | O_CREAT, 0600, tdb_log_to_stderr);
        if (!db) {
                fatal("db open failed");
        }
index 68a47a199d0031fd53d19a5395574447a4af3086..038d98eef99ea46b46708178cc55043a554b56ae 100644 (file)
@@ -355,10 +355,8 @@ TDB_CONTEXT *tdb_open_log(char *name, int hash_size, int tdb_flags,
        if (!lp_use_mmap()) tdb_flags |= TDB_NOMMAP;
 
        tdb = tdb_open(name, hash_size, tdb_flags, 
-                                   open_flags, mode);
+                      open_flags, mode, tdb_log);
        if (!tdb) return NULL;
 
-       tdb_logging_function(tdb, tdb_log);
-
        return tdb;
 }