#include "db_config.h"
#include "db_int.h"
#include "dbinc/log.h"
#include "dbinc/mp.h"
#include "dbinc/txn.h"
int
__memp_dirty(dbmfp, addrp, ip, txn, priority, flags)
DB_MPOOLFILE *dbmfp;
void *addrp;
DB_THREAD_INFO *ip;
DB_TXN *txn;
DB_CACHE_PRIORITY priority;
u_int32_t flags;
{
BH *bhp;
DB_MPOOL *dbmp;
DB_MPOOL_HASH *hp;
DB_TXN *ancestor;
ENV *env;
MPOOL *c_mp;
#ifdef DIAG_MVCC
MPOOLFILE *mfp;
#endif
REGINFO *infop;
int mvcc, ret;
db_pgno_t pgno;
void *pgaddr;
env = dbmfp->env;
dbmp = env->mp_handle;
mvcc = atomic_read(&dbmfp->mfp->multiversion);
pgaddr = *(void **)addrp;
bhp = (BH *)((u_int8_t *)pgaddr - SSZA(BH, buf));
pgno = bhp->pgno;
if (F_ISSET(bhp, BH_EXCLUSIVE)) {
DB_ASSERT(env, F_ISSET(bhp, BH_DIRTY));
return (0);
}
if (flags == 0)
flags = DB_MPOOL_DIRTY;
DB_ASSERT(env, flags == DB_MPOOL_DIRTY || flags == DB_MPOOL_EDIT);
if (F_ISSET(dbmfp, MP_READONLY)) {
__db_errx(env, DB_STR_A("3008",
"%s: dirty flag set for readonly file page", "%s"),
__memp_fn(dbmfp));
return (EACCES);
}
for (ancestor = txn;
ancestor != NULL && ancestor->parent != NULL;
ancestor = ancestor->parent)
;
if (mvcc && txn != NULL && flags == DB_MPOOL_DIRTY &&
(!BH_OWNED_BY(env, bhp, ancestor) || SH_CHAIN_HASNEXT(bhp, vc))) {
atomic_inc(env, &bhp->ref);
*(void **)addrp = NULL;
if ((ret = __memp_fput(dbmfp, ip, pgaddr, priority)) != 0) {
__db_errx(env, DB_STR_A("3009",
"%s: error releasing a read-only page", "%s"),
__memp_fn(dbmfp));
atomic_dec(env, &bhp->ref);
return (ret);
}
if ((ret = __memp_fget(dbmfp,
&pgno, ip, txn, flags, addrp)) != 0) {
if (ret != DB_LOCK_DEADLOCK)
__db_errx(env, DB_STR_A("3010",
"%s: error getting a page for writing",
"%s"), __memp_fn(dbmfp));
atomic_dec(env, &bhp->ref);
return (ret);
}
atomic_dec(env, &bhp->ref);
DB_ASSERT(env, *(void **)addrp != pgaddr ||
mvcc != atomic_read(&dbmfp->mfp->multiversion));
pgaddr = *(void **)addrp;
bhp = (BH *)((u_int8_t *)pgaddr - SSZA(BH, buf));
DB_ASSERT(env, pgno == bhp->pgno);
return (0);
}
infop = &dbmp->reginfo[bhp->region];
c_mp = infop->primary;
hp = R_ADDR(infop, c_mp->htab);
hp = &hp[bhp->bucket];
MUTEX_UNLOCK(env, bhp->mtx_buf);
MUTEX_LOCK(env, bhp->mtx_buf);
DB_ASSERT(env, !F_ISSET(bhp, BH_EXCLUSIVE));
F_SET(bhp, BH_EXCLUSIVE);
if (!F_ISSET(bhp, BH_DIRTY)) {
#ifdef DIAGNOSTIC
MUTEX_LOCK(env, hp->mtx_hash);
#endif
atomic_inc(env, &hp->hash_page_dirty);
F_SET(bhp, BH_DIRTY);
#ifdef DIAGNOSTIC
MUTEX_UNLOCK(env, hp->mtx_hash);
#endif
}
#ifdef DIAG_MVCC
mfp = R_ADDR(env->mp_handle->reginfo, bhp->mf_offset);
MVCC_MPROTECT(bhp->buf, mfp->pagesize, PROT_READ | PROT_WRITE);
#endif
DB_ASSERT(env, !F_ISSET(bhp, BH_DIRTY) ||
atomic_read(&hp->hash_page_dirty) != 0);
return (0);
}
int
__memp_shared(dbmfp, pgaddr)
DB_MPOOLFILE *dbmfp;
void *pgaddr;
{
BH *bhp;
ENV *env;
env = dbmfp->env;
bhp = (BH *)((u_int8_t *)pgaddr - SSZA(BH, buf));
if (F_ISSET(bhp, BH_DIRTY))
dbmfp->mfp->file_written = 1;
DB_ASSERT(env, F_ISSET(bhp, BH_EXCLUSIVE));
F_CLR(bhp, BH_EXCLUSIVE);
MUTEX_UNLOCK(env, bhp->mtx_buf);
MUTEX_READLOCK(env, bhp->mtx_buf);
return (0);
}