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

MySQL数据库

开发平台:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1996-2002
  5.  * Sleepycat Software.  All rights reserved.
  6.  */
  7. #include "db_config.h"
  8. #ifndef lint
  9. static const char revid[] = "$Id: db_err.c,v 11.80 2002/07/30 01:21:53 bostic Exp $";
  10. #endif /* not lint */
  11. #ifndef NO_SYSTEM_INCLUDES
  12. #include <sys/types.h>
  13. #include <stdio.h>
  14. #include <stdlib.h>
  15. #include <string.h>
  16. #endif
  17. #include "db_int.h"
  18. #include "dbinc/db_page.h"
  19. #include "dbinc/db_am.h"
  20. #include "dbinc/db_shash.h"
  21. #include "dbinc/lock.h"
  22. #include "dbinc/log.h"
  23. #include "dbinc/txn.h"
  24. /*
  25.  * __db_fchk --
  26.  * General flags checking routine.
  27.  *
  28.  * PUBLIC: int __db_fchk __P((DB_ENV *, const char *, u_int32_t, u_int32_t));
  29.  */
  30. int
  31. __db_fchk(dbenv, name, flags, ok_flags)
  32. DB_ENV *dbenv;
  33. const char *name;
  34. u_int32_t flags, ok_flags;
  35. {
  36. return (LF_ISSET(~ok_flags) ? __db_ferr(dbenv, name, 0) : 0);
  37. }
  38. /*
  39.  * __db_fcchk --
  40.  * General combination flags checking routine.
  41.  *
  42.  * PUBLIC: int __db_fcchk
  43.  * PUBLIC:    __P((DB_ENV *, const char *, u_int32_t, u_int32_t, u_int32_t));
  44.  */
  45. int
  46. __db_fcchk(dbenv, name, flags, flag1, flag2)
  47. DB_ENV *dbenv;
  48. const char *name;
  49. u_int32_t flags, flag1, flag2;
  50. {
  51. return (LF_ISSET(flag1) &&
  52.     LF_ISSET(flag2) ? __db_ferr(dbenv, name, 1) : 0);
  53. }
  54. /*
  55.  * __db_ferr --
  56.  * Common flag errors.
  57.  *
  58.  * PUBLIC: int __db_ferr __P((const DB_ENV *, const char *, int));
  59.  */
  60. int
  61. __db_ferr(dbenv, name, iscombo)
  62. const DB_ENV *dbenv;
  63. const char *name;
  64. int iscombo;
  65. {
  66. __db_err(dbenv, "illegal flag %sspecified to %s",
  67.     iscombo ? "combination " : "", name);
  68. return (EINVAL);
  69. }
  70. /*
  71.  * __db_pgerr --
  72.  * Error when unable to retrieve a specified page.
  73.  *
  74.  * PUBLIC: void __db_pgerr __P((DB *, db_pgno_t, int));
  75.  */
  76. void
  77. __db_pgerr(dbp, pgno, errval)
  78. DB *dbp;
  79. db_pgno_t pgno;
  80. int errval;
  81. {
  82. /*
  83.  * Three things are certain:
  84.  * Death, taxes, and lost data.
  85.  * Guess which has occurred.
  86.  */
  87. __db_err(dbp->dbenv,
  88.     "unable to create/retrieve page %lu", (u_long)pgno);
  89. (void)__db_panic(dbp->dbenv, errval);
  90. }
  91. /*
  92.  * __db_pgfmt --
  93.  * Error when a page has the wrong format.
  94.  *
  95.  * PUBLIC: int __db_pgfmt __P((DB_ENV *, db_pgno_t));
  96.  */
  97. int
  98. __db_pgfmt(dbenv, pgno)
  99. DB_ENV *dbenv;
  100. db_pgno_t pgno;
  101. {
  102. __db_err(dbenv, "page %lu: illegal page type or format", (u_long)pgno);
  103. return (__db_panic(dbenv, EINVAL));
  104. }
  105. /*
  106.  * __db_eopnotsup --
  107.  * Common operation not supported message.
  108.  *
  109.  * PUBLIC: int __db_eopnotsup __P((const DB_ENV *));
  110.  */
  111. int
  112. __db_eopnotsup(dbenv)
  113. const DB_ENV *dbenv;
  114. {
  115. __db_err(dbenv, "operation not supported");
  116. #ifdef EOPNOTSUPP
  117. return (EOPNOTSUPP);
  118. #else
  119. return (EINVAL);
  120. #endif
  121. }
  122. #ifdef DIAGNOSTIC
  123. /*
  124.  * __db_assert --
  125.  * Error when an assertion fails.  Only checked if #DIAGNOSTIC defined.
  126.  *
  127.  * PUBLIC: #ifdef DIAGNOSTIC
  128.  * PUBLIC: void __db_assert __P((const char *, const char *, int));
  129.  * PUBLIC: #endif
  130.  */
  131. void
  132. __db_assert(failedexpr, file, line)
  133. const char *failedexpr, *file;
  134. int line;
  135. {
  136. (void)fprintf(stderr,
  137.     "__db_assert: "%s" failed: file "%s", line %dn",
  138.     failedexpr, file, line);
  139. (void)fflush(stderr);
  140. /* We want a stack trace of how this could possibly happen. */
  141. abort();
  142. /* NOTREACHED */
  143. }
  144. #endif
  145. /*
  146.  * __db_panic_msg --
  147.  * Just report that someone else paniced.
  148.  *
  149.  * PUBLIC: int __db_panic_msg __P((DB_ENV *));
  150.  */
  151. int
  152. __db_panic_msg(dbenv)
  153. DB_ENV *dbenv;
  154. {
  155. __db_err(dbenv, "fatal region error detected; run recovery");
  156. return (DB_RUNRECOVERY);
  157. }
  158. /*
  159.  * __db_panic --
  160.  * Lock out the tree due to unrecoverable error.
  161.  *
  162.  * PUBLIC: int __db_panic __P((DB_ENV *, int));
  163.  */
  164. int
  165. __db_panic(dbenv, errval)
  166. DB_ENV *dbenv;
  167. int errval;
  168. {
  169. if (dbenv != NULL) {
  170. PANIC_SET(dbenv, 1);
  171. dbenv->panic_errval = errval;
  172. __db_err(dbenv, "PANIC: %s", db_strerror(errval));
  173. if (dbenv->db_paniccall != NULL)
  174. dbenv->db_paniccall(dbenv, errval);
  175. }
  176. #if defined(DIAGNOSTIC) && !defined(CONFIG_TEST)
  177. /*
  178.  * We want a stack trace of how this could possibly happen.
  179.  *
  180.  * Don't drop core if it's the test suite -- it's reasonable for the
  181.  * test suite to check to make sure that DB_RUNRECOVERY is returned
  182.  * under certain conditions.
  183.  */
  184. abort();
  185. #endif
  186. /*
  187.  * Chaos reigns within.
  188.  * Reflect, repent, and reboot.
  189.  * Order shall return.
  190.  */
  191. return (DB_RUNRECOVERY);
  192. }
  193. /*
  194.  * db_strerror --
  195.  * ANSI C strerror(3) for DB.
  196.  *
  197.  * EXTERN: char *db_strerror __P((int));
  198.  */
  199. char *
  200. db_strerror(error)
  201. int error;
  202. {
  203. if (error == 0)
  204. return ("Successful return: 0");
  205. if (error > 0)
  206. return (strerror(error));
  207. /*
  208.  * !!!
  209.  * The Tcl API requires that some of these return strings be compared
  210.  * against strings stored in application scripts.  So, any of these
  211.  * errors that do not invariably result in a Tcl exception may not be
  212.  * altered.
  213.  */
  214. switch (error) {
  215. case DB_DONOTINDEX:
  216. return ("DB_DONOTINDEX: Secondary index callback returns null");
  217. case DB_KEYEMPTY:
  218. return ("DB_KEYEMPTY: Non-existent key/data pair");
  219. case DB_KEYEXIST:
  220. return ("DB_KEYEXIST: Key/data pair already exists");
  221. case DB_LOCK_DEADLOCK:
  222. return
  223.     ("DB_LOCK_DEADLOCK: Locker killed to resolve a deadlock");
  224. case DB_LOCK_NOTGRANTED:
  225. return ("DB_LOCK_NOTGRANTED: Lock not granted");
  226. case DB_NOSERVER:
  227. return ("DB_NOSERVER: Fatal error, no server");
  228. case DB_NOSERVER_HOME:
  229. return ("DB_NOSERVER_HOME: Home unrecognized at server");
  230. case DB_NOSERVER_ID:
  231. return ("DB_NOSERVER_ID: Identifier unrecognized at server");
  232. case DB_NOTFOUND:
  233. return ("DB_NOTFOUND: No matching key/data pair found");
  234. case DB_OLD_VERSION:
  235. return ("DB_OLDVERSION: Database requires a version upgrade");
  236. case DB_PAGE_NOTFOUND:
  237. return ("DB_PAGE_NOTFOUND: Requested page not found");
  238. case DB_REP_DUPMASTER:
  239. return ("DB_REP_DUPMASTER: A second master site appeared");
  240. case DB_REP_HOLDELECTION:
  241. return ("DB_REP_HOLDELECTION: Need to hold an election");
  242. case DB_REP_NEWMASTER:
  243. return ("DB_REP_NEWMASTER: A new master has declared itself");
  244. case DB_REP_NEWSITE:
  245. return ("DB_REP_NEWSITE: A new site has entered the system");
  246. case DB_REP_OUTDATED:
  247. return
  248.     ("DB_REP_OUTDATED: Insufficient logs on master to recover");
  249. case DB_REP_UNAVAIL:
  250. return ("DB_REP_UNAVAIL: Unable to elect a master");
  251. case DB_RUNRECOVERY:
  252. return ("DB_RUNRECOVERY: Fatal error, run database recovery");
  253. case DB_SECONDARY_BAD:
  254. return
  255.     ("DB_SECONDARY_BAD: Secondary index item missing from primary");
  256. case DB_VERIFY_BAD:
  257. return ("DB_VERIFY_BAD: Database verification failed");
  258. default: {
  259. /*
  260.  * !!!
  261.  * Room for a 64-bit number + slop.  This buffer is only used
  262.  * if we're given an unknown error, which should never happen.
  263.  * Note, however, we're no longer thread-safe if it does.
  264.  */
  265. static char ebuf[40];
  266. (void)snprintf(ebuf, sizeof(ebuf), "Unknown error: %d", error);
  267. return (ebuf);
  268. }
  269. }
  270. }
  271. /*
  272.  * __db_err --
  273.  * Standard DB error routine.  The same as errx, except we don't write
  274.  * to stderr if no output mechanism was specified.
  275.  *
  276.  * PUBLIC: void __db_err __P((const DB_ENV *, const char *, ...));
  277.  */
  278. void
  279. #ifdef __STDC__
  280. __db_err(const DB_ENV *dbenv, const char *fmt, ...)
  281. #else
  282. __db_err(dbenv, fmt, va_alist)
  283. const DB_ENV *dbenv;
  284. const char *fmt;
  285. va_dcl
  286. #endif
  287. {
  288. DB_REAL_ERR(dbenv, 0, 0, 0, fmt);
  289. }
  290. /*
  291.  * __db_errcall --
  292.  * Do the error message work for callback functions.
  293.  *
  294.  * PUBLIC: void __db_errcall
  295.  * PUBLIC:          __P((const DB_ENV *, int, int, const char *, va_list));
  296.  */
  297. void
  298. __db_errcall(dbenv, error, error_set, fmt, ap)
  299. const DB_ENV *dbenv;
  300. int error, error_set;
  301. const char *fmt;
  302. va_list ap;
  303. {
  304. char *p;
  305. char errbuf[2048]; /* !!!: END OF THE STACK DON'T TRUST SPRINTF. */
  306. p = errbuf;
  307. if (fmt != NULL)
  308. p += vsnprintf(errbuf, sizeof(errbuf), fmt, ap);
  309. if (error_set)
  310. p += snprintf(p,
  311.     sizeof(errbuf) - (p - errbuf), ": %s", db_strerror(error));
  312. /*
  313.  * !!!
  314.  * We're potentially manipulating strings handed us by the application,
  315.  * and on systems without a real snprintf() the sprintf() calls could
  316.  * have overflowed the buffer.  We can't do anything about it now, but
  317.  * we don't want to return control to the application, we might have
  318.  * overwritten the stack with a Trojan horse.  We're not trying to do
  319.  * anything recoverable here because systems without snprintf support
  320.  * are pretty rare anymore.
  321.  */
  322. if ((size_t)(p - errbuf) > sizeof(errbuf)) {
  323. (void)fprintf(stderr,
  324.     "Berkeley DB: error callback interface buffer overflown");
  325. (void)fflush(stderr);
  326. abort();
  327. /* NOTREACHED */
  328. }
  329. dbenv->db_errcall(dbenv->db_errpfx, errbuf);
  330. }
  331. /*
  332.  * __db_errfile --
  333.  * Do the error message work for FILE *s.
  334.  *
  335.  * PUBLIC: void __db_errfile
  336.  * PUBLIC:          __P((const DB_ENV *, int, int, const char *, va_list));
  337.  */
  338. void
  339. __db_errfile(dbenv, error, error_set, fmt, ap)
  340. const DB_ENV *dbenv;
  341. int error, error_set;
  342. const char *fmt;
  343. va_list ap;
  344. {
  345. FILE *fp;
  346. fp = dbenv == NULL ||
  347.     dbenv->db_errfile == NULL ? stderr : dbenv->db_errfile;
  348. if (dbenv != NULL && dbenv->db_errpfx != NULL)
  349. (void)fprintf(fp, "%s: ", dbenv->db_errpfx);
  350. if (fmt != NULL) {
  351. (void)vfprintf(fp, fmt, ap);
  352. if (error_set)
  353. (void)fprintf(fp, ": ");
  354. }
  355. if (error_set)
  356. (void)fprintf(fp, "%s", db_strerror(error));
  357. (void)fprintf(fp, "n");
  358. (void)fflush(fp);
  359. }
  360. /*
  361.  * __db_logmsg --
  362.  * Write information into the DB log.
  363.  *
  364.  * PUBLIC: void __db_logmsg __P((const DB_ENV *,
  365.  * PUBLIC:     DB_TXN *, const char *, u_int32_t, const char *, ...));
  366.  */
  367. void
  368. #ifdef __STDC__
  369. __db_logmsg(const DB_ENV *dbenv,
  370.     DB_TXN *txnid, const char *opname, u_int32_t flags, const char *fmt, ...)
  371. #else
  372. __db_logmsg(dbenv, txnid, opname, flags, fmt, va_alist)
  373. const DB_ENV *dbenv;
  374. DB_TXN *txnid;
  375. const char *opname, *fmt;
  376. u_int32_t flags;
  377. va_dcl
  378. #endif
  379. {
  380. DBT opdbt, msgdbt;
  381. DB_LSN lsn;
  382. va_list ap;
  383. char __logbuf[2048]; /* !!!: END OF THE STACK DON'T TRUST SPRINTF. */
  384. if (!LOGGING_ON(dbenv))
  385. return;
  386. #ifdef __STDC__
  387. va_start(ap, fmt);
  388. #else
  389. va_start(ap);
  390. #endif
  391. memset(&opdbt, 0, sizeof(opdbt));
  392. opdbt.data = (void *)opname;
  393. opdbt.size = (u_int32_t)(strlen(opname) + 1);
  394. memset(&msgdbt, 0, sizeof(msgdbt));
  395. msgdbt.data = __logbuf;
  396. msgdbt.size = vsnprintf(__logbuf, sizeof(__logbuf), fmt, ap);
  397. /*
  398.  * XXX
  399.  * Explicitly discard the const.  Otherwise, we have to const DB_ENV
  400.  * references throughout the logging subsystem.
  401.  */
  402. __db_debug_log(
  403.     (DB_ENV *)dbenv, txnid, &lsn, flags, &opdbt, -1, &msgdbt, NULL, 0);
  404. va_end(ap);
  405. }
  406. /*
  407.  * __db_unknown_flag -- report internal error
  408.  *
  409.  * PUBLIC: int __db_unknown_flag __P((DB_ENV *, char *, u_int32_t));
  410.  */
  411. int
  412. __db_unknown_flag(dbenv, routine, flag)
  413. DB_ENV *dbenv;
  414. char *routine;
  415. u_int32_t flag;
  416. {
  417. __db_err(dbenv, "%s: Unknown flag: 0x%x", routine, flag);
  418. DB_ASSERT(0);
  419. return (EINVAL);
  420. }
  421. /*
  422.  * __db_unknown_type -- report internal error
  423.  *
  424.  * PUBLIC: int __db_unknown_type __P((DB_ENV *, char *, DBTYPE));
  425.  */
  426. int
  427. __db_unknown_type(dbenv, routine, type)
  428. DB_ENV *dbenv;
  429. char *routine;
  430. DBTYPE type;
  431. {
  432. __db_err(dbenv, "%s: Unknown db type: 0x%x", routine, type);
  433. DB_ASSERT(0);
  434. return (EINVAL);
  435. }
  436. /*
  437.  * __db_check_txn --
  438.  * Check for common transaction errors.
  439.  *
  440.  * PUBLIC: int __db_check_txn __P((DB *, DB_TXN *, u_int32_t, int));
  441.  */
  442. int
  443. __db_check_txn(dbp, txn, assoc_lid, read_op)
  444. DB *dbp;
  445. DB_TXN *txn;
  446. u_int32_t assoc_lid;
  447. int read_op;
  448. {
  449. DB_ENV *dbenv;
  450. dbenv = dbp->dbenv;
  451. /*
  452.  * If we are in recovery or aborting a transaction, then we
  453.  * don't need to enforce the rules about dbp's not allowing
  454.  * transactional operations in non-transactional dbps and
  455.  * vica-versa.  This happens all the time as the dbp during
  456.  * an abort may be transactional, but we undo operations
  457.  * outside a transaction since we're aborting.
  458.  */
  459. if (IS_RECOVERING(dbenv) || F_ISSET(dbp, DB_AM_RECOVER))
  460. return (0);
  461. /*
  462.  * Check for common transaction errors:
  463.  * Failure to pass a transaction handle to a DB operation
  464.  * Failure to configure the DB handle in a proper environment
  465.  * Operation on a handle whose open commit hasn't completed.
  466.  *
  467.  * Read operations don't require a txn even if we've used one before
  468.  * with this handle, although if they do have a txn, we'd better be
  469.  * prepared for it.
  470.  */
  471. if (txn == NULL) {
  472. if (!read_op && F_ISSET(dbp, DB_AM_TXN)) {
  473. __db_err(dbenv,
  474.     "DB handle previously used in transaction, missing transaction handle");
  475. return (EINVAL);
  476. }
  477. if (dbp->cur_lid >= TXN_MINIMUM)
  478. goto open_err;
  479. } else {
  480. if (dbp->cur_lid >= TXN_MINIMUM && dbp->cur_lid != txn->txnid)
  481. goto open_err;
  482. if (!TXN_ON(dbenv))
  483.  return (__db_not_txn_env(dbenv));
  484. if (!F_ISSET(dbp, DB_AM_TXN)) {
  485. __db_err(dbenv,
  486.     "Transaction specified for a DB handle opened outside a transaction");
  487. return (EINVAL);
  488. }
  489. }
  490. /*
  491.  * If dbp->associate_lid is not DB_LOCK_INVALIDID, that means we're in
  492.  * the middle of a DB->associate with DB_CREATE (i.e., a secondary index
  493.  * creation).
  494.  *
  495.  * In addition to the usual transaction rules, we need to lock out
  496.  * non-transactional updates that aren't part of the associate (and
  497.  * thus are using some other locker ID).
  498.  *
  499.  * Transactional updates should simply block;  from the time we
  500.  * decide to build the secondary until commit, we'll hold a write
  501.  * lock on all of its pages, so it should be safe to attempt to update
  502.  * the secondary in another transaction (presumably by updating the
  503.  * primary).
  504.  */
  505. if (!read_op && dbp->associate_lid != DB_LOCK_INVALIDID &&
  506.     txn != NULL && dbp->associate_lid != assoc_lid) {
  507. __db_err(dbenv,
  508.     "Operation forbidden while secondary index is being created");
  509. return (EINVAL);
  510. }
  511. return (0);
  512. open_err:
  513. __db_err(dbenv,
  514.     "Transaction that opened the DB handle is still active");
  515. return (EINVAL);
  516. }
  517. /*
  518.  * __db_not_txn_env --
  519.  * DB handle must be in an environment that supports transactions.
  520.  *
  521.  * PUBLIC: int __db_not_txn_env __P((DB_ENV *));
  522.  */
  523. int
  524. __db_not_txn_env(dbenv)
  525. DB_ENV *dbenv;
  526. {
  527. __db_err(dbenv, "DB environment not configured for transactions");
  528. return (EINVAL);
  529. }