db_method.c
上传用户:romrleung
上传日期:2022-05-23
资源大小:18897k
文件大小:16k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1999-2002
  5.  * Sleepycat Software.  All rights reserved.
  6.  */
  7. #include "db_config.h"
  8. #ifndef lint
  9. static const char revid[] = "$Id: db_method.c,v 11.78 2002/07/02 19:26:55 sue Exp $";
  10. #endif /* not lint */
  11. #ifndef NO_SYSTEM_INCLUDES
  12. #include <sys/types.h>
  13. #ifdef HAVE_RPC
  14. #include <rpc/rpc.h>
  15. #endif
  16. #include <string.h>
  17. #endif
  18. #include "db_int.h"
  19. #include "dbinc/crypto.h"
  20. #include "dbinc/db_page.h"
  21. #include "dbinc/db_shash.h"
  22. #include "dbinc/btree.h"
  23. #include "dbinc/hash.h"
  24. #include "dbinc/qam.h"
  25. #include "dbinc/xa.h"
  26. #include "dbinc_auto/xa_ext.h"
  27. #include "dbinc/db_shash.h"
  28. #include "dbinc/lock.h"
  29. #ifdef HAVE_RPC
  30. #include "dbinc_auto/db_server.h"
  31. #include "dbinc_auto/rpc_client_ext.h"
  32. #endif
  33. static int  __db_get_byteswapped __P((DB *, int *));
  34. static int  __db_get_type __P((DB *, DBTYPE *dbtype));
  35. static int  __db_init __P((DB *, u_int32_t));
  36. static int  __db_key_range
  37. __P((DB *, DB_TXN *, DBT *, DB_KEY_RANGE *, u_int32_t));
  38. static int  __db_set_alloc __P((DB *, void *(*)(size_t),
  39. void *(*)(void *, size_t), void (*)(void *)));
  40. static int  __db_set_append_recno __P((DB *, int (*)(DB *, DBT *, db_recno_t)));
  41. static int  __db_set_cachesize __P((DB *, u_int32_t, u_int32_t, int));
  42. static int  __db_set_cache_priority __P((DB *, DB_CACHE_PRIORITY));
  43. static int  __db_set_dup_compare
  44. __P((DB *, int (*)(DB *, const DBT *, const DBT *)));
  45. static int  __db_set_encrypt __P((DB *, const char *, u_int32_t));
  46. static int  __db_set_feedback __P((DB *, void (*)(DB *, int, int)));
  47. static int  __db_set_flags __P((DB *, u_int32_t));
  48. static int  __db_set_pagesize __P((DB *, u_int32_t));
  49. static int  __db_set_paniccall __P((DB *, void (*)(DB_ENV *, int)));
  50. static void __db_set_errcall __P((DB *, void (*)(const char *, char *)));
  51. static void __db_set_errfile __P((DB *, FILE *));
  52. static void __db_set_errpfx __P((DB *, const char *));
  53. static int  __db_stat_fail __P((DB *, void *, u_int32_t));
  54. static void __dbh_err __P((DB *, int, const char *, ...));
  55. static void __dbh_errx __P((DB *, const char *, ...));
  56. #ifdef HAVE_RPC
  57. static int  __dbcl_init __P((DB *, DB_ENV *, u_int32_t));
  58. #endif
  59. /*
  60.  * db_create --
  61.  * DB constructor.
  62.  *
  63.  * EXTERN: int db_create __P((DB **, DB_ENV *, u_int32_t));
  64.  */
  65. int
  66. db_create(dbpp, dbenv, flags)
  67. DB **dbpp;
  68. DB_ENV *dbenv;
  69. u_int32_t flags;
  70. {
  71. DB *dbp;
  72. int ret;
  73. /* Check for invalid function flags. */
  74. switch (flags) {
  75. case 0:
  76. break;
  77. case DB_XA_CREATE:
  78. if (dbenv != NULL) {
  79. __db_err(dbenv,
  80. "XA applications may not specify an environment to db_create");
  81. return (EINVAL);
  82. }
  83. /*
  84.  * If it's an XA database, open it within the XA environment,
  85.  * taken from the global list of environments.  (When the XA
  86.  * transaction manager called our xa_start() routine the
  87.  * "current" environment was moved to the start of the list.
  88.  */
  89. dbenv = TAILQ_FIRST(&DB_GLOBAL(db_envq));
  90. break;
  91. default:
  92. return (__db_ferr(dbenv, "db_create", 0));
  93. }
  94. /* Allocate the DB. */
  95. if ((ret = __os_calloc(dbenv, 1, sizeof(*dbp), &dbp)) != 0)
  96. return (ret);
  97. #ifdef HAVE_RPC
  98. if (dbenv != NULL && RPC_ON(dbenv))
  99. ret = __dbcl_init(dbp, dbenv, flags);
  100. else
  101. #endif
  102. ret = __db_init(dbp, flags);
  103. if (ret != 0) {
  104. __os_free(dbenv, dbp);
  105. return (ret);
  106. }
  107. /* If we don't have an environment yet, allocate a local one. */
  108. if (dbenv == NULL) {
  109. if ((ret = db_env_create(&dbenv, 0)) != 0) {
  110. __os_free(dbenv, dbp);
  111. return (ret);
  112. }
  113. F_SET(dbenv, DB_ENV_DBLOCAL);
  114. }
  115. ++dbenv->db_ref;
  116. dbp->dbenv = dbenv;
  117. *dbpp = dbp;
  118. return (0);
  119. }
  120. /*
  121.  * __db_init --
  122.  * Initialize a DB structure.
  123.  */
  124. static int
  125. __db_init(dbp, flags)
  126. DB *dbp;
  127. u_int32_t flags;
  128. {
  129. int ret;
  130. dbp->lid = DB_LOCK_INVALIDID;
  131. LOCK_INIT(dbp->handle_lock);
  132. TAILQ_INIT(&dbp->free_queue);
  133. TAILQ_INIT(&dbp->active_queue);
  134. TAILQ_INIT(&dbp->join_queue);
  135. LIST_INIT(&dbp->s_secondaries);
  136. FLD_SET(dbp->am_ok,
  137.     DB_OK_BTREE | DB_OK_HASH | DB_OK_QUEUE | DB_OK_RECNO);
  138. dbp->associate = __db_associate;
  139. dbp->close = __db_close;
  140. dbp->cursor = __db_cursor;
  141. dbp->del = __db_delete;
  142. dbp->err = __dbh_err;
  143. dbp->errx = __dbh_errx;
  144. dbp->fd = __db_fd;
  145. dbp->get = __db_get;
  146. dbp->get_byteswapped = __db_get_byteswapped;
  147. dbp->get_type = __db_get_type;
  148. dbp->join = __db_join;
  149. dbp->key_range = __db_key_range;
  150. dbp->open = __db_open;
  151. dbp->pget = __db_pget;
  152. dbp->put = __db_put;
  153. dbp->remove = __db_remove;
  154. dbp->rename = __db_rename;
  155. dbp->truncate = __db_truncate;
  156. dbp->set_alloc = __db_set_alloc;
  157. dbp->set_append_recno = __db_set_append_recno;
  158. dbp->set_cachesize = __db_set_cachesize;
  159. dbp->set_cache_priority = __db_set_cache_priority;
  160. dbp->set_dup_compare = __db_set_dup_compare;
  161. dbp->set_encrypt = __db_set_encrypt;
  162. dbp->set_errcall = __db_set_errcall;
  163. dbp->set_errfile = __db_set_errfile;
  164. dbp->set_errpfx = __db_set_errpfx;
  165. dbp->set_feedback = __db_set_feedback;
  166. dbp->set_flags = __db_set_flags;
  167. dbp->set_lorder = __db_set_lorder;
  168. dbp->set_pagesize = __db_set_pagesize;
  169. dbp->set_paniccall = __db_set_paniccall;
  170. dbp->stat = __db_stat_fail;
  171. dbp->sync = __db_sync;
  172. dbp->upgrade = __db_upgrade;
  173. dbp->verify = __db_verify;
  174. /* Access method specific. */
  175. if ((ret = __bam_db_create(dbp)) != 0)
  176. return (ret);
  177. if ((ret = __ham_db_create(dbp)) != 0)
  178. return (ret);
  179. if ((ret = __qam_db_create(dbp)) != 0)
  180. return (ret);
  181. /*
  182.  * XA specific: must be last, as we replace methods set by the
  183.  * access methods.
  184.  */
  185. if (LF_ISSET(DB_XA_CREATE) && (ret = __db_xa_create(dbp)) != 0)
  186. return (ret);
  187. return (0);
  188. }
  189. /*
  190.  * __dbh_am_chk --
  191.  * Error if an unreasonable method is called.
  192.  *
  193.  * PUBLIC: int __dbh_am_chk __P((DB *, u_int32_t));
  194.  */
  195. int
  196. __dbh_am_chk(dbp, flags)
  197. DB *dbp;
  198. u_int32_t flags;
  199. {
  200. /*
  201.  * We start out allowing any access methods to be called, and as the
  202.  * application calls the methods the options become restricted.  The
  203.  * idea is to quit as soon as an illegal method combination is called.
  204.  */
  205. if ((LF_ISSET(DB_OK_BTREE) && FLD_ISSET(dbp->am_ok, DB_OK_BTREE)) ||
  206.     (LF_ISSET(DB_OK_HASH) && FLD_ISSET(dbp->am_ok, DB_OK_HASH)) ||
  207.     (LF_ISSET(DB_OK_QUEUE) && FLD_ISSET(dbp->am_ok, DB_OK_QUEUE)) ||
  208.     (LF_ISSET(DB_OK_RECNO) && FLD_ISSET(dbp->am_ok, DB_OK_RECNO))) {
  209. FLD_CLR(dbp->am_ok, ~flags);
  210. return (0);
  211. }
  212. __db_err(dbp->dbenv,
  213.     "call implies an access method which is inconsistent with previous calls");
  214. return (EINVAL);
  215. }
  216. /*
  217.  * __dbh_err --
  218.  * Error message, including the standard error string.
  219.  */
  220. static void
  221. #ifdef __STDC__
  222. __dbh_err(DB *dbp, int error, const char *fmt, ...)
  223. #else
  224. __dbh_err(dbp, error, fmt, va_alist)
  225. DB *dbp;
  226. int error;
  227. const char *fmt;
  228. va_dcl
  229. #endif
  230. {
  231. DB_REAL_ERR(dbp->dbenv, error, 1, 1, fmt);
  232. }
  233. /*
  234.  * __dbh_errx --
  235.  * Error message.
  236.  */
  237. static void
  238. #ifdef __STDC__
  239. __dbh_errx(DB *dbp, const char *fmt, ...)
  240. #else
  241. __dbh_errx(dbp, fmt, va_alist)
  242. DB *dbp;
  243. const char *fmt;
  244. va_dcl
  245. #endif
  246. {
  247. DB_REAL_ERR(dbp->dbenv, 0, 0, 1, fmt);
  248. }
  249. /*
  250.  * __db_get_byteswapped --
  251.  * Return if database requires byte swapping.
  252.  */
  253. static int
  254. __db_get_byteswapped(dbp, isswapped)
  255. DB *dbp;
  256. int *isswapped;
  257. {
  258. DB_ILLEGAL_BEFORE_OPEN(dbp, "get_byteswapped");
  259. *isswapped = F_ISSET(dbp, DB_AM_SWAP) ? 1 : 0;
  260. return (0);
  261. }
  262. /*
  263.  * __db_get_type --
  264.  * Return type of underlying database.
  265.  */
  266. static int
  267. __db_get_type(dbp, dbtype)
  268. DB *dbp;
  269. DBTYPE *dbtype;
  270. {
  271. DB_ILLEGAL_BEFORE_OPEN(dbp, "get_type");
  272. *dbtype = dbp->type;
  273. return (0);
  274. }
  275. /*
  276.  * __db_key_range --
  277.  * Return proportion of keys above and below given key.
  278.  */
  279. static int
  280. __db_key_range(dbp, txn, key, kr, flags)
  281. DB *dbp;
  282. DB_TXN *txn;
  283. DBT *key;
  284. DB_KEY_RANGE *kr;
  285. u_int32_t flags;
  286. {
  287. COMPQUIET(txn, NULL);
  288. COMPQUIET(key, NULL);
  289. COMPQUIET(kr, NULL);
  290. COMPQUIET(flags, 0);
  291. DB_ILLEGAL_BEFORE_OPEN(dbp, "key_range");
  292. DB_ILLEGAL_METHOD(dbp, DB_OK_BTREE);
  293. return (EINVAL);
  294. }
  295. /*
  296.  * __db_set_append_recno --
  297.  * Set record number append routine.
  298.  */
  299. static int
  300. __db_set_append_recno(dbp, func)
  301. DB *dbp;
  302. int (*func) __P((DB *, DBT *, db_recno_t));
  303. {
  304. DB_ILLEGAL_AFTER_OPEN(dbp, "set_append_recno");
  305. DB_ILLEGAL_METHOD(dbp, DB_OK_QUEUE | DB_OK_RECNO);
  306. dbp->db_append_recno = func;
  307. return (0);
  308. }
  309. /*
  310.  * __db_set_cachesize --
  311.  * Set underlying cache size.
  312.  */
  313. static int
  314. __db_set_cachesize(dbp, cache_gbytes, cache_bytes, ncache)
  315. DB *dbp;
  316. u_int32_t cache_gbytes, cache_bytes;
  317. int ncache;
  318. {
  319. DB_ILLEGAL_IN_ENV(dbp, "set_cachesize");
  320. DB_ILLEGAL_AFTER_OPEN(dbp, "set_cachesize");
  321. return (dbp->dbenv->set_cachesize(
  322.     dbp->dbenv, cache_gbytes, cache_bytes, ncache));
  323. }
  324. /*
  325.  * __db_set_cache_priority --
  326.  * Set cache priority for pages from this file.
  327.  */
  328. static int
  329. __db_set_cache_priority(dbp, priority)
  330. DB *dbp;
  331. DB_CACHE_PRIORITY priority;
  332. {
  333. /*
  334.  * If an underlying DB_MPOOLFILE exists, call it.  Otherwise, save
  335.  * the information away until DB->open is called.
  336.  */
  337. if (dbp->mpf == NULL) {
  338. dbp->priority = priority;
  339. return (0);
  340. }
  341. return (dbp->mpf->set_priority(dbp->mpf, priority));
  342. }
  343. /*
  344.  * __db_set_dup_compare --
  345.  * Set duplicate comparison routine.
  346.  */
  347. static int
  348. __db_set_dup_compare(dbp, func)
  349. DB *dbp;
  350. int (*func) __P((DB *, const DBT *, const DBT *));
  351. {
  352. int ret;
  353. DB_ILLEGAL_AFTER_OPEN(dbp, "dup_compare");
  354. DB_ILLEGAL_METHOD(dbp, DB_OK_BTREE | DB_OK_HASH);
  355. if ((ret = dbp->set_flags(dbp, DB_DUPSORT)) != 0)
  356. return (ret);
  357. dbp->dup_compare = func;
  358. return (0);
  359. }
  360. /*
  361.  * __db_set_encrypt --
  362.  * Set database passwd.
  363.  */
  364. static int
  365. __db_set_encrypt(dbp, passwd, flags)
  366. DB *dbp;
  367. const char *passwd;
  368. u_int32_t flags;
  369. {
  370. DB_CIPHER *db_cipher;
  371. int ret;
  372. DB_ILLEGAL_IN_ENV(dbp, "set_encrypt");
  373. DB_ILLEGAL_AFTER_OPEN(dbp, "set_encrypt");
  374. if ((ret = dbp->dbenv->set_encrypt(dbp->dbenv, passwd, flags)) != 0)
  375. return (ret);
  376. /*
  377.  * In a real env, this gets initialized with the region.  In a local
  378.  * env, we must do it here.
  379.  */
  380. db_cipher = (DB_CIPHER *)dbp->dbenv->crypto_handle;
  381. if (!F_ISSET(db_cipher, CIPHER_ANY) &&
  382.     (ret = db_cipher->init(dbp->dbenv, db_cipher)) != 0)
  383. return (ret);
  384. return (dbp->set_flags(dbp, DB_ENCRYPT));
  385. }
  386. static void
  387. __db_set_errcall(dbp, errcall)
  388. DB *dbp;
  389. void (*errcall) __P((const char *, char *));
  390. {
  391. dbp->dbenv->set_errcall(dbp->dbenv, errcall);
  392. }
  393. static void
  394. __db_set_errfile(dbp, errfile)
  395. DB *dbp;
  396. FILE *errfile;
  397. {
  398. dbp->dbenv->set_errfile(dbp->dbenv, errfile);
  399. }
  400. static void
  401. __db_set_errpfx(dbp, errpfx)
  402. DB *dbp;
  403. const char *errpfx;
  404. {
  405. dbp->dbenv->set_errpfx(dbp->dbenv, errpfx);
  406. }
  407. static int
  408. __db_set_feedback(dbp, feedback)
  409. DB *dbp;
  410. void (*feedback) __P((DB *, int, int));
  411. {
  412. dbp->db_feedback = feedback;
  413. return (0);
  414. }
  415. static int
  416. __db_set_flags(dbp, flags)
  417. DB *dbp;
  418. u_int32_t flags;
  419. {
  420. int ret;
  421. /*
  422.  * !!!
  423.  * The hash access method only takes two flags: DB_DUP and DB_DUPSORT.
  424.  * The Btree access method uses them for the same purposes, and so we
  425.  * resolve them there.
  426.  *
  427.  * The queue access method takes no flags.
  428.  */
  429. if (LF_ISSET(DB_ENCRYPT)) {
  430. if (!CRYPTO_ON(dbp->dbenv)) {
  431. __db_err(dbp->dbenv,
  432.     "Database environment not configured for encryption");
  433. return (EINVAL);
  434. }
  435. F_SET(dbp, DB_AM_ENCRYPT);
  436. F_SET(dbp, DB_AM_CHKSUM);
  437. LF_CLR(DB_ENCRYPT);
  438. }
  439. if (LF_ISSET(DB_CHKSUM_SHA1)) {
  440. F_SET(dbp, DB_AM_CHKSUM);
  441. LF_CLR(DB_CHKSUM_SHA1);
  442. }
  443. if ((ret = __bam_set_flags(dbp, &flags)) != 0)
  444. return (ret);
  445. if ((ret = __ram_set_flags(dbp, &flags)) != 0)
  446. return (ret);
  447. return (flags == 0 ? 0 : __db_ferr(dbp->dbenv, "DB->set_flags", 0));
  448. }
  449. /*
  450.  * __db_set_lorder --
  451.  * Set whether lorder is swapped or not.
  452.  *
  453.  * PUBLIC: int  __db_set_lorder __P((DB *, int));
  454.  */
  455. int
  456. __db_set_lorder(dbp, db_lorder)
  457. DB *dbp;
  458. int db_lorder;
  459. {
  460. int ret;
  461. DB_ILLEGAL_AFTER_OPEN(dbp, "set_lorder");
  462. /* Flag if the specified byte order requires swapping. */
  463. switch (ret = __db_byteorder(dbp->dbenv, db_lorder)) {
  464. case 0:
  465. F_CLR(dbp, DB_AM_SWAP);
  466. break;
  467. case DB_SWAPBYTES:
  468. F_SET(dbp, DB_AM_SWAP);
  469. break;
  470. default:
  471. return (ret);
  472. /* NOTREACHED */
  473. }
  474. return (0);
  475. }
  476. static int
  477. __db_set_alloc(dbp, mal_func, real_func, free_func)
  478. DB *dbp;
  479. void *(*mal_func) __P((size_t));
  480. void *(*real_func) __P((void *, size_t));
  481. void (*free_func) __P((void *));
  482. {
  483. DB_ILLEGAL_IN_ENV(dbp, "set_alloc");
  484. DB_ILLEGAL_AFTER_OPEN(dbp, "set_alloc");
  485. return (dbp->dbenv->set_alloc(dbp->dbenv,
  486.     mal_func, real_func, free_func));
  487. }
  488. static int
  489. __db_set_pagesize(dbp, db_pagesize)
  490. DB *dbp;
  491. u_int32_t db_pagesize;
  492. {
  493. DB_ILLEGAL_AFTER_OPEN(dbp, "set_pagesize");
  494. if (db_pagesize < DB_MIN_PGSIZE) {
  495. __db_err(dbp->dbenv, "page sizes may not be smaller than %lu",
  496.     (u_long)DB_MIN_PGSIZE);
  497. return (EINVAL);
  498. }
  499. if (db_pagesize > DB_MAX_PGSIZE) {
  500. __db_err(dbp->dbenv, "page sizes may not be larger than %lu",
  501.     (u_long)DB_MAX_PGSIZE);
  502. return (EINVAL);
  503. }
  504. /*
  505.  * We don't want anything that's not a power-of-2, as we rely on that
  506.  * for alignment of various types on the pages.
  507.  */
  508. if (!POWER_OF_TWO(db_pagesize)) {
  509. __db_err(dbp->dbenv, "page sizes must be a power-of-2");
  510. return (EINVAL);
  511. }
  512. /*
  513.  * XXX
  514.  * Should we be checking for a page size that's not a multiple of 512,
  515.  * so that we never try and write less than a disk sector?
  516.  */
  517. dbp->pgsize = db_pagesize;
  518. return (0);
  519. }
  520. static int
  521. __db_set_paniccall(dbp, paniccall)
  522. DB *dbp;
  523. void (*paniccall) __P((DB_ENV *, int));
  524. {
  525. return (dbp->dbenv->set_paniccall(dbp->dbenv, paniccall));
  526. }
  527. static int
  528. __db_stat_fail(dbp, sp, flags)
  529. DB *dbp;
  530. void *sp;
  531. u_int32_t flags;
  532. {
  533. COMPQUIET(sp, NULL);
  534. COMPQUIET(flags, 0);
  535. /*
  536.  * DB->stat isn't initialized until the actual DB->open call,
  537.  * but we don't want to core dump.
  538.  */
  539. PANIC_CHECK(dbp->dbenv);
  540. DB_ILLEGAL_BEFORE_OPEN(dbp, "DB->stat");
  541. /* NOTREACHED */
  542. return (EINVAL);
  543. }
  544. #ifdef HAVE_RPC
  545. /*
  546.  * __dbcl_init --
  547.  * Initialize a DB structure on the server.
  548.  */
  549. static int
  550. __dbcl_init(dbp, dbenv, flags)
  551. DB *dbp;
  552. DB_ENV *dbenv;
  553. u_int32_t flags;
  554. {
  555. TAILQ_INIT(&dbp->free_queue);
  556. TAILQ_INIT(&dbp->active_queue);
  557. /* !!!
  558.  * Note that we don't need to initialize the join_queue;  it's
  559.  * not used in RPC clients.  See the comment in __dbcl_db_join_ret().
  560.  */
  561. dbp->associate = __dbcl_db_associate;
  562. dbp->close = __dbcl_db_close;
  563. dbp->cursor = __dbcl_db_cursor;
  564. dbp->del = __dbcl_db_del;
  565. dbp->err = __dbh_err;
  566. dbp->errx = __dbh_errx;
  567. dbp->fd = __dbcl_db_fd;
  568. dbp->get = __dbcl_db_get;
  569. dbp->get_byteswapped = __db_get_byteswapped;
  570. dbp->get_type = __db_get_type;
  571. dbp->join = __dbcl_db_join;
  572. dbp->key_range = __dbcl_db_key_range;
  573. dbp->open = __dbcl_db_open_wrap;
  574. dbp->pget = __dbcl_db_pget;
  575. dbp->put = __dbcl_db_put;
  576. dbp->remove = __dbcl_db_remove;
  577. dbp->rename = __dbcl_db_rename;
  578. dbp->set_alloc = __dbcl_db_alloc;
  579. dbp->set_append_recno = __dbcl_db_set_append_recno;
  580. dbp->set_cachesize = __dbcl_db_cachesize;
  581. dbp->set_cache_priority = __dbcl_db_cache_priority;
  582. dbp->set_dup_compare = __dbcl_db_dup_compare;
  583. dbp->set_encrypt = __dbcl_db_encrypt;
  584. dbp->set_errcall = __db_set_errcall;
  585. dbp->set_errfile = __db_set_errfile;
  586. dbp->set_errpfx = __db_set_errpfx;
  587. dbp->set_feedback = __dbcl_db_feedback;
  588. dbp->set_flags = __dbcl_db_flags;
  589. dbp->set_lorder = __dbcl_db_lorder;
  590. dbp->set_pagesize = __dbcl_db_pagesize;
  591. dbp->set_paniccall = __dbcl_db_panic;
  592. dbp->stat = __dbcl_db_stat;
  593. dbp->sync = __dbcl_db_sync;
  594. dbp->truncate = __dbcl_db_truncate;
  595. dbp->upgrade = __dbcl_db_upgrade;
  596. dbp->verify = __dbcl_db_verify;
  597. /*
  598.  * Set all the method specific functions to client funcs as well.
  599.  */
  600. dbp->set_bt_compare = __dbcl_db_bt_compare;
  601. dbp->set_bt_maxkey = __dbcl_db_bt_maxkey;
  602. dbp->set_bt_minkey = __dbcl_db_bt_minkey;
  603. dbp->set_bt_prefix = __dbcl_db_bt_prefix;
  604. dbp->set_h_ffactor = __dbcl_db_h_ffactor;
  605. dbp->set_h_hash = __dbcl_db_h_hash;
  606. dbp->set_h_nelem = __dbcl_db_h_nelem;
  607. dbp->set_q_extentsize = __dbcl_db_extentsize;
  608. dbp->set_re_delim = __dbcl_db_re_delim;
  609. dbp->set_re_len = __dbcl_db_re_len;
  610. dbp->set_re_pad = __dbcl_db_re_pad;
  611. dbp->set_re_source = __dbcl_db_re_source;
  612. return (__dbcl_db_create(dbp, dbenv, flags));
  613. }
  614. #endif