Files
cpython-source-deps/db/db_cds.c
2017-09-04 13:40:25 -05:00

178 lines
3.8 KiB
C

/*-
* See the file LICENSE for redistribution information.
*
* Copyright (c) 2000,2008 Oracle. All rights reserved.
*
* $Id: db_cds.c 63573 2008-05-23 21:43:21Z trent.nelson $
*/
#include "db_config.h"
#include "db_int.h"
#include "dbinc/db_page.h"
#include "dbinc/db_am.h"
#include "dbinc/lock.h"
#include "dbinc/txn.h"
static int __cdsgroup_abort __P((DB_TXN *txn));
static int __cdsgroup_commit __P((DB_TXN *txn, u_int32_t flags));
static int __cdsgroup_discard __P((DB_TXN *txn, u_int32_t flags));
static u_int32_t __cdsgroup_id __P((DB_TXN *txn));
static int __cdsgroup_notsup __P((ENV *env, const char *meth));
static int __cdsgroup_prepare __P((DB_TXN *txn, u_int8_t *gid));
static int __cdsgroup_set_name __P((DB_TXN *txn, const char *name));
static int __cdsgroup_set_timeout
__P((DB_TXN *txn, db_timeout_t timeout, u_int32_t flags));
/*
* __cdsgroup_notsup --
* Error when CDS groups don't support a method.
*/
static int
__cdsgroup_notsup(env, meth)
ENV *env;
const char *meth;
{
__db_errx(env, "CDS groups do not support %s", meth);
return (DB_OPNOTSUP);
}
static int
__cdsgroup_abort(txn)
DB_TXN *txn;
{
return (__cdsgroup_notsup(txn->mgrp->env, "abort"));
}
static int
__cdsgroup_commit(txn, flags)
DB_TXN *txn;
u_int32_t flags;
{
DB_LOCKER *locker;
DB_LOCKREQ lreq;
ENV *env;
int ret, t_ret;
COMPQUIET(flags, 0);
env = txn->mgrp->env;
/* Check for live cursors. */
if (txn->cursors != 0) {
__db_errx(env, "CDS group has active cursors");
return (EINVAL);
}
/* We may be holding handle locks; release them. */
lreq.op = DB_LOCK_PUT_ALL;
lreq.obj = NULL;
ret = __lock_vec(env, txn->locker, 0, &lreq, 1, NULL);
env = txn->mgrp->env;
locker = txn->locker;
__os_free(env, txn->mgrp);
__os_free(env, txn);
if ((t_ret = __lock_id_free(env, locker)) != 0 && ret == 0)
ret = t_ret;
return (ret);
}
static int __cdsgroup_discard(txn, flags)
DB_TXN *txn;
u_int32_t flags;
{
COMPQUIET(flags, 0);
return (__cdsgroup_notsup(txn->mgrp->env, "discard"));
}
static u_int32_t __cdsgroup_id(txn)
DB_TXN *txn;
{
return (txn->txnid);
}
static int __cdsgroup_prepare(txn, gid)
DB_TXN *txn;
u_int8_t *gid;
{
COMPQUIET(gid, NULL);
return (__cdsgroup_notsup(txn->mgrp->env, "prepare"));
}
static int __cdsgroup_set_name(txn, name)
DB_TXN *txn;
const char *name;
{
COMPQUIET(name, NULL);
return (__cdsgroup_notsup(txn->mgrp->env, "set_name"));
}
static int __cdsgroup_set_timeout(txn, timeout, flags)
DB_TXN *txn;
db_timeout_t timeout;
u_int32_t flags;
{
COMPQUIET(timeout, 0);
COMPQUIET(flags, 0);
return (__cdsgroup_notsup(txn->mgrp->env, "set_timeout"));
}
/*
* __cds_txn_begin --
* ENV->cdsgroup_begin
*
* PUBLIC: int __cdsgroup_begin __P((DB_ENV *, DB_TXN **));
*/
int
__cdsgroup_begin(dbenv, txnpp)
DB_ENV *dbenv;
DB_TXN **txnpp;
{
DB_THREAD_INFO *ip;
DB_TXN *txn;
ENV *env;
int ret;
env = dbenv->env;
ENV_ILLEGAL_BEFORE_OPEN(env, "cdsgroup_begin");
if (!CDB_LOCKING(env))
return (__env_not_config(env, "cdsgroup_begin", DB_INIT_CDB));
ENV_ENTER(env, ip);
*txnpp = txn = NULL;
if ((ret = __os_calloc(env, 1, sizeof(DB_TXN), &txn)) != 0)
goto err;
/*
* We need a dummy DB_TXNMGR -- it's the only way to get from a
* transaction handle to the environment handle.
*/
if ((ret = __os_calloc(env, 1, sizeof(DB_TXNMGR), &txn->mgrp)) != 0)
goto err;
txn->mgrp->env = env;
if ((ret = __lock_id(env, &txn->txnid, &txn->locker)) != 0)
goto err;
txn->flags = TXN_CDSGROUP;
txn->abort = __cdsgroup_abort;
txn->commit = __cdsgroup_commit;
txn->discard = __cdsgroup_discard;
txn->id = __cdsgroup_id;
txn->prepare = __cdsgroup_prepare;
txn->set_name = __cdsgroup_set_name;
txn->set_timeout = __cdsgroup_set_timeout;
*txnpp = txn;
if (0) {
err: if (txn != NULL) {
if (txn->mgrp != NULL)
__os_free(env, txn->mgrp);
__os_free(env, txn);
}
}
ENV_LEAVE(env, ip);
return (ret);
}