/*- * See the file LICENSE for redistribution information. * * Copyright (c) 1999-2002 * Sleepycat Software. All rights reserved. */ #include "db_config.h" #ifndef lint static const char revid[] = "$Id: db_method.c,v 11.78 2002/07/02 19:26:55 sue Exp $"; #endif /* not lint */ #ifndef NO_SYSTEM_INCLUDES #include #ifdef HAVE_RPC #include #endif #include #endif #include "db_int.h" #include "dbinc/crypto.h" #include "dbinc/db_page.h" #include "dbinc/db_shash.h" #include "dbinc/btree.h" #include "dbinc/hash.h" #include "dbinc/qam.h" #include "dbinc/xa.h" #include "dbinc_auto/xa_ext.h" #include "dbinc/db_shash.h" #include "dbinc/lock.h" #ifdef HAVE_RPC #include "dbinc_auto/db_server.h" #include "dbinc_auto/rpc_client_ext.h" #endif static int __db_get_byteswapped __P((DB *, int *)); static int __db_get_type __P((DB *, DBTYPE *dbtype)); static int __db_init __P((DB *, u_int32_t)); static int __db_key_range __P((DB *, DB_TXN *, DBT *, DB_KEY_RANGE *, u_int32_t)); static int __db_set_alloc __P((DB *, void *(*)(size_t), void *(*)(void *, size_t), void (*)(void *))); static int __db_set_append_recno __P((DB *, int (*)(DB *, DBT *, db_recno_t))); static int __db_set_cachesize __P((DB *, u_int32_t, u_int32_t, int)); static int __db_set_cache_priority __P((DB *, DB_CACHE_PRIORITY)); static int __db_set_dup_compare __P((DB *, int (*)(DB *, const DBT *, const DBT *))); static int __db_set_encrypt __P((DB *, const char *, u_int32_t)); static int __db_set_feedback __P((DB *, void (*)(DB *, int, int))); static int __db_set_flags __P((DB *, u_int32_t)); static int __db_set_pagesize __P((DB *, u_int32_t)); static int __db_set_paniccall __P((DB *, void (*)(DB_ENV *, int))); static void __db_set_errcall __P((DB *, void (*)(const char *, char *))); static void __db_set_errfile __P((DB *, FILE *)); static void __db_set_errpfx __P((DB *, const char *)); static int __db_stat_fail __P((DB *, void *, u_int32_t)); static void __dbh_err __P((DB *, int, const char *, ...)); static void __dbh_errx __P((DB *, const char *, ...)); #ifdef HAVE_RPC static int __dbcl_init __P((DB *, DB_ENV *, u_int32_t)); #endif /* * db_create -- * DB constructor. * * EXTERN: int db_create __P((DB **, DB_ENV *, u_int32_t)); */ int db_create(dbpp, dbenv, flags) DB **dbpp; DB_ENV *dbenv; u_int32_t flags; { DB *dbp; int ret; /* Check for invalid function flags. */ switch (flags) { case 0: break; case DB_XA_CREATE: if (dbenv != NULL) { __db_err(dbenv, "XA applications may not specify an environment to db_create"); return (EINVAL); } /* * If it's an XA database, open it within the XA environment, * taken from the global list of environments. (When the XA * transaction manager called our xa_start() routine the * "current" environment was moved to the start of the list. */ dbenv = TAILQ_FIRST(&DB_GLOBAL(db_envq)); break; default: return (__db_ferr(dbenv, "db_create", 0)); } /* Allocate the DB. */ if ((ret = __os_calloc(dbenv, 1, sizeof(*dbp), &dbp)) != 0) return (ret); #ifdef HAVE_RPC if (dbenv != NULL && RPC_ON(dbenv)) ret = __dbcl_init(dbp, dbenv, flags); else #endif ret = __db_init(dbp, flags); if (ret != 0) { __os_free(dbenv, dbp); return (ret); } /* If we don't have an environment yet, allocate a local one. */ if (dbenv == NULL) { if ((ret = db_env_create(&dbenv, 0)) != 0) { __os_free(dbenv, dbp); return (ret); } F_SET(dbenv, DB_ENV_DBLOCAL); } ++dbenv->db_ref; dbp->dbenv = dbenv; *dbpp = dbp; return (0); } /* * __db_init -- * Initialize a DB structure. */ static int __db_init(dbp, flags) DB *dbp; u_int32_t flags; { int ret; dbp->lid = DB_LOCK_INVALIDID; LOCK_INIT(dbp->handle_lock); TAILQ_INIT(&dbp->free_queue); TAILQ_INIT(&dbp->active_queue); TAILQ_INIT(&dbp->join_queue); LIST_INIT(&dbp->s_secondaries); FLD_SET(dbp->am_ok, DB_OK_BTREE | DB_OK_HASH | DB_OK_QUEUE | DB_OK_RECNO); dbp->associate = __db_associate; dbp->close = __db_close; dbp->cursor = __db_cursor; dbp->del = __db_delete; dbp->err = __dbh_err; dbp->errx = __dbh_errx; dbp->fd = __db_fd; dbp->get = __db_get; dbp->get_byteswapped = __db_get_byteswapped; dbp->get_type = __db_get_type; dbp->join = __db_join; dbp->key_range = __db_key_range; dbp->open = __db_open; dbp->pget = __db_pget; dbp->put = __db_put; dbp->remove = __db_remove; dbp->rename = __db_rename; dbp->truncate = __db_truncate; dbp->set_alloc = __db_set_alloc; dbp->set_append_recno = __db_set_append_recno; dbp->set_cachesize = __db_set_cachesize; dbp->set_cache_priority = __db_set_cache_priority; dbp->set_dup_compare = __db_set_dup_compare; dbp->set_encrypt = __db_set_encrypt; dbp->set_errcall = __db_set_errcall; dbp->set_errfile = __db_set_errfile; dbp->set_errpfx = __db_set_errpfx; dbp->set_feedback = __db_set_feedback; dbp->set_flags = __db_set_flags; dbp->set_lorder = __db_set_lorder; dbp->set_pagesize = __db_set_pagesize; dbp->set_paniccall = __db_set_paniccall; dbp->stat = __db_stat_fail; dbp->sync = __db_sync; dbp->upgrade = __db_upgrade; dbp->verify = __db_verify; /* Access method specific. */ if ((ret = __bam_db_create(dbp)) != 0) return (ret); if ((ret = __ham_db_create(dbp)) != 0) return (ret); if ((ret = __qam_db_create(dbp)) != 0) return (ret); /* * XA specific: must be last, as we replace methods set by the * access methods. */ if (LF_ISSET(DB_XA_CREATE) && (ret = __db_xa_create(dbp)) != 0) return (ret); return (0); } /* * __dbh_am_chk -- * Error if an unreasonable method is called. * * PUBLIC: int __dbh_am_chk __P((DB *, u_int32_t)); */ int __dbh_am_chk(dbp, flags) DB *dbp; u_int32_t flags; { /* * We start out allowing any access methods to be called, and as the * application calls the methods the options become restricted. The * idea is to quit as soon as an illegal method combination is called. */ if ((LF_ISSET(DB_OK_BTREE) && FLD_ISSET(dbp->am_ok, DB_OK_BTREE)) || (LF_ISSET(DB_OK_HASH) && FLD_ISSET(dbp->am_ok, DB_OK_HASH)) || (LF_ISSET(DB_OK_QUEUE) && FLD_ISSET(dbp->am_ok, DB_OK_QUEUE)) || (LF_ISSET(DB_OK_RECNO) && FLD_ISSET(dbp->am_ok, DB_OK_RECNO))) { FLD_CLR(dbp->am_ok, ~flags); return (0); } __db_err(dbp->dbenv, "call implies an access method which is inconsistent with previous calls"); return (EINVAL); } /* * __dbh_err -- * Error message, including the standard error string. */ static void #ifdef __STDC__ __dbh_err(DB *dbp, int error, const char *fmt, ...) #else __dbh_err(dbp, error, fmt, va_alist) DB *dbp; int error; const char *fmt; va_dcl #endif { DB_REAL_ERR(dbp->dbenv, error, 1, 1, fmt); } /* * __dbh_errx -- * Error message. */ static void #ifdef __STDC__ __dbh_errx(DB *dbp, const char *fmt, ...) #else __dbh_errx(dbp, fmt, va_alist) DB *dbp; const char *fmt; va_dcl #endif { DB_REAL_ERR(dbp->dbenv, 0, 0, 1, fmt); } /* * __db_get_byteswapped -- * Return if database requires byte swapping. */ static int __db_get_byteswapped(dbp, isswapped) DB *dbp; int *isswapped; { DB_ILLEGAL_BEFORE_OPEN(dbp, "get_byteswapped"); *isswapped = F_ISSET(dbp, DB_AM_SWAP) ? 1 : 0; return (0); } /* * __db_get_type -- * Return type of underlying database. */ static int __db_get_type(dbp, dbtype) DB *dbp; DBTYPE *dbtype; { DB_ILLEGAL_BEFORE_OPEN(dbp, "get_type"); *dbtype = dbp->type; return (0); } /* * __db_key_range -- * Return proportion of keys above and below given key. */ static int __db_key_range(dbp, txn, key, kr, flags) DB *dbp; DB_TXN *txn; DBT *key; DB_KEY_RANGE *kr; u_int32_t flags; { COMPQUIET(txn, NULL); COMPQUIET(key, NULL); COMPQUIET(kr, NULL); COMPQUIET(flags, 0); DB_ILLEGAL_BEFORE_OPEN(dbp, "key_range"); DB_ILLEGAL_METHOD(dbp, DB_OK_BTREE); return (EINVAL); } /* * __db_set_append_recno -- * Set record number append routine. */ static int __db_set_append_recno(dbp, func) DB *dbp; int (*func) __P((DB *, DBT *, db_recno_t)); { DB_ILLEGAL_AFTER_OPEN(dbp, "set_append_recno"); DB_ILLEGAL_METHOD(dbp, DB_OK_QUEUE | DB_OK_RECNO); dbp->db_append_recno = func; return (0); } /* * __db_set_cachesize -- * Set underlying cache size. */ static int __db_set_cachesize(dbp, cache_gbytes, cache_bytes, ncache) DB *dbp; u_int32_t cache_gbytes, cache_bytes; int ncache; { DB_ILLEGAL_IN_ENV(dbp, "set_cachesize"); DB_ILLEGAL_AFTER_OPEN(dbp, "set_cachesize"); return (dbp->dbenv->set_cachesize( dbp->dbenv, cache_gbytes, cache_bytes, ncache)); } /* * __db_set_cache_priority -- * Set cache priority for pages from this file. */ static int __db_set_cache_priority(dbp, priority) DB *dbp; DB_CACHE_PRIORITY priority; { /* * If an underlying DB_MPOOLFILE exists, call it. Otherwise, save * the information away until DB->open is called. */ if (dbp->mpf == NULL) { dbp->priority = priority; return (0); } return (dbp->mpf->set_priority(dbp->mpf, priority)); } /* * __db_set_dup_compare -- * Set duplicate comparison routine. */ static int __db_set_dup_compare(dbp, func) DB *dbp; int (*func) __P((DB *, const DBT *, const DBT *)); { int ret; DB_ILLEGAL_AFTER_OPEN(dbp, "dup_compare"); DB_ILLEGAL_METHOD(dbp, DB_OK_BTREE | DB_OK_HASH); if ((ret = dbp->set_flags(dbp, DB_DUPSORT)) != 0) return (ret); dbp->dup_compare = func; return (0); } /* * __db_set_encrypt -- * Set database passwd. */ static int __db_set_encrypt(dbp, passwd, flags) DB *dbp; const char *passwd; u_int32_t flags; { DB_CIPHER *db_cipher; int ret; DB_ILLEGAL_IN_ENV(dbp, "set_encrypt"); DB_ILLEGAL_AFTER_OPEN(dbp, "set_encrypt"); if ((ret = dbp->dbenv->set_encrypt(dbp->dbenv, passwd, flags)) != 0) return (ret); /* * In a real env, this gets initialized with the region. In a local * env, we must do it here. */ db_cipher = (DB_CIPHER *)dbp->dbenv->crypto_handle; if (!F_ISSET(db_cipher, CIPHER_ANY) && (ret = db_cipher->init(dbp->dbenv, db_cipher)) != 0) return (ret); return (dbp->set_flags(dbp, DB_ENCRYPT)); } static void __db_set_errcall(dbp, errcall) DB *dbp; void (*errcall) __P((const char *, char *)); { dbp->dbenv->set_errcall(dbp->dbenv, errcall); } static void __db_set_errfile(dbp, errfile) DB *dbp; FILE *errfile; { dbp->dbenv->set_errfile(dbp->dbenv, errfile); } static void __db_set_errpfx(dbp, errpfx) DB *dbp; const char *errpfx; { dbp->dbenv->set_errpfx(dbp->dbenv, errpfx); } static int __db_set_feedback(dbp, feedback) DB *dbp; void (*feedback) __P((DB *, int, int)); { dbp->db_feedback = feedback; return (0); } static int __db_set_flags(dbp, flags) DB *dbp; u_int32_t flags; { int ret; /* * !!! * The hash access method only takes two flags: DB_DUP and DB_DUPSORT. * The Btree access method uses them for the same purposes, and so we * resolve them there. * * The queue access method takes no flags. */ if (LF_ISSET(DB_ENCRYPT)) { if (!CRYPTO_ON(dbp->dbenv)) { __db_err(dbp->dbenv, "Database environment not configured for encryption"); return (EINVAL); } F_SET(dbp, DB_AM_ENCRYPT); F_SET(dbp, DB_AM_CHKSUM); LF_CLR(DB_ENCRYPT); } if (LF_ISSET(DB_CHKSUM_SHA1)) { F_SET(dbp, DB_AM_CHKSUM); LF_CLR(DB_CHKSUM_SHA1); } if ((ret = __bam_set_flags(dbp, &flags)) != 0) return (ret); if ((ret = __ram_set_flags(dbp, &flags)) != 0) return (ret); return (flags == 0 ? 0 : __db_ferr(dbp->dbenv, "DB->set_flags", 0)); } /* * __db_set_lorder -- * Set whether lorder is swapped or not. * * PUBLIC: int __db_set_lorder __P((DB *, int)); */ int __db_set_lorder(dbp, db_lorder) DB *dbp; int db_lorder; { int ret; DB_ILLEGAL_AFTER_OPEN(dbp, "set_lorder"); /* Flag if the specified byte order requires swapping. */ switch (ret = __db_byteorder(dbp->dbenv, db_lorder)) { case 0: F_CLR(dbp, DB_AM_SWAP); break; case DB_SWAPBYTES: F_SET(dbp, DB_AM_SWAP); break; default: return (ret); /* NOTREACHED */ } return (0); } static int __db_set_alloc(dbp, mal_func, real_func, free_func) DB *dbp; void *(*mal_func) __P((size_t)); void *(*real_func) __P((void *, size_t)); void (*free_func) __P((void *)); { DB_ILLEGAL_IN_ENV(dbp, "set_alloc"); DB_ILLEGAL_AFTER_OPEN(dbp, "set_alloc"); return (dbp->dbenv->set_alloc(dbp->dbenv, mal_func, real_func, free_func)); } static int __db_set_pagesize(dbp, db_pagesize) DB *dbp; u_int32_t db_pagesize; { DB_ILLEGAL_AFTER_OPEN(dbp, "set_pagesize"); if (db_pagesize < DB_MIN_PGSIZE) { __db_err(dbp->dbenv, "page sizes may not be smaller than %lu", (u_long)DB_MIN_PGSIZE); return (EINVAL); } if (db_pagesize > DB_MAX_PGSIZE) { __db_err(dbp->dbenv, "page sizes may not be larger than %lu", (u_long)DB_MAX_PGSIZE); return (EINVAL); } /* * We don't want anything that's not a power-of-2, as we rely on that * for alignment of various types on the pages. */ if (!POWER_OF_TWO(db_pagesize)) { __db_err(dbp->dbenv, "page sizes must be a power-of-2"); return (EINVAL); } /* * XXX * Should we be checking for a page size that's not a multiple of 512, * so that we never try and write less than a disk sector? */ dbp->pgsize = db_pagesize; return (0); } static int __db_set_paniccall(dbp, paniccall) DB *dbp; void (*paniccall) __P((DB_ENV *, int)); { return (dbp->dbenv->set_paniccall(dbp->dbenv, paniccall)); } static int __db_stat_fail(dbp, sp, flags) DB *dbp; void *sp; u_int32_t flags; { COMPQUIET(sp, NULL); COMPQUIET(flags, 0); /* * DB->stat isn't initialized until the actual DB->open call, * but we don't want to core dump. */ PANIC_CHECK(dbp->dbenv); DB_ILLEGAL_BEFORE_OPEN(dbp, "DB->stat"); /* NOTREACHED */ return (EINVAL); } #ifdef HAVE_RPC /* * __dbcl_init -- * Initialize a DB structure on the server. */ static int __dbcl_init(dbp, dbenv, flags) DB *dbp; DB_ENV *dbenv; u_int32_t flags; { TAILQ_INIT(&dbp->free_queue); TAILQ_INIT(&dbp->active_queue); /* !!! * Note that we don't need to initialize the join_queue; it's * not used in RPC clients. See the comment in __dbcl_db_join_ret(). */ dbp->associate = __dbcl_db_associate; dbp->close = __dbcl_db_close; dbp->cursor = __dbcl_db_cursor; dbp->del = __dbcl_db_del; dbp->err = __dbh_err; dbp->errx = __dbh_errx; dbp->fd = __dbcl_db_fd; dbp->get = __dbcl_db_get; dbp->get_byteswapped = __db_get_byteswapped; dbp->get_type = __db_get_type; dbp->join = __dbcl_db_join; dbp->key_range = __dbcl_db_key_range; dbp->open = __dbcl_db_open_wrap; dbp->pget = __dbcl_db_pget; dbp->put = __dbcl_db_put; dbp->remove = __dbcl_db_remove; dbp->rename = __dbcl_db_rename; dbp->set_alloc = __dbcl_db_alloc; dbp->set_append_recno = __dbcl_db_set_append_recno; dbp->set_cachesize = __dbcl_db_cachesize; dbp->set_cache_priority = __dbcl_db_cache_priority; dbp->set_dup_compare = __dbcl_db_dup_compare; dbp->set_encrypt = __dbcl_db_encrypt; dbp->set_errcall = __db_set_errcall; dbp->set_errfile = __db_set_errfile; dbp->set_errpfx = __db_set_errpfx; dbp->set_feedback = __dbcl_db_feedback; dbp->set_flags = __dbcl_db_flags; dbp->set_lorder = __dbcl_db_lorder; dbp->set_pagesize = __dbcl_db_pagesize; dbp->set_paniccall = __dbcl_db_panic; dbp->stat = __dbcl_db_stat; dbp->sync = __dbcl_db_sync; dbp->truncate = __dbcl_db_truncate; dbp->upgrade = __dbcl_db_upgrade; dbp->verify = __dbcl_db_verify; /* * Set all the method specific functions to client funcs as well. */ dbp->set_bt_compare = __dbcl_db_bt_compare; dbp->set_bt_maxkey = __dbcl_db_bt_maxkey; dbp->set_bt_minkey = __dbcl_db_bt_minkey; dbp->set_bt_prefix = __dbcl_db_bt_prefix; dbp->set_h_ffactor = __dbcl_db_h_ffactor; dbp->set_h_hash = __dbcl_db_h_hash; dbp->set_h_nelem = __dbcl_db_h_nelem; dbp->set_q_extentsize = __dbcl_db_extentsize; dbp->set_re_delim = __dbcl_db_re_delim; dbp->set_re_len = __dbcl_db_re_len; dbp->set_re_pad = __dbcl_db_re_pad; dbp->set_re_source = __dbcl_db_re_source; return (__dbcl_db_create(dbp, dbenv, flags)); } #endif