You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

427 lines
9.5 KiB

/*-
* See the file LICENSE for redistribution information.
*
* Copyright (c) 1996, 1997, 1998, 1999
* Sleepycat Software. All rights reserved.
*/
#include "db_config.h"
#ifndef lint
static const char sccsid[] = "@(#)db_err.c 11.10 (Sleepycat) 11/8/99";
#endif /* not lint */
#ifndef NO_SYSTEM_INCLUDES
#include <sys/types.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifndef _MSC_VER /* WIN32 */
#ifdef __STDC__
#include <stdarg.h>
#else
#include <varargs.h>
#endif
#endif
#endif
#include "db_int.h"
#include "db_shash.h"
#include "lock.h"
#include "lock_ext.h"
#include "log.h"
#include "log_ext.h"
#include "mp.h"
#include "mp_ext.h"
#include "txn.h"
#include "txn_ext.h"
#include "common_ext.h"
#include "db_auto.h"
static void CDB___db_errcall __P((const DB_ENV *, int, int, const char *, va_list));
static void CDB___db_errfile __P((const DB_ENV *, int, int, const char *, va_list));
/*
* CDB___db_fchk --
* General flags checking routine.
*
* PUBLIC: int CDB___db_fchk __P((DB_ENV *, const char *, u_int32_t, u_int32_t));
*/
int
CDB___db_fchk(dbenv, name, flags, ok_flags)
DB_ENV *dbenv;
const char *name;
u_int32_t flags, ok_flags;
{
return (LF_ISSET(~ok_flags) ? CDB___db_ferr(dbenv, name, 0) : 0);
}
/*
* CDB___db_fcchk --
* General combination flags checking routine.
*
* PUBLIC: int CDB___db_fcchk
* PUBLIC: __P((DB_ENV *, const char *, u_int32_t, u_int32_t, u_int32_t));
*/
int
CDB___db_fcchk(dbenv, name, flags, flag1, flag2)
DB_ENV *dbenv;
const char *name;
u_int32_t flags, flag1, flag2;
{
return (LF_ISSET(flag1) &&
LF_ISSET(flag2) ? CDB___db_ferr(dbenv, name, 1) : 0);
}
/*
* CDB___db_ferr --
* Common flag errors.
*
* PUBLIC: int CDB___db_ferr __P((const DB_ENV *, const char *, int));
*/
int
CDB___db_ferr(dbenv, name, iscombo)
const DB_ENV *dbenv;
const char *name;
int iscombo;
{
CDB___db_err(dbenv, "illegal flag %sspecified to %s",
iscombo ? "combination " : "", name);
return (EINVAL);
}
/*
* CDB___db_pgerr --
* Error when unable to retrieve a specified page.
*
* PUBLIC: int CDB___db_pgerr __P((DB *, db_pgno_t));
*/
int
CDB___db_pgerr(dbp, pgno)
DB *dbp;
db_pgno_t pgno;
{
/*
* Three things are certain:
* Death, taxes, and lost data.
* Guess which has occurred.
*/
CDB___db_err(dbp->dbenv,
"unable to create/retrieve page %lu", (u_long)pgno);
return (CDB___db_panic(dbp->dbenv, EIO));
}
/*
* CDB___db_pgfmt --
* Error when a page has the wrong format.
*
* PUBLIC: int CDB___db_pgfmt __P((DB *, db_pgno_t));
*/
int
CDB___db_pgfmt(dbp, pgno)
DB *dbp;
db_pgno_t pgno;
{
CDB___db_err(dbp->dbenv,
"page %lu: illegal page type or format", (u_long)pgno);
return (CDB___db_panic(dbp->dbenv, EINVAL));
}
#ifdef DIAGNOSTIC
/*
* __db_assert --
* Error when an assertion fails. Only checked if #DIAGNOSTIC defined.
*
* PUBLIC: #ifdef DIAGNOSTIC
* PUBLIC: void __db_assert __P((const char *, const char *, int));
* PUBLIC: #endif
*/
void
__db_assert(failedexpr, file, line)
const char *failedexpr, *file;
int line;
{
(void)fprintf(stderr,
"__db_assert: \"%s\" failed: file \"%s\", line %d\n",
failedexpr, file, line);
fflush(stderr);
/* We want a stack trace of how this could possibly happen. */
abort();
/* NOTREACHED */
}
#endif
/*
* CDB___db_panic_msg --
* Just report that someone else paniced.
*
* PUBLIC: int CDB___db_panic_msg __P((DB_ENV *));
*/
int
CDB___db_panic_msg(dbenv)
DB_ENV *dbenv;
{
CDB___db_err(dbenv, "region error detected; run recovery.");
/* Hack to make fatal errors really fatal... */
fprintf(stderr,"DB_RUNRECOVERY: Fatal error, run database recovery\n");
exit(1);
return (DB_RUNRECOVERY);
}
/*
* CDB___db_panic --
* Lock out the tree due to unrecoverable error.
*
* PUBLIC: int CDB___db_panic __P((DB_ENV *, int));
*/
int
CDB___db_panic(dbenv, errval)
DB_ENV *dbenv;
int errval;
{
if (dbenv != NULL) {
((REGENV *)((REGINFO *)dbenv->reginfo)->addr)->panic = 1;
dbenv->db_panic = errval;
CDB___db_err(dbenv, "PANIC: %s", CDB_db_strerror(errval));
if (dbenv->db_paniccall != NULL)
dbenv->db_paniccall(dbenv, errval);
}
/*
* Chaos reigns within.
* Reflect, repent, and reboot.
* Order shall return.
*/
/* Hack to make fatal errors really fatal... */
fprintf(stderr,"DB_RUNRECOVERY: Fatal error, run database recovery\n");
exit(1);
return (DB_RUNRECOVERY);
}
/*
* CDB_db_strerror --
* ANSI C strerror(3) for DB.
*/
char *
CDB_db_strerror(error)
int error;
{
if (error == 0)
return ("Successful return: 0");
if (error > 0)
return (strerror(error));
/*
* !!!
* The Tcl API requires that some of these return strings be compared
* against strings stored in application scripts. So, any of these
* errors that do not invariably result in a Tcl exception may not be
* altered.
*/
switch (error) {
case DB_INCOMPLETE:
return ("DB_INCOMPLETE: Cache flush was unable to complete");
case DB_KEYEMPTY:
return ("DB_KEYEMPTY: Non-existent key/data pair");
case DB_KEYEXIST:
return ("DB_KEYEXIST: Key/data pair already exists");
case DB_LOCK_DEADLOCK:
return
("DB_LOCK_DEADLOCK: Locker killed to resolve a deadlock");
case DB_LOCK_NOTGRANTED:
return ("DB_LOCK_NOTGRANTED: Lock not granted");
case DB_NOTFOUND:
return ("DB_NOTFOUND: No matching key/data pair found");
case DB_OLD_VERSION:
return ("DB_OLDVERSION: Database requires a version upgrade");
case DB_RUNRECOVERY:
return ("DB_RUNRECOVERY: Fatal error, run database recovery");
default: {
/*
* !!!
* Room for a 64-bit number + slop. This buffer is only used
* if we're given an unknown error, which should never happen.
* Note, however, we're no longer thread-safe if it does.
*/
static char ebuf[40];
(void)snprintf(ebuf, sizeof(ebuf), "Unknown error: %d", error);
return(ebuf);
}
}
}
/*
* CDB___db_err --
* Standard DB error routine. The same as db_errx, except that we
* don't write to stderr if no output mechanism was specified.
*
* PUBLIC: #ifdef __STDC__
* PUBLIC: void CDB___db_err __P((const DB_ENV *, const char *, ...));
* PUBLIC: #else
* PUBLIC: void CDB___db_err();
* PUBLIC: #endif
*/
void
#if defined( __STDC__) || defined(_MSC_VER) /* WIN32 */
CDB___db_err(const DB_ENV *dbenv, const char *fmt, ...)
#else
CDB___db_err(dbenv, fmt, va_alist)
const DB_ENV *dbenv;
const char *fmt;
va_dcl
#endif
{
va_list ap;
#if defined(__STDC__) || defined(_MSC_VER) /* WIN32 */
va_start(ap, fmt);
#else
va_start(ap);
#endif
CDB___db_real_err(dbenv, 0, 0, 0, fmt, ap);
va_end(ap);
}
/*
* CDB___db_real_err --
* All the DB error routines end up here.
*
* PUBLIC: void CDB___db_real_err
* PUBLIC: __P((const DB_ENV *, int, int, int, const char *, va_list));
*/
void
CDB___db_real_err(dbenv, error, error_set, stderr_default, fmt, ap)
const DB_ENV *dbenv;
int error, error_set, stderr_default;
const char *fmt;
va_list ap;
{
if (dbenv != NULL && dbenv->db_errcall != NULL)
CDB___db_errcall(dbenv, error, error_set, fmt, ap);
if (dbenv != NULL && dbenv->db_errfile != NULL)
CDB___db_errfile(dbenv, error, error_set, fmt, ap);
if (stderr_default && (dbenv == NULL ||
(dbenv->db_errcall == NULL && dbenv->db_errfile == NULL)))
CDB___db_errfile(NULL, error, error_set, fmt, ap);
}
/*
* CDB___db_errcall --
* Do the error message work for callback functions.
*/
static void
CDB___db_errcall(dbenv, error, error_set, fmt, ap)
const DB_ENV *dbenv;
int error, error_set;
const char *fmt;
va_list ap;
{
char *p;
char __errbuf[2048]; /* XXX: END OF THE STACK DON'T TRUST SPRINTF. */
p = __errbuf;
if (fmt != NULL) {
p += vsnprintf(__errbuf, sizeof(__errbuf), fmt, ap);
if (error_set) {
*p++ = ':';
*p++ = ' ';
}
}
if (error_set)
(void)strcpy(p, CDB_db_strerror(error));
dbenv->db_errcall(dbenv->db_errpfx, __errbuf);
}
/*
* CDB___db_errfile --
* Do the error message work for FILE *s.
*/
static void
CDB___db_errfile(dbenv, error, error_set, fmt, ap)
const DB_ENV *dbenv;
int error, error_set;
const char *fmt;
va_list ap;
{
FILE *fp;
fp = dbenv == NULL ||
dbenv->db_errfile == NULL ? stderr : dbenv->db_errfile;
if (dbenv != NULL && dbenv->db_errpfx != NULL)
(void)fprintf(fp, "%s: ", dbenv->db_errpfx);
if (fmt != NULL) {
(void)vfprintf(fp, fmt, ap);
if (error_set)
(void)fprintf(fp, ": ");
}
if (error_set)
(void)fprintf(fp, "%s", CDB_db_strerror(error));
(void)fprintf(fp, "\n");
(void)fflush(fp);
}
/*
* CDB___db_logmsg --
* Write information into the DB log.
*
* PUBLIC: #ifdef __STDC__
* PUBLIC: int CDB___db_logmsg __P((DB_ENV *,
* PUBLIC: DB_TXN *, const char *, u_int32_t, const char *, ...));
* PUBLIC: #else
* PUBLIC: int CDB___db_logmsg();
* PUBLIC: #endif
*/
int
#if defined(__STDC__) || defined(_MSC_VER) /* WIN32 */
CDB___db_logmsg(DB_ENV *dbenv,
DB_TXN *txnid, const char *opname, u_int32_t flags, const char *fmt, ...)
#else
CDB___db_logmsg(dbenv, txnid, opname, flags, fmt, va_alist)
DB_ENV *dbenv;
DB_TXN *txnid;
const char *opname, *fmt;
u_int32_t flags;
va_dcl
#endif
{
DBT opdbt, msgdbt;
DB_LSN lsn;
va_list ap;
char __logbuf[2048]; /* XXX: END OF THE STACK DON'T TRUST SPRINTF. */
if (!F_ISSET(dbenv, DB_ENV_LOGGING))
return (0);
memset(&opdbt, 0, sizeof(opdbt));
opdbt.data = (void *)opname;
opdbt.size = strlen(opname) + 1;
memset(&msgdbt, 0, sizeof(msgdbt));
msgdbt.data = __logbuf;
#if defined(__STDC__) || defined(_MSC_VER) /* WIN32 */
va_start(ap, fmt);
#else
va_start(ap);
#endif
msgdbt.size = vsnprintf(__logbuf, sizeof(__logbuf), fmt, ap);
va_end(ap);
return (CDB___db_debug_log(dbenv,
txnid, &lsn, flags, &opdbt, -1, &msgdbt, NULL, 0));
}