db_method.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:14k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1999, 2000
  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.36 2000/12/21 09:17:04 krinsky 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. #ifdef HAVE_RPC
  19. #include "db_server.h"
  20. #endif
  21. #include "db_int.h"
  22. #include "db_page.h"
  23. #include "db_am.h"
  24. #include "btree.h"
  25. #include "hash.h"
  26. #include "qam.h"
  27. #include "xa.h"
  28. #include "xa_ext.h"
  29. #ifdef HAVE_RPC
  30. #include "gen_client_ext.h"
  31. #include "rpc_client_ext.h"
  32. #endif
  33. static int  __db_get_byteswapped __P((DB *));
  34. static DBTYPE
  35.     __db_get_type __P((DB *));
  36. static int  __db_init __P((DB *, u_int32_t));
  37. static int  __db_key_range
  38. __P((DB *, DB_TXN *, DBT *, DB_KEY_RANGE *, u_int32_t));
  39. static int  __db_set_append_recno __P((DB *, int (*)(DB *, DBT *, db_recno_t)));
  40. static int  __db_set_cachesize __P((DB *, u_int32_t, u_int32_t, int));
  41. static int  __db_set_dup_compare
  42. __P((DB *, int (*)(DB *, const DBT *, const DBT *)));
  43. static void __db_set_errcall __P((DB *, void (*)(const char *, char *)));
  44. static void __db_set_errfile __P((DB *, FILE *));
  45. static int  __db_set_feedback __P((DB *, void (*)(DB *, int, int)));
  46. static int  __db_set_flags __P((DB *, u_int32_t));
  47. static int  __db_set_lorder __P((DB *, int));
  48. static int  __db_set_malloc __P((DB *, void *(*)(size_t)));
  49. static int  __db_set_pagesize __P((DB *, u_int32_t));
  50. static int  __db_set_realloc __P((DB *, void *(*)(void *, size_t)));
  51. static void __db_set_errpfx __P((DB *, const char *));
  52. static int  __db_set_paniccall __P((DB *, void (*)(DB_ENV *, int)));
  53. static void __dbh_err __P((DB *, int, const char *, ...));
  54. static void __dbh_errx __P((DB *, const char *, ...));
  55. /*
  56.  * db_create --
  57.  * DB constructor.
  58.  */
  59. int
  60. db_create(dbpp, dbenv, flags)
  61. DB **dbpp;
  62. DB_ENV *dbenv;
  63. u_int32_t flags;
  64. {
  65. DB *dbp;
  66. int ret;
  67. /* Check for invalid function flags. */
  68. switch (flags) {
  69. case 0:
  70. break;
  71. case DB_XA_CREATE:
  72. if (dbenv != NULL) {
  73. __db_err(dbenv,
  74. "XA applications may not specify an environment to db_create");
  75. return (EINVAL);
  76. }
  77. /*
  78.  * If it's an XA database, open it within the XA environment,
  79.  * taken from the global list of environments.  (When the XA
  80.  * transaction manager called our xa_start() routine the
  81.  * "current" environment was moved to the start of the list.
  82.  */
  83. dbenv = TAILQ_FIRST(&DB_GLOBAL(db_envq));
  84. break;
  85. default:
  86. return (__db_ferr(dbenv, "db_create", 0));
  87. }
  88. /* Allocate the DB. */
  89. if ((ret = __os_calloc(dbenv, 1, sizeof(*dbp), &dbp)) != 0)
  90. return (ret);
  91. #ifdef HAVE_RPC
  92. if (dbenv != NULL && dbenv->cl_handle != NULL)
  93. ret = __dbcl_init(dbp, dbenv, flags);
  94. else
  95. #endif
  96. ret = __db_init(dbp, flags);
  97. if (ret != 0) {
  98. __os_free(dbp, sizeof(*dbp));
  99. return (ret);
  100. }
  101. /* If we don't have an environment yet, allocate a local one. */
  102. if (dbenv == NULL) {
  103. if ((ret = db_env_create(&dbenv, 0)) != 0) {
  104. __os_free(dbp, sizeof(*dbp));
  105. return (ret);
  106. }
  107. dbenv->dblocal_ref = 0;
  108. F_SET(dbenv, DB_ENV_DBLOCAL);
  109. }
  110. if (F_ISSET(dbenv, DB_ENV_DBLOCAL))
  111. ++dbenv->dblocal_ref;
  112. dbp->dbenv = dbenv;
  113. *dbpp = dbp;
  114. return (0);
  115. }
  116. /*
  117.  * __db_init --
  118.  * Initialize a DB structure.
  119.  */
  120. static int
  121. __db_init(dbp, flags)
  122. DB *dbp;
  123. u_int32_t flags;
  124. {
  125. int ret;
  126. dbp->log_fileid = DB_LOGFILEID_INVALID;
  127. TAILQ_INIT(&dbp->free_queue);
  128. TAILQ_INIT(&dbp->active_queue);
  129. TAILQ_INIT(&dbp->join_queue);
  130. FLD_SET(dbp->am_ok,
  131.     DB_OK_BTREE | DB_OK_HASH | DB_OK_QUEUE | DB_OK_RECNO);
  132. dbp->close = __db_close;
  133. dbp->cursor = __db_cursor;
  134. dbp->del = NULL; /* !!! Must be set by access method. */
  135. dbp->err = __dbh_err;
  136. dbp->errx = __dbh_errx;
  137. dbp->fd = __db_fd;
  138. dbp->get = __db_get;
  139. dbp->get_byteswapped = __db_get_byteswapped;
  140. dbp->get_type = __db_get_type;
  141. dbp->join = __db_join;
  142. dbp->key_range = __db_key_range;
  143. dbp->open = __db_open;
  144. dbp->put = __db_put;
  145. dbp->remove = __db_remove;
  146. dbp->rename = __db_rename;
  147. dbp->set_append_recno = __db_set_append_recno;
  148. dbp->set_cachesize = __db_set_cachesize;
  149. dbp->set_dup_compare = __db_set_dup_compare;
  150. dbp->set_errcall = __db_set_errcall;
  151. dbp->set_errfile = __db_set_errfile;
  152. dbp->set_errpfx = __db_set_errpfx;
  153. dbp->set_feedback = __db_set_feedback;
  154. dbp->set_flags = __db_set_flags;
  155. dbp->set_lorder = __db_set_lorder;
  156. dbp->set_malloc = __db_set_malloc;
  157. dbp->set_pagesize = __db_set_pagesize;
  158. dbp->set_paniccall = __db_set_paniccall;
  159. dbp->set_realloc = __db_set_realloc;
  160. dbp->stat = NULL; /* !!! Must be set by access method. */
  161. dbp->sync = __db_sync;
  162. dbp->upgrade = __db_upgrade;
  163. dbp->verify = __db_verify;
  164. /* Access method specific. */
  165. if ((ret = __bam_db_create(dbp)) != 0)
  166. return (ret);
  167. if ((ret = __ham_db_create(dbp)) != 0)
  168. return (ret);
  169. if ((ret = __qam_db_create(dbp)) != 0)
  170. return (ret);
  171. /*
  172.  * XA specific: must be last, as we replace methods set by the
  173.  * access methods.
  174.  */
  175. if (LF_ISSET(DB_XA_CREATE) && (ret = __db_xa_create(dbp)) != 0)
  176. return (ret);
  177. return (0);
  178. }
  179. /*
  180.  * __dbh_am_chk --
  181.  * Error if an unreasonable method is called.
  182.  *
  183.  * PUBLIC: int __dbh_am_chk __P((DB *, u_int32_t));
  184.  */
  185. int
  186. __dbh_am_chk(dbp, flags)
  187. DB *dbp;
  188. u_int32_t flags;
  189. {
  190. /*
  191.  * We start out allowing any access methods to be called, and as the
  192.  * application calls the methods the options become restricted.  The
  193.  * idea is to quit as soon as an illegal method combination is called.
  194.  */
  195. if ((LF_ISSET(DB_OK_BTREE) && FLD_ISSET(dbp->am_ok, DB_OK_BTREE)) ||
  196.     (LF_ISSET(DB_OK_HASH) && FLD_ISSET(dbp->am_ok, DB_OK_HASH)) ||
  197.     (LF_ISSET(DB_OK_QUEUE) && FLD_ISSET(dbp->am_ok, DB_OK_QUEUE)) ||
  198.     (LF_ISSET(DB_OK_RECNO) && FLD_ISSET(dbp->am_ok, DB_OK_RECNO))) {
  199. FLD_CLR(dbp->am_ok, ~flags);
  200. return (0);
  201. }
  202. __db_err(dbp->dbenv,
  203.     "call implies an access method which is inconsistent with previous calls");
  204. return (EINVAL);
  205. }
  206. /*
  207.  * __dbh_err --
  208.  * Error message, including the standard error string.
  209.  */
  210. static void
  211. #ifdef __STDC__
  212. __dbh_err(DB *dbp, int error, const char *fmt, ...)
  213. #else
  214. __dbh_err(dbp, error, fmt, va_alist)
  215. DB *dbp;
  216. int error;
  217. const char *fmt;
  218. va_dcl
  219. #endif
  220. {
  221. va_list ap;
  222. #ifdef __STDC__
  223. va_start(ap, fmt);
  224. #else
  225. va_start(ap);
  226. #endif
  227. __db_real_err(dbp->dbenv, error, 1, 1, fmt, ap);
  228. va_end(ap);
  229. }
  230. /*
  231.  * __dbh_errx --
  232.  * Error message.
  233.  */
  234. static void
  235. #ifdef __STDC__
  236. __dbh_errx(DB *dbp, const char *fmt, ...)
  237. #else
  238. __dbh_errx(dbp, fmt, va_alist)
  239. DB *dbp;
  240. const char *fmt;
  241. va_dcl
  242. #endif
  243. {
  244. va_list ap;
  245. #ifdef __STDC__
  246. va_start(ap, fmt);
  247. #else
  248. va_start(ap);
  249. #endif
  250. __db_real_err(dbp->dbenv, 0, 0, 1, fmt, ap);
  251. va_end(ap);
  252. }
  253. /*
  254.  * __db_get_byteswapped --
  255.  * Return if database requires byte swapping.
  256.  */
  257. static int
  258. __db_get_byteswapped(dbp)
  259. DB *dbp;
  260. {
  261. DB_ILLEGAL_BEFORE_OPEN(dbp, "get_byteswapped");
  262. return (F_ISSET(dbp, DB_AM_SWAP) ? 1 : 0);
  263. }
  264. /*
  265.  * __db_get_type --
  266.  * Return type of underlying database.
  267.  */
  268. static DBTYPE
  269. __db_get_type(dbp)
  270. DB *dbp;
  271. {
  272. DB_ILLEGAL_BEFORE_OPEN(dbp, "get_type");
  273. return (dbp->type);
  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_dup_compare --
  326.  * Set duplicate comparison routine.
  327.  */
  328. static int
  329. __db_set_dup_compare(dbp, func)
  330. DB *dbp;
  331. int (*func) __P((DB *, const DBT *, const DBT *));
  332. {
  333. DB_ILLEGAL_AFTER_OPEN(dbp, "dup_compare");
  334. DB_ILLEGAL_METHOD(dbp, DB_OK_BTREE | DB_OK_HASH);
  335. dbp->dup_compare = func;
  336. return (0);
  337. }
  338. static void
  339. __db_set_errcall(dbp, errcall)
  340. DB *dbp;
  341. void (*errcall) __P((const char *, char *));
  342. {
  343. dbp->dbenv->set_errcall(dbp->dbenv, errcall);
  344. }
  345. static void
  346. __db_set_errfile(dbp, errfile)
  347. DB *dbp;
  348. FILE *errfile;
  349. {
  350. dbp->dbenv->set_errfile(dbp->dbenv, errfile);
  351. }
  352. static void
  353. __db_set_errpfx(dbp, errpfx)
  354. DB *dbp;
  355. const char *errpfx;
  356. {
  357. dbp->dbenv->set_errpfx(dbp->dbenv, errpfx);
  358. }
  359. static int
  360. __db_set_feedback(dbp, feedback)
  361. DB *dbp;
  362. void (*feedback) __P((DB *, int, int));
  363. {
  364. dbp->db_feedback = feedback;
  365. return (0);
  366. }
  367. static int
  368. __db_set_flags(dbp, flags)
  369. DB *dbp;
  370. u_int32_t flags;
  371. {
  372. int ret;
  373. /*
  374.  * !!!
  375.  * The hash access method only takes two flags: DB_DUP and DB_DUPSORT.
  376.  * The Btree access method uses them for the same purposes, and so we
  377.  * resolve them there.
  378.  *
  379.  * The queue access method takes no flags.
  380.  */
  381. if ((ret = __bam_set_flags(dbp, &flags)) != 0)
  382. return (ret);
  383. if ((ret = __ram_set_flags(dbp, &flags)) != 0)
  384. return (ret);
  385. return (flags == 0 ? 0 : __db_ferr(dbp->dbenv, "DB->set_flags", 0));
  386. }
  387. static int
  388. __db_set_lorder(dbp, db_lorder)
  389. DB *dbp;
  390. int db_lorder;
  391. {
  392. int ret;
  393. DB_ILLEGAL_AFTER_OPEN(dbp, "set_lorder");
  394. /* Flag if the specified byte order requires swapping. */
  395. switch (ret = __db_byteorder(dbp->dbenv, db_lorder)) {
  396. case 0:
  397. F_CLR(dbp, DB_AM_SWAP);
  398. break;
  399. case DB_SWAPBYTES:
  400. F_SET(dbp, DB_AM_SWAP);
  401. break;
  402. default:
  403. return (ret);
  404. /* NOTREACHED */
  405. }
  406. return (0);
  407. }
  408. static int
  409. __db_set_malloc(dbp, func)
  410. DB *dbp;
  411. void *(*func) __P((size_t));
  412. {
  413. DB_ILLEGAL_AFTER_OPEN(dbp, "set_malloc");
  414. dbp->db_malloc = func;
  415. return (0);
  416. }
  417. static int
  418. __db_set_pagesize(dbp, db_pagesize)
  419. DB *dbp;
  420. u_int32_t db_pagesize;
  421. {
  422. DB_ILLEGAL_AFTER_OPEN(dbp, "set_pagesize");
  423. if (db_pagesize < DB_MIN_PGSIZE) {
  424. __db_err(dbp->dbenv, "page sizes may not be smaller than %lu",
  425.     (u_long)DB_MIN_PGSIZE);
  426. return (EINVAL);
  427. }
  428. if (db_pagesize > DB_MAX_PGSIZE) {
  429. __db_err(dbp->dbenv, "page sizes may not be larger than %lu",
  430.     (u_long)DB_MAX_PGSIZE);
  431. return (EINVAL);
  432. }
  433. /*
  434.  * We don't want anything that's not a power-of-2, as we rely on that
  435.  * for alignment of various types on the pages.
  436.  */
  437. if ((u_int32_t)1 << __db_log2(db_pagesize) != db_pagesize) {
  438. __db_err(dbp->dbenv, "page sizes must be a power-of-2");
  439. return (EINVAL);
  440. }
  441. /*
  442.  * XXX
  443.  * Should we be checking for a page size that's not a multiple of 512,
  444.  * so that we never try and write less than a disk sector?
  445.  */
  446. dbp->pgsize = db_pagesize;
  447. return (0);
  448. }
  449. static int
  450. __db_set_realloc(dbp, func)
  451. DB *dbp;
  452. void *(*func) __P((void *, size_t));
  453. {
  454. DB_ILLEGAL_AFTER_OPEN(dbp, "set_realloc");
  455. dbp->db_realloc = func;
  456. return (0);
  457. }
  458. static int
  459. __db_set_paniccall(dbp, paniccall)
  460. DB *dbp;
  461. void (*paniccall) __P((DB_ENV *, int));
  462. {
  463. return (dbp->dbenv->set_paniccall(dbp->dbenv, paniccall));
  464. }
  465. #ifdef HAVE_RPC
  466. /*
  467.  * __dbcl_init --
  468.  * Initialize a DB structure on the server.
  469.  *
  470.  * PUBLIC: #ifdef HAVE_RPC
  471.  * PUBLIC: int __dbcl_init __P((DB *, DB_ENV *, u_int32_t));
  472.  * PUBLIC: #endif
  473.  */
  474. int
  475. __dbcl_init(dbp, dbenv, flags)
  476. DB *dbp;
  477. DB_ENV *dbenv;
  478. u_int32_t flags;
  479. {
  480. CLIENT *cl;
  481. __db_create_reply *replyp;
  482. __db_create_msg req;
  483. int ret;
  484. TAILQ_INIT(&dbp->free_queue);
  485. TAILQ_INIT(&dbp->active_queue);
  486. /* !!!
  487.  * Note that we don't need to initialize the join_queue;  it's
  488.  * not used in RPC clients.  See the comment in __dbcl_db_join_ret().
  489.  */
  490. dbp->close = __dbcl_db_close;
  491. dbp->cursor = __dbcl_db_cursor;
  492. dbp->del = __dbcl_db_del;
  493. dbp->err = __dbh_err;
  494. dbp->errx = __dbh_errx;
  495. dbp->fd = __dbcl_db_fd;
  496. dbp->get = __dbcl_db_get;
  497. dbp->get_byteswapped = __dbcl_db_swapped;
  498. dbp->get_type = __db_get_type;
  499. dbp->join = __dbcl_db_join;
  500. dbp->key_range = __dbcl_db_key_range;
  501. dbp->open = __dbcl_db_open;
  502. dbp->put = __dbcl_db_put;
  503. dbp->remove = __dbcl_db_remove;
  504. dbp->rename = __dbcl_db_rename;
  505. dbp->set_append_recno = __dbcl_db_set_append_recno;
  506. dbp->set_cachesize = __dbcl_db_cachesize;
  507. dbp->set_dup_compare = NULL;
  508. dbp->set_errcall = __db_set_errcall;
  509. dbp->set_errfile = __db_set_errfile;
  510. dbp->set_errpfx = __db_set_errpfx;
  511. dbp->set_feedback = __dbcl_db_feedback;
  512. dbp->set_flags = __dbcl_db_flags;
  513. dbp->set_lorder = __dbcl_db_lorder;
  514. dbp->set_malloc = __dbcl_db_malloc;
  515. dbp->set_pagesize = __dbcl_db_pagesize;
  516. dbp->set_paniccall = __dbcl_db_panic;
  517. dbp->set_q_extentsize = __dbcl_db_extentsize;
  518. dbp->set_realloc = __dbcl_db_realloc;
  519. dbp->stat = __dbcl_db_stat;
  520. dbp->sync = __dbcl_db_sync;
  521. dbp->upgrade = __dbcl_db_upgrade;
  522. /*
  523.  * Set all the method specific functions to client funcs as well.
  524.  */
  525. dbp->set_bt_compare = __dbcl_db_bt_compare;
  526. dbp->set_bt_maxkey = __dbcl_db_bt_maxkey;
  527. dbp->set_bt_minkey = __dbcl_db_bt_minkey;
  528. dbp->set_bt_prefix = __dbcl_db_bt_prefix;
  529. dbp->set_h_ffactor = __dbcl_db_h_ffactor;
  530. dbp->set_h_hash = __dbcl_db_h_hash;
  531. dbp->set_h_nelem = __dbcl_db_h_nelem;
  532. dbp->set_re_delim = __dbcl_db_re_delim;
  533. dbp->set_re_len = __dbcl_db_re_len;
  534. dbp->set_re_pad = __dbcl_db_re_pad;
  535. dbp->set_re_source = __dbcl_db_re_source;
  536. /*
  537. dbp->set_q_extentsize = __dbcl_db_q_extentsize;
  538. */
  539. cl = (CLIENT *)dbenv->cl_handle;
  540. req.flags = flags;
  541. req.envpcl_id = dbenv->cl_id;
  542. /*
  543.  * CALL THE SERVER
  544.  */
  545. replyp = __db_db_create_1(&req, cl);
  546. if (replyp == NULL) {
  547. __db_err(dbenv, clnt_sperror(cl, "Berkeley DB"));
  548. return (DB_NOSERVER);
  549. }
  550. if ((ret = replyp->status) != 0)
  551. return (ret);
  552. dbp->cl_id = replyp->dbpcl_id;
  553. return (0);
  554. }
  555. #endif