#include "db_config.h"
#include "db_int.h"
#include "dbinc/log.h"
#include "dbinc/txn.h"
static int __txn_init __P((ENV *, DB_TXNMGR *));
int
__txn_open(env)
ENV *env;
{
DB_TXNMGR *mgr;
DB_TXNREGION *region;
int ret;
if ((ret = __os_calloc(env, 1, sizeof(DB_TXNMGR), &mgr)) != 0)
return (ret);
TAILQ_INIT(&mgr->txn_chain);
mgr->env = env;
if ((ret = __env_region_share(env, &mgr->reginfo)) != 0)
goto err;
if (F_ISSET(&mgr->reginfo, REGION_CREATE))
if ((ret = __txn_init(env, mgr)) != 0)
goto err;
region = mgr->reginfo.primary =
R_ADDR(&mgr->reginfo,
((REGENV *)env->reginfo->primary)->tx_primary);
if ((ret = __mutex_alloc(
env, MTX_TXN_ACTIVE, DB_MUTEX_PROCESS_ONLY, &mgr->mutex)) != 0)
goto err;
mgr->reginfo.mtx_alloc = region->mtx_region;
env->tx_handle = mgr;
return (0);
err: env->tx_handle = NULL;
if (mgr->reginfo.addr != NULL)
(void)__env_region_detach(env, &mgr->reginfo, 0);
(void)__mutex_free(env, &mgr->mutex);
__os_free(env, mgr);
return (ret);
}
static int
__txn_init(env, mgr)
ENV *env;
DB_TXNMGR *mgr;
{
DB_ENV *dbenv;
DB_LSN last_ckp;
DB_TXNREGION *region;
int ret;
dbenv = env->dbenv;
ZERO_LSN(last_ckp);
if (LOGGING_ON(env)) {
if ((ret = __log_get_cached_ckp_lsn(env, &last_ckp)) != 0)
return (ret);
if (IS_ZERO_LSN(last_ckp) &&
(ret = __txn_findlastckp(env, &last_ckp, NULL)) != 0)
return (ret);
}
if ((ret = __env_alloc(&mgr->reginfo,
sizeof(DB_TXNREGION), &mgr->reginfo.primary)) != 0) {
__db_errx(env, DB_STR("4508",
"Unable to allocate memory for the transaction region"));
return (ret);
}
((REGENV *)env->reginfo->primary)->tx_primary =
R_OFFSET(&mgr->reginfo, mgr->reginfo.primary);
region = mgr->reginfo.primary;
memset(region, 0, sizeof(*region));
region->mtx_region = ((REGENV *)env->reginfo->primary)->mtx_regenv;
mgr->reginfo.mtx_alloc = region->mtx_region;
region->maxtxns = dbenv->tx_max;
region->inittxns = dbenv->tx_init;
region->last_txnid = TXN_MINIMUM;
region->cur_maxid = TXN_MAXIMUM;
if ((ret = __mutex_alloc(
env, MTX_TXN_CHKPT, 0, ®ion->mtx_ckp)) != 0)
return (ret);
region->last_ckp = last_ckp;
region->time_ckp = time(NULL);
memset(®ion->stat, 0, sizeof(region->stat));
#ifdef HAVE_STATISTICS
region->stat.st_maxtxns = region->maxtxns;
region->stat.st_inittxns = region->inittxns;
#endif
SH_TAILQ_INIT(®ion->active_txn);
SH_TAILQ_INIT(®ion->mvcc_txn);
return (ret);
}
int
__txn_findlastckp(env, lsnp, max_lsn)
ENV *env;
DB_LSN *lsnp;
DB_LSN *max_lsn;
{
DBT dbt;
DB_LOGC *logc;
DB_LSN lsn;
int ret, t_ret;
u_int32_t rectype;
ZERO_LSN(*lsnp);
if ((ret = __log_cursor(env, &logc)) != 0)
return (ret);
memset(&dbt, 0, sizeof(dbt));
if (max_lsn != NULL) {
lsn = *max_lsn;
if ((ret = __logc_get(logc, &lsn, &dbt, DB_SET)) != 0)
goto err;
} else {
if ((ret = __logc_get(logc, &lsn, &dbt, DB_LAST)) != 0)
goto err;
lsn.offset = 0;
}
while ((ret = __logc_get(logc, &lsn, &dbt, DB_PREV)) == 0) {
if (dbt.size < sizeof(u_int32_t))
continue;
LOGCOPY_32(env, &rectype, dbt.data);
if (rectype == DB___txn_ckp) {
*lsnp = lsn;
break;
}
}
err: if ((t_ret = __logc_close(logc)) != 0 && ret == 0)
ret = t_ret;
return ((ret == 0 || ret == DB_NOTFOUND) ? 0 : ret);
}
int
__txn_env_refresh(env)
ENV *env;
{
DB_TXN *txn;
DB_TXNMGR *mgr;
REGINFO *reginfo;
u_int32_t txnid;
int aborted, ret, t_ret;
ret = 0;
mgr = env->tx_handle;
reginfo = &mgr->reginfo;
aborted = 0;
if (TAILQ_FIRST(&mgr->txn_chain) != NULL) {
while ((txn = TAILQ_FIRST(&mgr->txn_chain)) != NULL) {
txnid = txn->txnid;
if (((TXN_DETAIL *)txn->td)->status == TXN_PREPARED) {
if ((ret = __txn_discard_int(txn, 0)) != 0) {
__db_err(env, ret, DB_STR_A("4509",
"unable to discard txn %#lx",
"%#lx"), (u_long)txnid);
break;
}
continue;
}
aborted = 1;
if ((t_ret = __txn_abort(txn)) != 0) {
__db_err(env, t_ret, DB_STR_A("4510",
"unable to abort transaction %#lx", "%#lx"),
(u_long)txnid);
ret = __env_panic(env, t_ret);
break;
}
}
if (aborted) {
__db_errx(env, DB_STR("4511",
"Error: closing the transaction region with active transactions"));
if (ret == 0)
ret = EINVAL;
}
}
if ((t_ret = __mutex_free(env, &mgr->mutex)) != 0 && ret == 0)
ret = t_ret;
if (F_ISSET(env, ENV_PRIVATE))
reginfo->mtx_alloc = MUTEX_INVALID;
if ((t_ret = __env_region_detach(env, reginfo, 0)) != 0 && ret == 0)
ret = t_ret;
__os_free(env, mgr);
env->tx_handle = NULL;
return (ret);
}
u_int32_t
__txn_region_mutex_count(env)
ENV *env;
{
COMPQUIET(env, NULL);
return (1 + 2);
}
u_int32_t
__txn_region_mutex_max(env)
ENV *env;
{
DB_ENV *dbenv;
u_int32_t count;
dbenv = env->dbenv;
if ((count = dbenv->tx_max) == 0)
count = DEF_MAX_TXNS;
return (count > dbenv->tx_init ? count - dbenv->tx_init : 0);
}
size_t
__txn_region_size(env)
ENV *env;
{
DB_ENV *dbenv;
size_t s;
dbenv = env->dbenv;
s = sizeof(DB_TXNREGION) + dbenv->tx_init *
(sizeof(TXN_DETAIL) + __env_alloc_overhead() + 20) + 10 * 1024;
return (s);
}
size_t
__txn_region_max(env)
ENV *env;
{
DB_ENV *dbenv;
size_t s;
u_int32_t count;
dbenv = env->dbenv;
if ((count = dbenv->tx_max) == 0)
count = DEF_MAX_TXNS;
if (count <= dbenv->tx_init)
return (0);
s = (count - dbenv->tx_init) *
(sizeof(TXN_DETAIL) + __env_alloc_overhead() + 20);
return (s);
}
int
__txn_id_set(env, cur_txnid, max_txnid)
ENV *env;
u_int32_t cur_txnid, max_txnid;
{
DB_TXNMGR *mgr;
DB_TXNREGION *region;
int ret;
ENV_REQUIRES_CONFIG(env, env->tx_handle, "txn_id_set", DB_INIT_TXN);
mgr = env->tx_handle;
region = mgr->reginfo.primary;
region->last_txnid = cur_txnid;
region->cur_maxid = max_txnid;
ret = 0;
if (cur_txnid < TXN_MINIMUM) {
__db_errx(env, DB_STR_A("4512",
"Current ID value %lu below minimum", "%lu"),
(u_long)cur_txnid);
ret = EINVAL;
}
if (max_txnid < TXN_MINIMUM) {
__db_errx(env, DB_STR_A("4513",
"Maximum ID value %lu below minimum", "%lu"),
(u_long)max_txnid);
ret = EINVAL;
}
return (ret);
}
int
__txn_oldest_reader(env, lsnp)
ENV *env;
DB_LSN *lsnp;
{
DB_LSN old_lsn;
DB_TXNMGR *mgr;
DB_TXNREGION *region;
TXN_DETAIL *td;
int ret;
if ((mgr = env->tx_handle) == NULL)
return (0);
region = mgr->reginfo.primary;
if ((ret = __log_current_lsn_int(env, &old_lsn, NULL, NULL)) != 0)
return (ret);
TXN_SYSTEM_LOCK(env);
SH_TAILQ_FOREACH(td, ®ion->active_txn, links, __txn_detail)
if (LOG_COMPARE(&td->read_lsn, &old_lsn) < 0)
old_lsn = td->read_lsn;
*lsnp = old_lsn;
TXN_SYSTEM_UNLOCK(env);
return (0);
}
int
__txn_add_buffer(env, td)
ENV *env;
TXN_DETAIL *td;
{
DB_ASSERT(env, td != NULL);
MUTEX_LOCK(env, td->mvcc_mtx);
DB_ASSERT(env, td->mvcc_ref < UINT32_MAX);
++td->mvcc_ref;
MUTEX_UNLOCK(env, td->mvcc_mtx);
COMPQUIET(env, NULL);
return (0);
}
int
__txn_remove_buffer(env, td, hash_mtx)
ENV *env;
TXN_DETAIL *td;
db_mutex_t hash_mtx;
{
DB_TXNMGR *mgr;
DB_TXNREGION *region;
int need_free, ret;
DB_ASSERT(env, td != NULL);
ret = 0;
mgr = env->tx_handle;
region = mgr->reginfo.primary;
MUTEX_LOCK(env, td->mvcc_mtx);
DB_ASSERT(env, td->mvcc_ref > 0);
need_free = (--td->mvcc_ref == 0) && F_ISSET(td, TXN_DTL_SNAPSHOT);
MUTEX_UNLOCK(env, td->mvcc_mtx);
if (need_free) {
MUTEX_UNLOCK(env, hash_mtx);
ret = __mutex_free(env, &td->mvcc_mtx);
td->mvcc_mtx = MUTEX_INVALID;
TXN_SYSTEM_LOCK(env);
SH_TAILQ_REMOVE(®ion->mvcc_txn, td, links, __txn_detail);
STAT_DEC(env,
txn, nsnapshot, region->stat.st_nsnapshot, td->txnid);
__env_alloc_free(&mgr->reginfo, td);
TXN_SYSTEM_UNLOCK(env);
MUTEX_READLOCK(env, hash_mtx);
}
return (ret);
}