#include "db_config.h"
#include "db_int.h"
#include "dbinc/db_page.h"
#include "dbinc/btree.h"
#include "dbinc/lock.h"
#include "dbinc/mp.h"
int
__bam_rsearch(dbc, recnop, flags, stop, exactp)
DBC *dbc;
db_recno_t *recnop;
u_int32_t flags;
int stop, *exactp;
{
BINTERNAL *bi;
BTREE_CURSOR *cp;
DB *dbp;
DB_LOCK lock;
DB_MPOOLFILE *mpf;
ENV *env;
PAGE *h;
RINTERNAL *ri;
db_indx_t adjust, deloffset, indx, top;
db_lockmode_t lock_mode;
db_pgno_t pg;
db_recno_t recno, t_recno, total;
u_int32_t get_mode;
int ret, stack, t_ret;
if (F_ISSET(dbc, DBC_OPD))
LOCK_CHECK_OFF(dbc->thread_info);
dbp = dbc->dbp;
env = dbp->env;
mpf = dbp->mpf;
cp = (BTREE_CURSOR *)dbc->internal;
h = NULL;
ret = 0;
BT_STK_CLR(cp);
if ((ret = __bam_get_root(dbc, PGNO_INVALID, stop, flags, &stack)) != 0)
goto done;
lock_mode = cp->csp->lock_mode;
get_mode = lock_mode == DB_LOCK_WRITE ? DB_MPOOL_DIRTY : 0;
lock = cp->csp->lock;
h = cp->csp->page;
BT_STK_CLR(cp);
total = RE_NREC(h);
if (LF_ISSET(SR_APPEND)) {
*exactp = 0;
*recnop = recno = total + 1;
} else {
recno = *recnop;
if (recno <= total)
*exactp = 1;
else {
*exactp = 0;
if (!LF_ISSET(SR_PAST_EOF) || recno > total + 1) {
ret = __memp_fput(mpf,
dbc->thread_info, h, dbc->priority);
if ((t_ret =
__TLPUT(dbc, lock)) != 0 && ret == 0)
ret = t_ret;
if (ret == 0)
ret = DB_NOTFOUND;
goto done;
}
}
}
for (total = 0;;) {
switch (TYPE(h)) {
case P_LBTREE:
if (LF_ISSET(SR_MAX)) {
indx = NUM_ENT(h) - 2;
goto enter;
}
case P_LDUP:
if (LF_ISSET(SR_MAX)) {
indx = NUM_ENT(h) - 1;
goto enter;
}
recno -= total;
if (TYPE(h) == P_LBTREE) {
adjust = P_INDX;
deloffset = O_INDX;
} else {
adjust = O_INDX;
deloffset = 0;
}
for (t_recno = 0, indx = 0;; indx += adjust) {
if (indx >= NUM_ENT(h)) {
*exactp = 0;
if (!LF_ISSET(SR_PAST_EOF) ||
recno > t_recno + 1) {
ret = __memp_fput(mpf,
dbc->thread_info,
h, dbc->priority);
h = NULL;
if ((t_ret = __TLPUT(dbc,
lock)) != 0 && ret == 0)
ret = t_ret;
if (ret == 0)
ret = DB_NOTFOUND;
goto err;
}
}
if (!B_DISSET(GET_BKEYDATA(dbp, h,
indx + deloffset)->type) &&
++t_recno == recno)
break;
}
BT_STK_ENTER(env, cp, h, indx, lock, lock_mode, ret);
if (ret != 0)
goto err;
if (LF_ISSET(SR_BOTH))
goto get_prev;
goto done;
case P_IBTREE:
if (LF_ISSET(SR_MAX)) {
indx = NUM_ENT(h);
bi = GET_BINTERNAL(dbp, h, indx - 1);
} else for (indx = 0, top = NUM_ENT(h);;) {
bi = GET_BINTERNAL(dbp, h, indx);
if (++indx == top || total + bi->nrecs >= recno)
break;
total += bi->nrecs;
}
pg = bi->pgno;
break;
case P_LRECNO:
if (LF_ISSET(SR_MAX))
recno = NUM_ENT(h);
else
recno -= total;
--recno;
enter: BT_STK_ENTER(env, cp, h, recno, lock, lock_mode, ret);
if (ret != 0)
goto err;
if (LF_ISSET(SR_BOTH)) {
get_prev: DB_ASSERT(env, LF_ISSET(SR_NEXT));
cp->csp++;
indx = cp->sp->indx - 1;
h = cp->sp->page;
if (TYPE(h) == P_IRECNO) {
ri = GET_RINTERNAL(dbp, h, indx);
pg = ri->pgno;
} else {
DB_ASSERT(env, TYPE(h) == P_IBTREE);
bi = GET_BINTERNAL(dbp, h, indx);
pg = bi->pgno;
}
LF_CLR(SR_NEXT | SR_BOTH);
LF_SET(SR_MAX);
stack = 1;
h = NULL;
goto lock_next;
}
goto done;
case P_IRECNO:
if (LF_ISSET(SR_MAX)) {
indx = NUM_ENT(h);
ri = GET_RINTERNAL(dbp, h, indx - 1);
} else for (indx = 0, top = NUM_ENT(h);;) {
ri = GET_RINTERNAL(dbp, h, indx);
if (++indx == top || total + ri->nrecs >= recno)
break;
total += ri->nrecs;
}
pg = ri->pgno;
break;
default:
ret = __db_pgfmt(env, h->pgno);
goto done;
}
--indx;
if (stop == LEVEL(h)) {
BT_STK_ENTER(env, cp, h, indx, lock, lock_mode, ret);
if (ret != 0)
goto err;
goto done;
}
if (stack) {
BT_STK_PUSH(env, cp, h, indx, lock, lock_mode, ret);
if (ret != 0)
goto err;
h = NULL;
lock_mode = DB_LOCK_WRITE;
get_mode = DB_MPOOL_DIRTY;
if ((ret =
__db_lget(dbc, 0, pg, lock_mode, 0, &lock)) != 0)
goto err;
} else if (LF_ISSET(SR_NEXT)) {
if (indx != 0 && cp->sp->page != NULL) {
BT_STK_POP(cp);
if ((ret = __bam_stkrel(dbc, STK_NOLOCK)) != 0)
goto err;
}
BT_STK_PUSH(env, cp, h, indx, lock, lock_mode, ret);
h = NULL;
if (ret != 0)
goto err;
lock_next: if ((ret =
__db_lget(dbc, 0, pg, lock_mode, 0, &lock)) != 0)
goto err;
} else {
if ((LF_ISSET(SR_PARENT) &&
(u_int8_t)(stop + 1) >= (u_int8_t)(LEVEL(h) - 1)) ||
(LEVEL(h) - 1) == LEAFLEVEL)
stack = 1;
if ((ret = __memp_fput(mpf,
dbc->thread_info, h, dbc->priority)) != 0)
goto err;
h = NULL;
lock_mode = stack &&
LF_ISSET(SR_WRITE) ? DB_LOCK_WRITE : DB_LOCK_READ;
if (lock_mode == DB_LOCK_WRITE)
get_mode = DB_MPOOL_DIRTY;
if ((ret = __db_lget(dbc,
LCK_COUPLE_ALWAYS, pg, lock_mode, 0, &lock)) != 0) {
(void)__LPUT(dbc, lock);
goto err;
}
}
if ((ret = __memp_fget(mpf, &pg,
dbc->thread_info, dbc->txn, get_mode, &h)) != 0)
goto err;
}
err: if (h != NULL && (t_ret = __memp_fput(mpf,
dbc->thread_info, h, dbc->priority)) != 0 && ret == 0)
ret = t_ret;
BT_STK_POP(cp);
(void)__bam_stkrel(dbc, 0);
done:
if (F_ISSET(dbc, DBC_OPD))
LOCK_CHECK_ON(dbc->thread_info);
return (ret);
}
int
__bam_adjust(dbc, adjust)
DBC *dbc;
int32_t adjust;
{
BTREE_CURSOR *cp;
DB *dbp;
DB_MPOOLFILE *mpf;
EPG *epg;
PAGE *h;
db_pgno_t root_pgno;
int ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
cp = (BTREE_CURSOR *)dbc->internal;
root_pgno = BAM_ROOT_PGNO(dbc);
for (epg = cp->sp; epg <= cp->csp; ++epg) {
h = epg->page;
if (TYPE(h) == P_IBTREE || TYPE(h) == P_IRECNO) {
ret = __memp_dirty(mpf, &h,
dbc->thread_info, dbc->txn, dbc->priority, 0);
epg->page = h;
if (ret != 0)
return (ret);
if (DBC_LOGGING(dbc)) {
if ((ret = __bam_cadjust_log(dbp, dbc->txn,
&LSN(h), 0, PGNO(h), &LSN(h),
(u_int32_t)epg->indx, adjust,
PGNO(h) == root_pgno ?
CAD_UPDATEROOT : 0)) != 0)
return (ret);
} else
LSN_NOT_LOGGED(LSN(h));
if (TYPE(h) == P_IBTREE)
GET_BINTERNAL(dbp, h, epg->indx)->nrecs +=
adjust;
else
GET_RINTERNAL(dbp, h, epg->indx)->nrecs +=
adjust;
if (PGNO(h) == root_pgno)
RE_NREC_ADJ(h, adjust);
}
}
return (0);
}
int
__bam_nrecs(dbc, rep)
DBC *dbc;
db_recno_t *rep;
{
DB *dbp;
DB_LOCK lock;
DB_MPOOLFILE *mpf;
PAGE *h;
db_pgno_t pgno;
int ret, t_ret;
COMPQUIET(h, NULL);
dbp = dbc->dbp;
mpf = dbp->mpf;
LOCK_INIT(lock);
pgno = PGNO_INVALID;
BAM_GET_ROOT(dbc, pgno, h, 0, DB_LOCK_READ, lock, ret);
if (ret != 0)
goto err;
DB_ASSERT(dbp->env, h != NULL);
*rep = RE_NREC(h);
ret = __memp_fput(mpf, dbc->thread_info, h, dbc->priority);
err: if ((t_ret = __TLPUT(dbc, lock)) != 0 && ret == 0)
ret = t_ret;
return (ret);
}
db_recno_t
__bam_total(dbp, h)
DB *dbp;
PAGE *h;
{
db_recno_t nrecs;
db_indx_t indx, top;
nrecs = 0;
top = NUM_ENT(h);
switch (TYPE(h)) {
case P_LBTREE:
for (indx = 0; indx < top; indx += P_INDX)
if (!B_DISSET(
GET_BKEYDATA(dbp, h, indx + O_INDX)->type))
++nrecs;
break;
case P_LDUP:
for (indx = 0; indx < top; indx += O_INDX)
if (!B_DISSET(GET_BKEYDATA(dbp, h, indx)->type))
++nrecs;
break;
case P_IBTREE:
for (indx = 0; indx < top; indx += O_INDX)
nrecs += GET_BINTERNAL(dbp, h, indx)->nrecs;
break;
case P_LRECNO:
nrecs = NUM_ENT(h);
break;
case P_IRECNO:
for (indx = 0; indx < top; indx += O_INDX)
nrecs += GET_RINTERNAL(dbp, h, indx)->nrecs;
break;
}
return (nrecs);
}