#include "db_config.h"
#include "db_int.h"
#include "dbinc/db_page.h"
#include "dbinc/db_am.h"
#include "dbinc/lock.h"
#include "dbinc/mp.h"
#include "dbinc/txn.h"
static void __db_msgcall __P((const DB_ENV *, const char *, va_list));
static void __db_msgfile __P((const DB_ENV *, const char *, va_list));
int
__db_fchk(env, name, flags, ok_flags)
ENV *env;
const char *name;
u_int32_t flags, ok_flags;
{
return (LF_ISSET(~ok_flags) ? __db_ferr(env, name, 0) : 0);
}
int
__db_fcchk(env, name, flags, flag1, flag2)
ENV *env;
const char *name;
u_int32_t flags, flag1, flag2;
{
return (LF_ISSET(flag1) &&
LF_ISSET(flag2) ? __db_ferr(env, name, 1) : 0);
}
int
__db_ferr(env, name, iscombo)
const ENV *env;
const char *name;
int iscombo;
{
if (iscombo)
__db_errx(env, DB_STR_A("0054",
"illegal flag combination specified to %s", "%s"), name);
else
__db_errx(env, DB_STR_A("0055",
"illegal flag specified to %s", "%s"), name);
return (EINVAL);
}
int
__db_fnl(env, name)
const ENV *env;
const char *name;
{
__db_errx(env, DB_STR_A("0056",
"%s: DB_READ_COMMITTED, DB_READ_UNCOMMITTED and DB_RMW require locking",
"%s"), name);
return (EINVAL);
}
int
__db_pgerr(dbp, pgno, errval)
DB *dbp;
db_pgno_t pgno;
int errval;
{
__db_errx(dbp->env, DB_STR_A("0057",
"unable to create/retrieve page %lu", "%lu"), (u_long)pgno);
return (__env_panic(dbp->env, errval));
}
int
__db_pgfmt(env, pgno)
ENV *env;
db_pgno_t pgno;
{
__db_errx(env, DB_STR_A("0058",
"page %lu: illegal page type or format", "%lu"), (u_long)pgno);
return (__env_panic(env, EINVAL));
}
#ifdef DIAGNOSTIC
void
__db_assert(env, e, file, line)
ENV *env;
const char *e, *file;
int line;
{
if (DB_GLOBAL(j_assert) != NULL)
DB_GLOBAL(j_assert)(e, file, line);
else {
__db_errx(env, DB_STR_A("0059",
"assert failure: %s/%d: \"%s\"",
"%s %d %s"), file, line, e);
__os_abort(env);
}
}
#endif
int
__env_panic_msg(env)
ENV *env;
{
DB_ENV *dbenv;
int ret;
dbenv = env->dbenv;
ret = DB_RUNRECOVERY;
__db_errx(env, DB_STR("0060",
"PANIC: fatal region error detected; run recovery"));
if (dbenv->db_paniccall != NULL)
dbenv->db_paniccall(dbenv, ret);
if ((env->reginfo != NULL) &&
(((REGENV *)env->reginfo->primary)->reg_panic))
DB_EVENT(env, DB_EVENT_REG_PANIC, &ret);
else
DB_EVENT(env, DB_EVENT_PANIC, &ret);
return (ret);
}
int
__env_panic(env, errval)
ENV *env;
int errval;
{
DB_ENV *dbenv;
dbenv = env->dbenv;
if (env != NULL) {
__env_panic_set(env, 1);
__db_err(env, errval, DB_STR("0061", "PANIC"));
if (dbenv->db_paniccall != NULL)
dbenv->db_paniccall(dbenv, errval);
if ((env->reginfo != NULL) &&
(((REGENV *)env->reginfo->primary)->reg_panic))
DB_EVENT(env, DB_EVENT_REG_PANIC, &errval);
else
DB_EVENT(env, DB_EVENT_PANIC, &errval);
}
#if defined(DIAGNOSTIC) && !defined(CONFIG_TEST)
__os_abort(env);
#endif
return (DB_RUNRECOVERY);
}
char *
db_strerror(error)
int error;
{
char *p;
if (error == 0)
return (DB_STR("0062", "Successful return: 0"));
if (error > 0) {
if ((p = strerror(error)) != NULL)
return (p);
return (__db_unknown_error(error));
}
switch (error) {
case DB_BUFFER_SMALL:
return (DB_STR("0063",
"DB_BUFFER_SMALL: User memory too small for return value"));
case DB_DONOTINDEX:
return (DB_STR("0064",
"DB_DONOTINDEX: Secondary index callback returns null"));
case DB_FOREIGN_CONFLICT:
return (DB_STR("0065",
"DB_FOREIGN_CONFLICT: A foreign database constraint has been violated"));
case DB_HEAP_FULL:
return (DB_STR("0208","DB_HEAP_FULL: no free space in db"));
case DB_KEYEMPTY:
return (DB_STR("0066",
"DB_KEYEMPTY: Non-existent key/data pair"));
case DB_KEYEXIST:
return (DB_STR("0067",
"DB_KEYEXIST: Key/data pair already exists"));
case DB_LOCK_DEADLOCK:
return (DB_STR("0068",
"DB_LOCK_DEADLOCK: Locker killed to resolve a deadlock"));
case DB_LOCK_NOTGRANTED:
return (DB_STR("0069", "DB_LOCK_NOTGRANTED: Lock not granted"));
case DB_LOG_BUFFER_FULL:
return (DB_STR("0070",
"DB_LOG_BUFFER_FULL: In-memory log buffer is full"));
case DB_LOG_VERIFY_BAD:
return (DB_STR("0071",
"DB_LOG_VERIFY_BAD: Log verification failed"));
case DB_NOSERVER:
return (DB_STR("0072",
"DB_NOSERVER: No message dispatch call-back function has been configured"));
case DB_NOTFOUND:
return (DB_STR("0073",
"DB_NOTFOUND: No matching key/data pair found"));
case DB_OLD_VERSION:
return (DB_STR("0074",
"DB_OLDVERSION: Database requires a version upgrade"));
case DB_PAGE_NOTFOUND:
return (DB_STR("0075",
"DB_PAGE_NOTFOUND: Requested page not found"));
case DB_REP_DUPMASTER:
return (DB_STR("0076",
"DB_REP_DUPMASTER: A second master site appeared"));
case DB_REP_HANDLE_DEAD:
return (DB_STR("0077",
"DB_REP_HANDLE_DEAD: Handle is no longer valid"));
case DB_REP_HOLDELECTION:
return (DB_STR("0078",
"DB_REP_HOLDELECTION: Need to hold an election"));
case DB_REP_IGNORE:
return (DB_STR("0079",
"DB_REP_IGNORE: Replication record/operation ignored"));
case DB_REP_ISPERM:
return (DB_STR("0080",
"DB_REP_ISPERM: Permanent record written"));
case DB_REP_JOIN_FAILURE:
return (DB_STR("0081",
"DB_REP_JOIN_FAILURE: Unable to join replication group"));
case DB_REP_LEASE_EXPIRED:
return (DB_STR("0082",
"DB_REP_LEASE_EXPIRED: Replication leases have expired"));
case DB_REP_LOCKOUT:
return (DB_STR("0083",
"DB_REP_LOCKOUT: Waiting for replication recovery to complete"));
case DB_REP_NEWSITE:
return (DB_STR("0084",
"DB_REP_NEWSITE: A new site has entered the system"));
case DB_REP_NOTPERM:
return (DB_STR("0085",
"DB_REP_NOTPERM: Permanent log record not written"));
case DB_REP_UNAVAIL:
return (DB_STR("0086",
"DB_REP_UNAVAIL: Too few remote sites to complete operation"));
case DB_REP_WOULDROLLBACK:
return (DB_STR("0207",
"DB_REP_WOULDROLLBACK: Client data has diverged"));
case DB_RUNRECOVERY:
return (DB_STR("0087",
"DB_RUNRECOVERY: Fatal error, run database recovery"));
case DB_SECONDARY_BAD:
return (DB_STR("0088",
"DB_SECONDARY_BAD: Secondary index inconsistent with primary"));
case DB_TIMEOUT:
return (DB_STR("0089", "DB_TIMEOUT: Operation timed out"));
case DB_VERIFY_BAD:
return (DB_STR("0090",
"DB_VERIFY_BAD: Database verification failed"));
case DB_VERSION_MISMATCH:
return (DB_STR("0091",
"DB_VERSION_MISMATCH: Database environment version mismatch"));
default:
break;
}
return (__db_unknown_error(error));
}
char *
__db_unknown_error(error)
int error;
{
(void)snprintf(DB_GLOBAL(error_buf),
sizeof(DB_GLOBAL(error_buf)), DB_STR_A("0092",
"Unknown error: %d", "%d"), error);
return (DB_GLOBAL(error_buf));
}
void
#ifdef STDC_HEADERS
__db_syserr(const ENV *env, int error, const char *fmt, ...)
#else
__db_syserr(env, error, fmt, va_alist)
const ENV *env;
int error;
const char *fmt;
va_dcl
#endif
{
DB_ENV *dbenv;
dbenv = env == NULL ? NULL : env->dbenv;
DB_REAL_ERR(dbenv, error, DB_ERROR_SYSTEM, 0, fmt);
}
void
#ifdef STDC_HEADERS
__db_err(const ENV *env, int error, const char *fmt, ...)
#else
__db_err(env, error, fmt, va_alist)
const ENV *env;
int error;
const char *fmt;
va_dcl
#endif
{
DB_ENV *dbenv;
dbenv = env == NULL ? NULL : env->dbenv;
DB_REAL_ERR(dbenv, error, DB_ERROR_SET, 0, fmt);
}
void
#ifdef STDC_HEADERS
__db_errx(const ENV *env, const char *fmt, ...)
#else
__db_errx(env, fmt, va_alist)
const ENV *env;
const char *fmt;
va_dcl
#endif
{
DB_ENV *dbenv;
dbenv = env == NULL ? NULL : env->dbenv;
DB_REAL_ERR(dbenv, 0, DB_ERROR_NOT_SET, 0, fmt);
}
void
__db_errcall(dbenv, error, error_set, fmt, ap)
const DB_ENV *dbenv;
int error;
db_error_set_t error_set;
const char *fmt;
va_list ap;
{
char *p;
char buf[2048];
char sysbuf[1024];
p = buf;
if (fmt != NULL)
p += vsnprintf(buf, sizeof(buf), fmt, ap);
if (error_set != DB_ERROR_NOT_SET)
p += snprintf(p,
sizeof(buf) - (size_t)(p - buf), ": %s",
error_set == DB_ERROR_SET ? db_strerror(error) :
__os_strerror(error, sysbuf, sizeof(sysbuf)));
dbenv->db_errcall(dbenv, dbenv->db_errpfx, buf);
}
void
__db_errfile(dbenv, error, error_set, fmt, ap)
const DB_ENV *dbenv;
int error;
db_error_set_t error_set;
const char *fmt;
va_list ap;
{
FILE *fp;
int need_sep;
char sysbuf[1024];
fp = dbenv == NULL ||
dbenv->db_errfile == NULL ? stderr : dbenv->db_errfile;
need_sep = 0;
if (dbenv != NULL && dbenv->db_errpfx != NULL) {
(void)fprintf(fp, "%s", dbenv->db_errpfx);
need_sep = 1;
}
if (fmt != NULL && fmt[0] != '\0') {
if (need_sep)
(void)fprintf(fp, ": ");
need_sep = 1;
(void)vfprintf(fp, fmt, ap);
}
if (error_set != DB_ERROR_NOT_SET)
(void)fprintf(fp, "%s%s",
need_sep ? ": " : "",
error_set == DB_ERROR_SET ? db_strerror(error) :
__os_strerror(error, sysbuf, sizeof(sysbuf)));
(void)fprintf(fp, "\n");
(void)fflush(fp);
}
void
#ifdef STDC_HEADERS
__db_msgadd(ENV *env, DB_MSGBUF *mbp, const char *fmt, ...)
#else
__db_msgadd(env, mbp, fmt, va_alist)
ENV *env;
DB_MSGBUF *mbp;
const char *fmt;
va_dcl
#endif
{
va_list ap;
#ifdef STDC_HEADERS
va_start(ap, fmt);
#else
va_start(ap);
#endif
__db_msgadd_ap(env, mbp, fmt, ap);
va_end(ap);
}
void
__db_msgadd_ap(env, mbp, fmt, ap)
ENV *env;
DB_MSGBUF *mbp;
const char *fmt;
va_list ap;
{
size_t len, olen;
char buf[2048];
len = (size_t)vsnprintf(buf, sizeof(buf), fmt, ap);
olen = (size_t)(mbp->cur - mbp->buf);
if (olen + len >= mbp->len) {
if (__os_realloc(env, mbp->len + len + 256, &mbp->buf))
return;
mbp->len += (len + 256);
mbp->cur = mbp->buf + olen;
}
memcpy(mbp->cur, buf, len + 1);
mbp->cur += len;
}
void
#ifdef STDC_HEADERS
__db_msg(const ENV *env, const char *fmt, ...)
#else
__db_msg(env, fmt, va_alist)
const ENV *env;
const char *fmt;
va_dcl
#endif
{
DB_ENV *dbenv;
dbenv = env == NULL ? NULL : env->dbenv;
DB_REAL_MSG(dbenv, fmt);
}
void
#ifdef STDC_HEADERS
__db_repmsg(const ENV *env, const char *fmt, ...)
#else
__db_repmsg(env, fmt, va_alist)
const ENV *env;
const char *fmt;
va_dcl
#endif
{
va_list ap;
char buf[2048];
#ifdef STDC_HEADERS
va_start(ap, fmt);
#else
va_start(ap);
#endif
(void)vsnprintf(buf, sizeof(buf), fmt, ap);
__rep_msg(env, buf);
va_end(ap);
}
static void
__db_msgcall(dbenv, fmt, ap)
const DB_ENV *dbenv;
const char *fmt;
va_list ap;
{
char buf[2048];
(void)vsnprintf(buf, sizeof(buf), fmt, ap);
dbenv->db_msgcall(dbenv, buf);
}
static void
__db_msgfile(dbenv, fmt, ap)
const DB_ENV *dbenv;
const char *fmt;
va_list ap;
{
FILE *fp;
fp = dbenv == NULL ||
dbenv->db_msgfile == NULL ? stdout : dbenv->db_msgfile;
(void)vfprintf(fp, fmt, ap);
(void)fprintf(fp, "\n");
(void)fflush(fp);
}
int
__db_unknown_flag(env, routine, flag)
ENV *env;
char *routine;
u_int32_t flag;
{
__db_errx(env, DB_STR_A("0093", "%s: Unknown flag: %#x", "%s %#x"),
routine, (u_int)flag);
#ifdef DIAGNOSTIC
__os_abort(env);
#endif
return (EINVAL);
}
int
__db_unknown_type(env, routine, type)
ENV *env;
char *routine;
DBTYPE type;
{
__db_errx(env, DB_STR_A("0094", "%s: Unexpected database type: %s",
"%s %s"), routine, __db_dbtype_to_string(type));
#ifdef DIAGNOSTIC
__os_abort(env);
#endif
return (EINVAL);
}
int
__db_unknown_path(env, routine)
ENV *env;
char *routine;
{
__db_errx(env, DB_STR_A("0095", "%s: Unexpected code path error",
"%s"), routine);
#ifdef DIAGNOSTIC
__os_abort(env);
#endif
return (EINVAL);
}
int
__db_check_txn(dbp, txn, assoc_locker, read_op)
DB *dbp;
DB_TXN *txn;
DB_LOCKER *assoc_locker;
int read_op;
{
ENV *env;
int related, ret;
env = dbp->env;
if (IS_RECOVERING(env) || F_ISSET(dbp, DB_AM_RECOVER))
return (0);
if (!read_op && txn != NULL && F_ISSET(txn, TXN_READONLY)) {
__db_errx(env, DB_STR("0096",
"Read-only transaction cannot be used for an update"));
return (EINVAL);
} else if (txn == NULL || F_ISSET(txn, TXN_PRIVATE)) {
if (dbp->cur_locker != NULL &&
dbp->cur_locker->id >= TXN_MINIMUM)
goto open_err;
if (!read_op && F_ISSET(dbp, DB_AM_TXN)) {
__db_errx(env, DB_STR("0097",
"Transaction not specified for a transactional database"));
return (EINVAL);
}
} else if (F_ISSET(txn, TXN_FAMILY)) {
return (0);
} else {
if (!TXN_ON(env))
return (__db_not_txn_env(env));
if (!F_ISSET(dbp, DB_AM_TXN)) {
__db_errx(env, DB_STR("0098",
"Transaction specified for a non-transactional database"));
return (EINVAL);
}
if (F_ISSET(txn, TXN_DEADLOCK))
return (__db_txn_deadlock_err(env, txn));
if (dbp->cur_locker != NULL &&
dbp->cur_locker->id >= TXN_MINIMUM &&
dbp->cur_locker->id != txn->txnid) {
if ((ret = __lock_locker_same_family(env,
dbp->cur_locker, txn->locker, &related)) != 0)
return (ret);
if (!related)
goto open_err;
}
}
if (!read_op && dbp->associate_locker != NULL &&
txn != NULL && dbp->associate_locker != assoc_locker) {
__db_errx(env, DB_STR("0099",
"Operation forbidden while secondary index is being created"));
return (EINVAL);
}
if (txn != NULL && env != txn->mgrp->env) {
__db_errx(env, DB_STR("0100",
"Transaction and database from different environments"));
return (EINVAL);
}
return (0);
open_err:
if (F2_ISSET(dbp, DB2_AM_EXCL))
__db_errx(env, DB_STR("0209",
"Exclusive database handles can only have one active transaction at a time."));
else
__db_errx(env, DB_STR("0101",
"Transaction that opened the DB handle is still active"));
return (EINVAL);
}
int
__db_txn_deadlock_err(env, txn)
ENV *env;
DB_TXN *txn;
{
const char *name;
name = NULL;
(void)__txn_get_name(txn, &name);
__db_errx(env, DB_STR_A("0102",
"%s%sprevious transaction deadlock return not resolved",
"%s %s"), name == NULL ? "" : name, name == NULL ? "" : ": ");
return (EINVAL);
}
int
__db_not_txn_env(env)
ENV *env;
{
__db_errx(env, DB_STR("0103",
"DB environment not configured for transactions"));
return (EINVAL);
}
int
__db_rec_toobig(env, data_len, fixed_rec_len)
ENV *env;
u_int32_t data_len, fixed_rec_len;
{
__db_errx(env, DB_STR_A("0104",
"%lu larger than database's maximum record length %lu",
"%lu %lu"), (u_long)data_len, (u_long)fixed_rec_len);
return (EINVAL);
}
int
__db_rec_repl(env, data_size, data_dlen)
ENV *env;
u_int32_t data_size, data_dlen;
{
__db_errx(env, DB_STR_A("0105",
"Record length error: "
"replacement length %lu differs from replaced length %lu",
"%lu %lu"), (u_long)data_size, (u_long)data_dlen);
return (EINVAL);
}
#if defined(DIAGNOSTIC) || defined(DEBUG_ROP) || defined(DEBUG_WOP)
int
__dbc_logging(dbc)
DBC *dbc;
{
DB_REP *db_rep;
ENV *env;
int ret;
env = dbc->env;
db_rep = env->rep_handle;
ret = LOGGING_ON(env) &&
!F_ISSET(dbc, DBC_RECOVER) && !IS_REP_CLIENT(env);
if (db_rep == NULL || F_ISSET(dbc, DBC_RECOVER))
return (ret);
#ifndef DEBUG_ROP
{
REP *rep;
rep = db_rep->region;
if (IS_REP_CLIENT(env) && !F_ISSET(dbc->dbp, DB_AM_NOT_DURABLE)) {
__db_errx(env, DB_STR("0106",
"dbc_logging: Client update"));
goto err;
}
#ifndef DEBUG_WOP
if (IS_REP_MASTER(env) &&
dbc->txn == NULL && !F_ISSET(dbc->dbp, DB_AM_NOT_DURABLE)) {
__db_errx(env, DB_STR("0107",
"Dbc_logging: Master non-txn update"));
goto err;
}
#endif
if (0) {
err: __db_errx(env, DB_STR_A("0108", "Rep: flags 0x%lx msg_th %lu",
"%lx %lu"), (u_long)rep->flags, (u_long)rep->msg_th);
__db_errx(env, DB_STR_A("0109", "Rep: handle %lu, opcnt %lu",
"%lu %lu"), (u_long)rep->handle_cnt, (u_long)rep->op_cnt);
__os_abort(env);
}
}
#endif
return (ret);
}
#endif
int
__db_check_lsn(env, lsn, prev)
ENV *env;
DB_LSN *lsn, *prev;
{
__db_errx(env, DB_STR_A("0110",
"Log sequence error: page LSN %lu %lu; previous LSN %lu %lu",
"%lu %lu %lu %lu"), (u_long)(lsn)->file,
(u_long)(lsn)->offset, (u_long)(prev)->file,
(u_long)(prev)->offset);
return (EINVAL);
}
int
__db_rdonly(env, name)
const ENV *env;
const char *name;
{
__db_errx(env, DB_STR_A("0111",
"%s: attempt to modify a read-only database", "%s"), name);
return (EACCES);
}
int
__db_space_err(dbp)
const DB *dbp;
{
__db_errx(dbp->env, DB_STR_A("0112",
"%s: file limited to %lu pages", "%s %lu"),
dbp->fname, (u_long)dbp->mpf->mfp->maxpgno);
return (ENOSPC);
}
int
__db_failed(env, msg, pid, tid)
const ENV *env;
const char *msg;
pid_t pid;
db_threadid_t tid;
{
DB_ENV *dbenv;
char buf[DB_THREADID_STRLEN];
dbenv = env->dbenv;
__db_errx(env, DB_STR_A("0113", "Thread/process %s failed: %s",
"%s %s"), dbenv->thread_id_string(dbenv, pid, tid, buf), msg);
return (DB_RUNRECOVERY);
}