#include "db_config.h"
#include "db_int.h"
#include "dbinc/db_page.h"
#include "dbinc/hash.h"
#include "dbinc/btree.h"
#include "dbinc/lock.h"
#include "dbinc/mp.h"
static int __hamc_delpg
__P((DBC *, db_pgno_t, db_pgno_t, u_int32_t, db_ham_mode, u_int32_t *));
static int __ham_getindex_sorted
__P((DBC *, PAGE *, const DBT *, u_int32_t, int *, db_indx_t *));
static int __ham_getindex_unsorted
__P((DBC *, PAGE *, const DBT *, int *, db_indx_t *));
static int __hamc_delpg_getorder
__P((DBC *, DBC *, u_int32_t *, db_pgno_t, u_int32_t, void *));
static int __hamc_delpg_setorder
__P((DBC *, DBC *, u_int32_t *, db_pgno_t, u_int32_t, void *));
int
__ham_item(dbc, mode, pgnop)
DBC *dbc;
db_lockmode_t mode;
db_pgno_t *pgnop;
{
DB *dbp;
HASH_CURSOR *hcp;
db_pgno_t next_pgno;
int ret;
dbp = dbc->dbp;
hcp = (HASH_CURSOR *)dbc->internal;
if (F_ISSET(hcp, H_DELETED)) {
__db_errx(dbp->env, DB_STR("1132",
"Attempt to return a deleted item"));
return (EINVAL);
}
F_CLR(hcp, H_OK | H_NOMORE);
if ((ret = __ham_get_cpage(dbc, mode)) != 0)
return (ret);
recheck:
if (hcp->seek_size != 0 && hcp->seek_found_page == PGNO_INVALID &&
hcp->seek_size < P_FREESPACE(dbp, hcp->page)) {
hcp->seek_found_page = hcp->pgno;
hcp->seek_found_indx = NDX_INVALID;
}
if (hcp->indx < NUM_ENT(hcp->page) &&
HPAGE_TYPE(dbp, hcp->page, H_DATAINDEX(hcp->indx)) == H_OFFDUP) {
memcpy(pgnop,
HOFFDUP_PGNO(H_PAIRDATA(dbp, hcp->page, hcp->indx)),
sizeof(db_pgno_t));
F_SET(hcp, H_OK);
return (0);
}
if (F_ISSET(hcp, H_ISDUP))
memcpy(&hcp->dup_len,
HKEYDATA_DATA(H_PAIRDATA(dbp, hcp->page, hcp->indx)) +
hcp->dup_off, sizeof(db_indx_t));
if (hcp->indx >= (db_indx_t)NUM_ENT(hcp->page)) {
if (NEXT_PGNO(hcp->page) == PGNO_INVALID) {
F_SET(hcp, H_NOMORE);
return (DB_NOTFOUND);
}
next_pgno = NEXT_PGNO(hcp->page);
hcp->indx = 0;
if ((ret = __ham_next_cpage(dbc, next_pgno)) != 0)
return (ret);
goto recheck;
}
F_SET(hcp, H_OK);
return (0);
}
int
__ham_item_reset(dbc)
DBC *dbc;
{
DB *dbp;
DB_MPOOLFILE *mpf;
HASH_CURSOR *hcp;
int ret, t_ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
ret = 0;
if (hcp->page != NULL) {
ret = __memp_fput(mpf,
dbc->thread_info, hcp->page, dbc->priority);
hcp->page = NULL;
}
if ((t_ret = __ham_item_init(dbc)) != 0 && ret == 0)
ret = t_ret;
return (ret);
}
int
__ham_item_init(dbc)
DBC *dbc;
{
HASH_CURSOR *hcp;
int ret;
hcp = (HASH_CURSOR *)dbc->internal;
ret = __TLPUT(dbc, hcp->lock);
hcp->bucket = BUCKET_INVALID;
hcp->lbucket = BUCKET_INVALID;
LOCK_INIT(hcp->lock);
hcp->lock_mode = DB_LOCK_NG;
hcp->dup_off = 0;
hcp->dup_len = 0;
hcp->dup_tlen = 0;
hcp->seek_size = 0;
hcp->seek_found_page = PGNO_INVALID;
hcp->seek_found_indx = NDX_INVALID;
hcp->flags = 0;
hcp->pgno = PGNO_INVALID;
hcp->indx = NDX_INVALID;
hcp->page = NULL;
return (ret);
}
int
__ham_item_last(dbc, mode, pgnop)
DBC *dbc;
db_lockmode_t mode;
db_pgno_t *pgnop;
{
HASH_CURSOR *hcp;
int ret;
hcp = (HASH_CURSOR *)dbc->internal;
if ((ret = __ham_item_reset(dbc)) != 0)
return (ret);
hcp->bucket = hcp->hdr->max_bucket;
hcp->pgno = BUCKET_TO_PAGE(hcp, hcp->bucket);
F_SET(hcp, H_OK);
return (__ham_item_prev(dbc, mode, pgnop));
}
int
__ham_item_first(dbc, mode, pgnop)
DBC *dbc;
db_lockmode_t mode;
db_pgno_t *pgnop;
{
HASH_CURSOR *hcp;
int ret;
hcp = (HASH_CURSOR *)dbc->internal;
if ((ret = __ham_item_reset(dbc)) != 0)
return (ret);
F_SET(hcp, H_OK);
hcp->bucket = 0;
hcp->pgno = BUCKET_TO_PAGE(hcp, hcp->bucket);
hcp->dup_off = 0;
return (__ham_item_next(dbc, mode, pgnop));
}
int
__ham_item_prev(dbc, mode, pgnop)
DBC *dbc;
db_lockmode_t mode;
db_pgno_t *pgnop;
{
DB *dbp;
HASH_CURSOR *hcp;
db_pgno_t next_pgno;
int ret;
hcp = (HASH_CURSOR *)dbc->internal;
dbp = dbc->dbp;
F_CLR(hcp, H_OK | H_NOMORE | H_DELETED);
if ((ret = __ham_get_cpage(dbc, mode)) != 0)
return (ret);
if (!F_ISSET(hcp, H_NEXT_NODUP) && F_ISSET(hcp, H_ISDUP)) {
if (HPAGE_TYPE(dbp, hcp->page, H_DATAINDEX(hcp->indx)) ==
H_OFFDUP) {
memcpy(pgnop,
HOFFDUP_PGNO(H_PAIRDATA(dbp, hcp->page, hcp->indx)),
sizeof(db_pgno_t));
F_SET(hcp, H_OK);
return (0);
}
if (hcp->dup_off != 0) {
memcpy(&hcp->dup_len, HKEYDATA_DATA(
H_PAIRDATA(dbp, hcp->page, hcp->indx))
+ hcp->dup_off - sizeof(db_indx_t),
sizeof(db_indx_t));
hcp->dup_off -=
DUP_SIZE(hcp->dup_len);
return (__ham_item(dbc, mode, pgnop));
}
}
if (F_ISSET(hcp, H_DUPONLY)) {
F_CLR(hcp, H_OK);
F_SET(hcp, H_NOMORE);
return (0);
} else
F_CLR(hcp, H_ISDUP);
if (hcp->indx == 0) {
hcp->pgno = PREV_PGNO(hcp->page);
if (hcp->pgno == PGNO_INVALID) {
F_SET(hcp, H_NOMORE);
return (DB_NOTFOUND);
} else if ((ret =
__ham_next_cpage(dbc, hcp->pgno)) != 0)
return (ret);
else
hcp->indx = NUM_ENT(hcp->page);
}
if (hcp->indx == NDX_INVALID) {
DB_ASSERT(dbp->env, hcp->page != NULL);
hcp->indx = NUM_ENT(hcp->page);
for (next_pgno = NEXT_PGNO(hcp->page);
next_pgno != PGNO_INVALID;
next_pgno = NEXT_PGNO(hcp->page)) {
if ((ret = __ham_next_cpage(dbc, next_pgno)) != 0)
return (ret);
hcp->indx = NUM_ENT(hcp->page);
}
if (hcp->indx == 0) {
F_SET(hcp, H_NOMORE);
return (DB_NOTFOUND);
}
}
hcp->indx -= 2;
return (__ham_item(dbc, mode, pgnop));
}
int
__ham_item_next(dbc, mode, pgnop)
DBC *dbc;
db_lockmode_t mode;
db_pgno_t *pgnop;
{
HASH_CURSOR *hcp;
int ret;
hcp = (HASH_CURSOR *)dbc->internal;
if ((ret = __ham_get_cpage(dbc, mode)) != 0)
return (ret);
if (F_ISSET(hcp, H_DELETED)) {
if (hcp->indx != NDX_INVALID &&
F_ISSET(hcp, H_ISDUP) &&
HPAGE_TYPE(dbc->dbp, hcp->page, H_DATAINDEX(hcp->indx))
== H_DUPLICATE && hcp->dup_tlen == hcp->dup_off) {
if (F_ISSET(hcp, H_DUPONLY)) {
F_CLR(hcp, H_OK);
F_SET(hcp, H_NOMORE);
return (0);
} else {
F_CLR(hcp, H_ISDUP);
hcp->indx += 2;
}
} else if (!F_ISSET(hcp, H_ISDUP) && F_ISSET(hcp, H_DUPONLY)) {
F_CLR(hcp, H_OK);
F_SET(hcp, H_NOMORE);
return (0);
} else if (F_ISSET(hcp, H_ISDUP) &&
F_ISSET(hcp, H_NEXT_NODUP)) {
F_CLR(hcp, H_ISDUP);
hcp->indx += 2;
}
F_CLR(hcp, H_DELETED);
} else if (hcp->indx == NDX_INVALID) {
hcp->indx = 0;
F_CLR(hcp, H_ISDUP);
} else if (F_ISSET(hcp, H_NEXT_NODUP)) {
hcp->indx += 2;
F_CLR(hcp, H_ISDUP);
} else if (F_ISSET(hcp, H_ISDUP) && hcp->dup_tlen != 0) {
if (hcp->dup_off + DUP_SIZE(hcp->dup_len) >=
hcp->dup_tlen && F_ISSET(hcp, H_DUPONLY)) {
F_CLR(hcp, H_OK);
F_SET(hcp, H_NOMORE);
return (0);
}
hcp->dup_off += DUP_SIZE(hcp->dup_len);
if (hcp->dup_off >= hcp->dup_tlen) {
F_CLR(hcp, H_ISDUP);
hcp->indx += 2;
}
} else if (F_ISSET(hcp, H_DUPONLY)) {
F_CLR(hcp, H_OK);
F_SET(hcp, H_NOMORE);
return (0);
} else {
hcp->indx += 2;
F_CLR(hcp, H_ISDUP);
}
ret = __ham_item(dbc, mode, pgnop);
return (ret);
}
int
__ham_insertpair(dbc, p, indxp, key_dbt, data_dbt, key_type, data_type)
DBC *dbc;
PAGE *p;
db_indx_t *indxp;
const DBT *key_dbt, *data_dbt;
u_int32_t key_type, data_type;
{
DB *dbp;
u_int16_t n, indx;
db_indx_t *inp;
u_int32_t ksize, dsize, increase, distance;
u_int8_t *offset;
int i;
dbp = dbc->dbp;
n = NUM_ENT(p);
inp = P_INP(dbp, p);
ksize = (key_type == H_OFFPAGE) ?
key_dbt->size : HKEYDATA_SIZE(key_dbt->size);
dsize = (data_type == H_OFFPAGE || data_type == H_OFFDUP) ?
data_dbt->size : HKEYDATA_SIZE(data_dbt->size);
increase = ksize + dsize;
DB_ASSERT(dbp->env, indxp != NULL && *indxp != NDX_INVALID);
DB_ASSERT(dbp->env,
P_FREESPACE(dbp, p) >= dsize + ksize + 2 * sizeof(db_indx_t));
indx = *indxp;
if (n == 0 || indx == n) {
inp[indx] = HOFFSET(p) - ksize;
inp[indx+1] = HOFFSET(p) - increase;
} else {
offset = (u_int8_t *)p + HOFFSET(p);
if (indx == 0)
distance = dbp->pgsize - HOFFSET(p);
else
distance = (u_int32_t)
(P_ENTRY(dbp, p, indx - 1) - offset);
memmove(offset - increase, offset, distance);
memmove(&inp[indx + 2], &inp[indx],
(n - indx) * sizeof(db_indx_t));
for (i = indx + 2; i < n + 2; i++)
inp[i] -= increase;
inp[indx] = (HOFFSET(p) - increase) + distance + dsize;
inp[indx + 1] = (HOFFSET(p) - increase) + distance;
}
HOFFSET(p) -= increase;
if (key_type == H_OFFPAGE)
memcpy(P_ENTRY(dbp, p, indx), key_dbt->data, key_dbt->size);
else
PUT_HKEYDATA(P_ENTRY(dbp, p, indx), key_dbt->data,
key_dbt->size, key_type);
if (data_type == H_OFFPAGE || data_type == H_OFFDUP)
memcpy(P_ENTRY(dbp, p, indx+1), data_dbt->data,
data_dbt->size);
else
PUT_HKEYDATA(P_ENTRY(dbp, p, indx+1), data_dbt->data,
data_dbt->size, data_type);
NUM_ENT(p) += 2;
return (0);
}
int
__ham_getindex(dbc, p, key, key_type, match, indx)
DBC *dbc;
PAGE *p;
const DBT *key;
u_int32_t key_type;
int *match;
db_indx_t *indx;
{
DB_ASSERT(dbc->env, NUM_ENT(p)%2 == 0 );
if (p->type == P_HASH_UNSORTED)
return (__ham_getindex_unsorted(dbc, p, key, match, indx));
else
return (__ham_getindex_sorted(dbc,
p, key, key_type, match, indx));
}
#undef min
#define min(a, b) (((a) < (b)) ? (a) : (b))
static int
__ham_getindex_unsorted(dbc, p, key, match, indx)
DBC *dbc;
PAGE *p;
const DBT *key;
int *match;
db_indx_t *indx;
{
DB *dbp;
DBT pg_dbt;
HASH *t;
db_pgno_t pgno;
int i, n, res, ret;
u_int32_t tlen;
u_int8_t *hk;
dbp = dbc->dbp;
n = NUM_ENT(p);
t = dbp->h_internal;
res = 1;
for (i = 0; i < n; i+=2) {
hk = H_PAIRKEY(dbp, p, i);
switch (HPAGE_PTYPE(hk)) {
case H_OFFPAGE:
memcpy(&tlen, HOFFPAGE_TLEN(hk), sizeof(u_int32_t));
if (tlen == key->size) {
memcpy(&pgno,
HOFFPAGE_PGNO(hk), sizeof(db_pgno_t));
if ((ret = __db_moff(dbc, key, pgno, tlen,
t->h_compare, &res)) != 0)
return (ret);
}
break;
case H_KEYDATA:
if (t->h_compare != NULL) {
DB_INIT_DBT(pg_dbt,
HKEYDATA_DATA(hk), key->size);
if (t->h_compare(
dbp, key, &pg_dbt) != 0)
break;
} else if (key->size ==
LEN_HKEY(dbp, p, dbp->pgsize, i))
res = memcmp(key->data, HKEYDATA_DATA(hk),
key->size);
break;
case H_DUPLICATE:
case H_OFFDUP:
default:
return (__db_pgfmt(dbp->env, PGNO(p)));
}
if (res == 0)
break;
}
*indx = i;
*match = (res == 0 ? 0 : 1);
return (0);
}
static int
__ham_getindex_sorted(dbc, p, key, key_type, match, indxp)
DBC *dbc;
PAGE *p;
const DBT *key;
u_int32_t key_type;
int *match;
db_indx_t *indxp;
{
DB *dbp;
DBT tmp_dbt;
HASH *t;
HOFFPAGE *offp;
db_indx_t indx;
db_pgno_t off_pgno, koff_pgno;
u_int32_t base, itemlen, lim, off_len;
u_int8_t *entry;
int res, ret;
void *data;
dbp = dbc->dbp;
DB_ASSERT(dbp->env, p->type == P_HASH );
t = dbp->h_internal;
res = indx = 0;
DB_BINARY_SEARCH_FOR(base, lim, NUM_ENT(p), 2) {
DB_BINARY_SEARCH_INCR(indx, base, lim, 2);
data = HKEYDATA_DATA(H_PAIRKEY(dbp, p, indx));
entry = P_ENTRY(dbp, p, indx);
if (*entry == H_OFFPAGE) {
offp = (HOFFPAGE*)P_ENTRY(dbp, p, indx);
(void)__ua_memcpy(&itemlen, HOFFPAGE_TLEN(offp),
sizeof(u_int32_t));
if (key_type == H_OFFPAGE) {
(void)__ua_memcpy(&koff_pgno,
HOFFPAGE_PGNO(key->data),
sizeof(db_pgno_t));
(void)__ua_memcpy(&off_pgno,
HOFFPAGE_PGNO(offp), sizeof(db_pgno_t));
if (koff_pgno == off_pgno)
res = 0;
else {
memset(&tmp_dbt, 0, sizeof(tmp_dbt));
tmp_dbt.size = HOFFPAGE_SIZE;
tmp_dbt.data = offp;
if ((ret = __db_coff(dbc, key, &tmp_dbt,
t->h_compare, &res)) != 0)
return (ret);
}
} else {
(void)__ua_memcpy(&off_pgno,
HOFFPAGE_PGNO(offp), sizeof(db_pgno_t));
if ((ret = __db_moff(dbc, key, off_pgno,
itemlen, t->h_compare, &res)) != 0)
return (ret);
}
} else {
itemlen = LEN_HKEYDATA(dbp, p, dbp->pgsize, indx);
if (key_type == H_OFFPAGE) {
tmp_dbt.data = data;
tmp_dbt.size = itemlen;
offp = (HOFFPAGE *)key->data;
(void)__ua_memcpy(&off_pgno,
HOFFPAGE_PGNO(offp), sizeof(db_pgno_t));
(void)__ua_memcpy(&off_len, HOFFPAGE_TLEN(offp),
sizeof(u_int32_t));
if ((ret = __db_moff(dbc, &tmp_dbt, off_pgno,
off_len, t->h_compare, &res)) != 0)
return (ret);
res = -res;
} else if (t->h_compare != NULL) {
DB_INIT_DBT(tmp_dbt, data, itemlen);
res = t->h_compare(dbp, key, &tmp_dbt);
} else {
if ((res = memcmp(key->data, data,
min(key->size, itemlen))) == 0)
res = itemlen > key->size ? 1 :
(itemlen < key->size ? -1 : 0);
}
}
if (res == 0) {
*indxp = indx;
*match = 0;
return (0);
} else if (res > 0)
DB_BINARY_SEARCH_SHIFT_BASE(indx, base, lim, 2);
}
if (res > 0)
indx += 2;
*indxp = indx;
*match = 1;
return (0);
}
int
__ham_verify_sorted_page (dbc, p)
DBC *dbc;
PAGE *p;
{
DB *dbp;
DBT prev_dbt, curr_dbt;
ENV *env;
HASH *t;
db_pgno_t tpgno;
u_int32_t curr_len, prev_len, tlen;
u_int16_t *indxp;
db_indx_t i, n;
int res, ret;
char *prev, *curr;
n = NUM_ENT(p);
dbp = dbc->dbp;
DB_ASSERT(dbp->env, n%2 == 0 );
env = dbp->env;
t = dbp->h_internal;
if (t->h_compare != NULL)
return (0);
prev = (char *)HKEYDATA_DATA(H_PAIRKEY(dbp, p, 0));
prev_len = LEN_HKEYDATA(dbp, p, dbp->pgsize, 0);
for (i = 2; i < n; i+=2) {
curr = (char *)HKEYDATA_DATA(H_PAIRKEY(dbp, p, i));
curr_len = LEN_HKEYDATA(dbp, p, dbp->pgsize, i);
if (HPAGE_TYPE(dbp, p, i-2) == H_OFFPAGE &&
HPAGE_TYPE(dbp, p, i) == H_OFFPAGE) {
memset(&prev_dbt, 0, sizeof(prev_dbt));
memset(&curr_dbt, 0, sizeof(curr_dbt));
prev_dbt.size = curr_dbt.size = HOFFPAGE_SIZE;
prev_dbt.data = H_PAIRKEY(dbp, p, i-2);
curr_dbt.data = H_PAIRKEY(dbp, p, i);
if ((ret = __db_coff(dbc,
&prev_dbt, &curr_dbt, t->h_compare, &res)) != 0)
return (ret);
} else if (HPAGE_TYPE(dbp, p, i-2) == H_OFFPAGE) {
memset(&curr_dbt, 0, sizeof(curr_dbt));
curr_dbt.size = curr_len;
curr_dbt.data = H_PAIRKEY(dbp, p, i);
memcpy(&tlen, HOFFPAGE_TLEN(H_PAIRKEY(dbp, p, i-2)),
sizeof(u_int32_t));
memcpy(&tpgno, HOFFPAGE_PGNO(H_PAIRKEY(dbp, p, i-2)),
sizeof(db_pgno_t));
if ((ret = __db_moff(dbc,
&curr_dbt, tpgno, tlen, t->h_compare, &res)) != 0)
return (ret);
} else if (HPAGE_TYPE(dbp, p, i) == H_OFFPAGE) {
memset(&prev_dbt, 0, sizeof(prev_dbt));
prev_dbt.size = prev_len;
prev_dbt.data = H_PAIRKEY(dbp, p, i);
memcpy(&tlen, HOFFPAGE_TLEN(H_PAIRKEY(dbp, p, i)),
sizeof(u_int32_t));
memcpy(&tpgno, HOFFPAGE_PGNO(H_PAIRKEY(dbp, p, i)),
sizeof(db_pgno_t));
if ((ret = __db_moff(dbc,
&prev_dbt, tpgno, tlen, t->h_compare, &res)) != 0)
return (ret);
} else
res = memcmp(prev, curr, min(curr_len, prev_len));
if (res == 0 && curr_len > prev_len)
res = 1;
else if (res == 0 && curr_len < prev_len)
res = -1;
if (res >= 0) {
__db_msg(env, "key1: %s, key2: %s, len: %lu\n",
(char *)prev, (char *)curr,
(u_long)min(curr_len, prev_len));
__db_msg(env, "curroffset %lu\n", (u_long)i);
__db_msg(env, "indexes: ");
for (i = 0; i < n; i++) {
indxp = P_INP(dbp, p) + i;
__db_msg(env, "%04X, ", *indxp);
}
__db_msg(env, "\n");
#ifdef HAVE_STATISTICS
if ((ret = __db_prpage(dbp, p, DB_PR_PAGE)) != 0)
return (ret);
#endif
DB_ASSERT(dbp->env, res < 0);
}
prev = curr;
prev_len = curr_len;
}
return (0);
}
int
__ham_sort_page_cursor(dbc, page)
DBC *dbc;
PAGE *page;
{
DB *dbp;
DBT page_dbt;
DB_LSN new_lsn;
HASH_CURSOR *hcp;
int ret;
dbp = dbc->dbp;
hcp = (HASH_CURSOR *)dbc->internal;
if (DBC_LOGGING(dbc)) {
page_dbt.size = dbp->pgsize;
page_dbt.data = page;
if ((ret = __ham_splitdata_log(dbp, dbc->txn,
&new_lsn, 0, SORTPAGE, PGNO(page),
&page_dbt, &LSN(page))) != 0)
return (ret);
} else
LSN_NOT_LOGGED(new_lsn);
LSN(page) = new_lsn;
hcp->seek_found_indx = NDX_INVALID;
hcp->seek_found_page = PGNO_INVALID;
return (__ham_sort_page(dbc, &hcp->split_buf, page));
}
int
__ham_sort_page(dbc, tmp_buf, page)
DBC *dbc;
PAGE **tmp_buf;
PAGE *page;
{
DB *dbp;
PAGE *temp_pagep;
db_indx_t i;
int ret;
dbp = dbc->dbp;
DB_ASSERT(dbp->env, page->type == P_HASH_UNSORTED);
ret = 0;
if (tmp_buf != NULL)
temp_pagep = *tmp_buf;
else if ((ret = __os_malloc(dbp->env, dbp->pgsize, &temp_pagep)) != 0)
return (ret);
memcpy(temp_pagep, page, dbp->pgsize);
P_INIT(page, dbp->pgsize,
page->pgno, page->prev_pgno, page->next_pgno, 0, P_HASH);
for (i = 0; i < NUM_ENT(temp_pagep); i += 2)
if ((ret =
__ham_copypair(dbc, temp_pagep, i, page, NULL, 0)) != 0)
break;
if (tmp_buf == NULL)
__os_free(dbp->env, temp_pagep);
return (ret);
}
int
__ham_del_pair(dbc, flags, ppg)
DBC *dbc;
int flags;
PAGE *ppg;
{
DB *dbp;
DBT data_dbt, key_dbt;
DB_LSN new_lsn, *n_lsn, tmp_lsn;
DB_MPOOLFILE *mpf;
HASH_CURSOR *hcp;
PAGE *n_pagep, *nn_pagep, *p, *p_pagep;
db_ham_mode op;
db_indx_t ndx;
db_pgno_t chg_pgno, pgno, tmp_pgno;
u_int32_t data_type, key_type, order;
int ret, t_ret;
u_int8_t *hk;
dbp = dbc->dbp;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
n_pagep = p_pagep = nn_pagep = NULL;
ndx = hcp->indx;
if (hcp->page == NULL &&
(ret = __memp_fget(mpf, &hcp->pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &hcp->page)) != 0)
return (ret);
p = hcp->page;
if (!LF_ISSET(HAM_DEL_IGNORE_OFFPAGE) &&
HPAGE_PTYPE(H_PAIRKEY(dbp, p, ndx)) == H_OFFPAGE) {
memcpy(&pgno, HOFFPAGE_PGNO(P_ENTRY(dbp, p, H_KEYINDEX(ndx))),
sizeof(db_pgno_t));
ret = __db_doff(dbc, pgno);
} else
ret = 0;
if (!LF_ISSET(HAM_DEL_IGNORE_OFFPAGE) && ret == 0)
switch (HPAGE_PTYPE(H_PAIRDATA(dbp, p, ndx))) {
case H_OFFPAGE:
memcpy(&pgno,
HOFFPAGE_PGNO(P_ENTRY(dbp, p, H_DATAINDEX(ndx))),
sizeof(db_pgno_t));
ret = __db_doff(dbc, pgno);
break;
case H_OFFDUP:
case H_DUPLICATE:
F_CLR(hcp, H_ISDUP);
break;
default:
break;
}
if (ret)
return (ret);
if (DBC_LOGGING(dbc)) {
hk = H_PAIRKEY(dbp, hcp->page, ndx);
if ((key_type = HPAGE_PTYPE(hk)) == H_OFFPAGE) {
key_dbt.data = hk;
key_dbt.size = HOFFPAGE_SIZE;
} else {
key_dbt.data = HKEYDATA_DATA(hk);
key_dbt.size =
LEN_HKEY(dbp, hcp->page, dbp->pgsize, ndx);
}
hk = H_PAIRDATA(dbp, hcp->page, ndx);
if ((data_type = HPAGE_PTYPE(hk)) == H_OFFPAGE) {
data_dbt.data = hk;
data_dbt.size = HOFFPAGE_SIZE;
} else if (data_type == H_OFFDUP) {
data_dbt.data = hk;
data_dbt.size = HOFFDUP_SIZE;
} else {
data_dbt.data = HKEYDATA_DATA(hk);
data_dbt.size =
LEN_HDATA(dbp, hcp->page, dbp->pgsize, ndx);
}
if ((ret = __ham_insdel_log(dbp, dbc->txn, &new_lsn, 0,
DELPAIR, PGNO(p), (u_int32_t)ndx, &LSN(p),
OP_SET(key_type, p), &key_dbt,
OP_SET(data_type, p), &data_dbt)) != 0)
return (ret);
} else
LSN_NOT_LOGGED(new_lsn);
LSN(p) = new_lsn;
__ham_dpair(dbp, p, ndx);
F_SET(hcp, H_DELETED);
F_CLR(hcp, H_OK);
hcp->stream_start_pgno = PGNO_INVALID;
if (!STD_LOCKING(dbc)) {
if ((ret = __ham_dirty_meta(dbc, 0)) != 0)
return (ret);
--hcp->hdr->nelem;
}
if (LF_ISSET(HAM_DEL_NO_CURSOR))
return (0);
if ((ret = __hamc_update(dbc, 0, DB_HAM_CURADJ_DEL, 0)) != 0)
return (ret);
if (LF_ISSET(HAM_DEL_NO_RECLAIM) ||
NUM_ENT(p) != 0 ||
(PREV_PGNO(p) == PGNO_INVALID && NEXT_PGNO(p) == PGNO_INVALID)) {
if (NUM_ENT(p) == 0)
F_SET(hcp, H_CONTRACT);
return (0);
}
if (PREV_PGNO(p) == PGNO_INVALID) {
if ((ret = __memp_fget(mpf,
&NEXT_PGNO(p), dbc->thread_info, dbc->txn,
DB_MPOOL_DIRTY, &n_pagep)) != 0)
return (ret);
if (NEXT_PGNO(n_pagep) != PGNO_INVALID &&
(ret = __memp_fget(mpf, &NEXT_PGNO(n_pagep),
dbc->thread_info, dbc->txn,
DB_MPOOL_DIRTY, &nn_pagep)) != 0)
goto err;
if (DBC_LOGGING(dbc)) {
key_dbt.data = n_pagep;
key_dbt.size = dbp->pgsize;
if ((ret = __ham_copypage_log(dbp,
dbc->txn, &new_lsn, 0, PGNO(p),
&LSN(p), PGNO(n_pagep), &LSN(n_pagep),
NEXT_PGNO(n_pagep),
nn_pagep == NULL ? NULL : &LSN(nn_pagep),
&key_dbt)) != 0)
goto err;
} else
LSN_NOT_LOGGED(new_lsn);
LSN(p) = new_lsn;
LSN(n_pagep) = new_lsn;
if (NEXT_PGNO(n_pagep) != PGNO_INVALID)
LSN(nn_pagep) = new_lsn;
if (nn_pagep != NULL) {
PREV_PGNO(nn_pagep) = PGNO(p);
ret = __memp_fput(mpf,
dbc->thread_info, nn_pagep, dbc->priority);
nn_pagep = NULL;
if (ret != 0)
goto err;
}
tmp_pgno = PGNO(p);
tmp_lsn = LSN(p);
memcpy(p, n_pagep, dbp->pgsize);
PGNO(p) = tmp_pgno;
LSN(p) = tmp_lsn;
PREV_PGNO(p) = PGNO_INVALID;
if ((ret = __hamc_delpg(dbc, PGNO(n_pagep),
PGNO(p), 0, DB_HAM_DELFIRSTPG, &order)) != 0)
goto err;
hcp->indx = 0;
hcp->pgno = PGNO(p);
hcp->order += order;
if ((ret = __db_free(dbc, n_pagep, 0)) != 0) {
n_pagep = NULL;
goto err;
}
} else {
if ((p_pagep = ppg) == NULL && (ret = __memp_fget(mpf,
&PREV_PGNO(p), dbc->thread_info, dbc->txn,
DB_MPOOL_DIRTY, &p_pagep)) != 0)
goto err;
if (NEXT_PGNO(p) != PGNO_INVALID) {
if ((ret = __memp_fget(mpf, &NEXT_PGNO(p),
dbc->thread_info, dbc->txn,
DB_MPOOL_DIRTY, &n_pagep)) != 0)
goto err;
n_lsn = &LSN(n_pagep);
} else {
n_pagep = NULL;
n_lsn = NULL;
}
if (DBC_LOGGING(dbc)) {
if ((ret = __ham_newpage_log(dbp, dbc->txn,
&new_lsn, 0, DELOVFL, PREV_PGNO(p), &LSN(p_pagep),
PGNO(p), &LSN(p), NEXT_PGNO(p), n_lsn)) != 0)
goto err;
} else
LSN_NOT_LOGGED(new_lsn);
LSN(p_pagep) = new_lsn;
if (n_pagep)
LSN(n_pagep) = new_lsn;
LSN(p) = new_lsn;
NEXT_PGNO(p_pagep) = NEXT_PGNO(p);
if (n_pagep != NULL)
PREV_PGNO(n_pagep) = PGNO(p_pagep);
if (NEXT_PGNO(p) == PGNO_INVALID) {
hcp->pgno = PGNO(p_pagep);
hcp->indx = NUM_ENT(p_pagep);
op = DB_HAM_DELLASTPG;
} else {
hcp->pgno = NEXT_PGNO(p);
hcp->indx = 0;
op = DB_HAM_DELMIDPG;
}
hcp->page = NULL;
chg_pgno = PGNO(p);
ret = __db_free(dbc, p, 0);
if (ppg == NULL && (t_ret = __memp_fput(mpf, dbc->thread_info,
p_pagep, dbc->priority)) != 0 && ret == 0)
ret = t_ret;
if (n_pagep != NULL && (t_ret = __memp_fput(mpf,
dbc->thread_info, n_pagep, dbc->priority)) != 0 && ret == 0)
ret = t_ret;
if (ret != 0)
return (ret);
if ((ret = __hamc_delpg(dbc,
chg_pgno, hcp->pgno, hcp->indx, op, &order)) != 0)
return (ret);
hcp->order += order;
}
return (ret);
err:
if (n_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, n_pagep, dbc->priority);
if (nn_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, nn_pagep, dbc->priority);
if (ppg == NULL && p_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, p_pagep, dbc->priority);
return (ret);
}
int
__ham_replpair(dbc, dbt, newtype)
DBC *dbc;
DBT *dbt;
u_int32_t newtype;
{
DB *dbp;
DBC **carray, *dbc_n;
DBT old_dbt, tdata, tmp, *new_dbt;
DB_LSN new_lsn;
ENV *env;
HASH_CURSOR *hcp, *cp;
db_indx_t orig_indx;
db_pgno_t off_pgno, orig_pgno;
u_int32_t change;
u_int32_t dup_flag, len, memsize, newlen, oldtype, type;
char tmp_ch;
int beyond_eor, is_big, is_plus, ret, i, found, t_ret;
u_int8_t *beg, *dest, *end, *hk, *src;
void *memp;
dbp = dbc->dbp;
env = dbp->env;
hcp = (HASH_CURSOR *)dbc->internal;
carray = NULL;
dbc_n = memp = NULL;
found = 0;
new_dbt = NULL;
off_pgno = PGNO_INVALID;
type = 0;
if (dbt->size > dbt->dlen) {
change = dbt->size - dbt->dlen;
is_plus = 1;
} else {
change = dbt->dlen - dbt->size;
is_plus = 0;
}
hk = H_PAIRDATA(dbp, hcp->page, hcp->indx);
oldtype = HPAGE_PTYPE(hk);
is_big = oldtype == H_OFFPAGE;
if (is_big) {
memcpy(&len, HOFFPAGE_TLEN(hk), sizeof(u_int32_t));
memcpy(&off_pgno, HOFFPAGE_PGNO(hk), sizeof(db_pgno_t));
} else
len = LEN_HKEYDATA(dbp, hcp->page,
dbp->pgsize, H_DATAINDEX(hcp->indx));
beyond_eor = dbt->doff + dbt->dlen > len;
if (beyond_eor) {
if (is_plus)
change += dbt->doff + dbt->dlen - len;
else if (dbt->doff + dbt->dlen - len > change) {
is_plus = 1;
change = (dbt->doff + dbt->dlen - len) - change;
} else
change -= (dbt->doff + dbt->dlen - len);
}
newlen = (is_plus ? len + change : len - change);
if (is_big || beyond_eor || ISBIG(hcp, newlen) ||
(is_plus && change > P_FREESPACE(dbp, hcp->page))) {
if (is_plus && dbc->txn == NULL &&
dbp->mpf->mfp->maxpgno != 0 &&
dbp->mpf->mfp->maxpgno == dbp->mpf->mfp->last_pgno)
return (__db_space_err(dbp));
memset(&tmp, 0, sizeof(tmp));
if ((ret = __db_ret(dbc, hcp->page, H_KEYINDEX(hcp->indx),
&tmp, &dbc->my_rkey.data, &dbc->my_rkey.ulen)) != 0)
return (ret);
dup_flag = F_ISSET(hcp, H_ISDUP);
if (is_big && !dup_flag && !DB_IS_PRIMARY(dbp) &&
F_ISSET(dbt, DB_DBT_PARTIAL) && dbt->doff == len) {
if (hcp->stream_start_pgno != off_pgno ||
hcp->stream_off > dbt->doff || dbt->doff >
hcp->stream_off + P_MAXSPACE(dbp, dbp->pgsize)) {
memset(&tdata, 0, sizeof(DBT));
tdata.doff = dbt->doff - 1;
tdata.dlen = tdata.ulen = 1;
tdata.data = &tmp_ch;
tdata.flags = DB_DBT_PARTIAL | DB_DBT_USERMEM;
if ((ret = __db_goff(dbc, &tdata, len,
off_pgno, NULL, NULL)) != 0)
return (ret);
}
tdata = *dbt;
tdata.dlen = dbt->size;
tdata.size = newlen;
new_dbt = &tdata;
F_SET(new_dbt, DB_DBT_STREAMING);
type = H_KEYDATA;
}
orig_pgno = PGNO(hcp->page);
orig_indx = hcp->indx;
if ((ret = __ham_get_clist(dbp,
orig_pgno, orig_indx, &carray)) != 0)
goto err;
if (dbt->doff == 0 && dbt->dlen == len) {
type = (dup_flag ? H_DUPLICATE : H_KEYDATA);
new_dbt = dbt;
} else if (!F_ISSET(dbt, DB_DBT_STREAMING)) {
type = HPAGE_PTYPE(hk) != H_OFFPAGE ?
HPAGE_PTYPE(hk) : H_KEYDATA;
memset(&tdata, 0, sizeof(tdata));
memsize = 0;
if ((ret = __db_ret(dbc, hcp->page,
H_DATAINDEX(hcp->indx), &tdata,
&memp, &memsize)) != 0)
goto err;
if (is_plus) {
if ((ret = __os_realloc(env,
tdata.size + change, &tdata.data)) != 0)
return (ret);
memp = tdata.data;
memsize = tdata.size + change;
memset((u_int8_t *)tdata.data + tdata.size,
0, change);
}
end = (u_int8_t *)tdata.data + tdata.size;
src = (u_int8_t *)tdata.data + dbt->doff + dbt->dlen;
if (src < end && tdata.size > dbt->doff + dbt->dlen) {
len = tdata.size - (dbt->doff + dbt->dlen);
if (is_plus)
dest = src + change;
else
dest = src - change;
memmove(dest, src, len);
}
memcpy((u_int8_t *)tdata.data + dbt->doff,
dbt->data, dbt->size);
if (is_plus)
tdata.size += change;
else
tdata.size -= change;
new_dbt = &tdata;
}
if ((ret = __ham_del_pair(dbc, HAM_DEL_NO_CURSOR |
(F_ISSET(dbt, DB_DBT_STREAMING) ? HAM_DEL_IGNORE_OFFPAGE :
0), NULL)) != 0)
goto err;
if ((ret = __dbc_dup(dbc, &dbc_n, DB_POSITION)) != 0)
goto err;
if ((ret = __ham_add_el(dbc, &tmp, new_dbt, type)) != 0)
goto err;
F_SET(hcp, dup_flag);
if (((HASH_CURSOR*)dbc_n->internal)->pgno != hcp->pgno ||
((HASH_CURSOR*)dbc_n->internal)->indx != hcp->indx) {
if (carray != NULL) {
for (i = 0; carray[i] != NULL; i++) {
cp = (HASH_CURSOR*)carray[i]->internal;
cp->pgno = hcp->pgno;
cp->indx = hcp->indx;
F_CLR(cp, H_DELETED);
found = 1;
}
if (found && DBC_LOGGING(dbc) &&
IS_SUBTRANSACTION(dbc->txn)) {
if ((ret =
__ham_chgpg_log(dbp,
dbc->txn, &new_lsn, 0,
DB_HAM_CHGPG, orig_pgno, hcp->pgno,
orig_indx, hcp->indx)) != 0)
goto err;
}
}
ret = __hamc_update(dbc_n, 0, DB_HAM_CURADJ_DEL, 0);
}
err: if (dbc_n != NULL && (t_ret = __dbc_close(dbc_n)) != 0 &&
ret == 0)
ret = t_ret;
if (carray != NULL)
__os_free(env, carray);
if (memp != NULL)
__os_free(env, memp);
return (ret);
}
beg = HKEYDATA_DATA(H_PAIRDATA(dbp, hcp->page, hcp->indx));
beg += dbt->doff;
if (DBC_LOGGING(dbc)) {
old_dbt.data = beg;
old_dbt.size = dbt->dlen;
if ((ret = __ham_replace_log(dbp, dbc->txn, &new_lsn,
0, PGNO(hcp->page),
(u_int32_t)H_DATAINDEX(hcp->indx), &LSN(hcp->page),
(int32_t)dbt->doff, OP_SET(oldtype, hcp->page),
&old_dbt, OP_SET(newtype, hcp->page), dbt)) != 0)
return (ret);
} else
LSN_NOT_LOGGED(new_lsn);
LSN(hcp->page) = new_lsn;
__ham_onpage_replace(dbp, hcp->page, (u_int32_t)H_DATAINDEX(hcp->indx),
(int32_t)dbt->doff, change, is_plus, dbt);
return (0);
}
void
__ham_onpage_replace(dbp, pagep, ndx, off, change, is_plus, dbt)
DB *dbp;
PAGE *pagep;
u_int32_t ndx;
int32_t off;
u_int32_t change;
int is_plus;
DBT *dbt;
{
db_indx_t i, *inp;
int32_t len;
size_t pgsize;
u_int8_t *src, *dest;
int zero_me;
pgsize = dbp->pgsize;
inp = P_INP(dbp, pagep);
if (change != 0) {
zero_me = 0;
src = (u_int8_t *)(pagep) + HOFFSET(pagep);
if (off < 0)
len = inp[ndx] - HOFFSET(pagep);
else if ((u_int32_t)off >=
LEN_HKEYDATA(dbp, pagep, pgsize, ndx)) {
len = (int32_t)(HKEYDATA_DATA(P_ENTRY(dbp, pagep, ndx))
+ LEN_HKEYDATA(dbp, pagep, pgsize, ndx) - src);
zero_me = 1;
} else
len = (int32_t)(
(HKEYDATA_DATA(P_ENTRY(dbp, pagep, ndx)) + off) -
src);
if (is_plus)
dest = src - change;
else
dest = src + change;
memmove(dest, src, (size_t)len);
if (zero_me)
memset(dest + len, 0, change);
for (i = ndx; i < NUM_ENT(pagep); i++) {
if (is_plus)
inp[i] -= change;
else
inp[i] += change;
}
if (is_plus)
HOFFSET(pagep) -= change;
else
HOFFSET(pagep) += change;
}
if (off >= 0)
memcpy(HKEYDATA_DATA(P_ENTRY(dbp, pagep, ndx)) + off,
dbt->data, dbt->size);
else
memcpy(P_ENTRY(dbp, pagep, ndx), dbt->data, dbt->size);
}
int
__ham_merge_pages(dbc, tobucket, frombucket, c_data)
DBC *dbc;
u_int32_t tobucket, frombucket;
DB_COMPACT *c_data;
{
DB *dbp;
DBC **carray;
DB_LOCK tlock, firstlock;
DB_LSN from_lsn;
DB_MPOOLFILE *mpf;
ENV *env;
HASH_CURSOR *hcp, *cp;
PAGE *to_pagep, *first_pagep,
*from_pagep, *last_pagep, *next_pagep, *prev_pagep;
db_pgno_t to_pgno, first_pgno, from_pgno;
u_int32_t len;
db_indx_t dest_indx, n, num_ent;
int check_trunc, found, i, ret;
dbp = dbc->dbp;
carray = NULL;
env = dbp->env;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
hcp->pgno = PGNO_INVALID;
to_pagep = first_pagep = NULL;
from_pagep = last_pagep = next_pagep = prev_pagep = NULL;
from_pgno = PGNO_INVALID;
LOCK_INIT(tlock);
LOCK_INIT(firstlock);
check_trunc =
c_data == NULL ? 0 : c_data->compact_truncate != PGNO_INVALID;
to_pgno = BUCKET_TO_PAGE(hcp, tobucket);
if ((ret = __db_lget(dbc,
0, to_pgno, DB_LOCK_WRITE, 0, &tlock)) != 0)
goto err;
if ((ret = __memp_fget(mpf, &to_pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &to_pagep)) != 0)
goto err;
if (to_pagep->type == P_HASH_UNSORTED)
if ((ret = __ham_sort_page_cursor(dbc, to_pagep)) != 0)
return (ret);
from_pgno = BUCKET_TO_PAGE(hcp, frombucket);
if ((ret = __db_lget(dbc,
0, from_pgno, DB_LOCK_WRITE, 0, &firstlock)) != 0)
goto err;
next_page:
if (from_pagep == NULL &&
(ret = __memp_fget(mpf, &from_pgno, dbc->thread_info,
dbc->txn, DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &from_pagep)) != 0)
goto err;
if ((ret = __ham_get_clist(dbp, from_pgno, NDX_INVALID, &carray)) != 0)
goto err;
hcp->indx = 0;
hcp->pgno = from_pgno;
hcp->page = from_pagep;
num_ent = NUM_ENT(from_pagep);
for (n = 0; n < num_ent; n += 2) {
len = LEN_HITEM(dbp, from_pagep,
dbp->pgsize, H_DATAINDEX(hcp->indx)) +
LEN_HITEM(dbp, from_pagep,
dbp->pgsize, H_KEYINDEX(hcp->indx)) +
2 * sizeof(db_indx_t);
while (P_FREESPACE(dbp, to_pagep) < len) {
to_pgno = NEXT_PGNO(to_pagep);
if (to_pgno == PGNO_INVALID) {
next_pagep = to_pagep;
if ((ret =
__ham_add_ovflpage(dbc, &next_pagep)) != 0)
goto err;
if ((ret = __memp_fput(mpf, dbc->thread_info,
to_pagep, dbc->priority)) != 0)
goto err;
to_pagep = next_pagep;
next_pagep = NULL;
if (c_data != NULL &&
c_data->compact_pages_free > 0)
c_data->compact_pages_free--;
to_pgno = PGNO(to_pagep);
} else {
if ((ret = __memp_fput(mpf, dbc->thread_info,
to_pagep, dbc->priority)) != 0)
goto err;
to_pagep = NULL;
if ((ret = __memp_fget(mpf,
&to_pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY,
&to_pagep)) != 0)
goto err;
if (to_pagep->type == P_HASH_UNSORTED)
if ((ret = __ham_sort_page_cursor(dbc,
to_pagep)) != 0)
goto err;
}
}
dest_indx = NDX_INVALID;
if ((ret = __ham_copypair(dbc,
from_pagep, hcp->indx, to_pagep, &dest_indx, 1)) != 0)
goto err;
if (carray != NULL) {
found = 0;
for (i = 0; carray[i] != NULL; i++) {
cp =
(HASH_CURSOR *)carray[i]->internal;
if (cp->pgno == from_pgno &&
cp->indx == n) {
cp->pgno = PGNO(to_pagep);
cp->indx = dest_indx;
cp->bucket = tobucket;
found = 1;
}
}
if (found && DBC_LOGGING(dbc) &&
IS_SUBTRANSACTION(dbc->txn)) {
if ((ret =
__ham_chgpg_log(dbp,
dbc->txn, &from_lsn, 0,
DB_HAM_SPLIT, from_pgno,
PGNO(to_pagep), n, dest_indx)) != 0)
goto err;
}
}
if (PREV_PGNO(from_pagep) == PGNO_INVALID) {
if ((ret = __ham_del_pair(dbc,
HAM_DEL_IGNORE_OFFPAGE | HAM_DEL_NO_CURSOR,
from_pagep)) != 0)
goto err;
if (!STD_LOCKING(dbc)) {
if ((ret = __ham_dirty_meta(dbc, 0)) != 0)
return (ret);
++hcp->hdr->nelem;
}
} else
hcp->indx += 2;
}
from_pgno = NEXT_PGNO(from_pagep);
if (PREV_PGNO(from_pagep) != PGNO_INVALID) {
if (DBC_LOGGING(dbc)) {
if ((ret = __db_relink_log(dbp, dbc->txn,
&LSN(prev_pagep), 0, PGNO(from_pagep),
PGNO_INVALID, PGNO(prev_pagep),
&LSN(prev_pagep), PGNO_INVALID, NULL)) != 0)
goto err;
} else
LSN_NOT_LOGGED(LSN(prev_pagep));
NEXT_PGNO(prev_pagep) = PGNO_INVALID;
if ((ret = __db_free(dbc, from_pagep, 0)) != 0) {
from_pagep = NULL;
goto err;
}
if (c_data != NULL)
c_data->compact_pages_free++;
if ((ret = __memp_fput(mpf,
dbc->thread_info, prev_pagep, dbc->priority)) != 0)
goto err;
prev_pagep = NULL;
} else if (from_pgno != PGNO_INVALID)
prev_pagep = from_pagep;
else if ((ret = __memp_fput(mpf,
dbc->thread_info, from_pagep, dbc->priority)) != 0)
goto err;
from_pagep = NULL;
hcp->page = NULL;
if (carray != NULL)
__os_free(env, carray);
carray = NULL;
if (check_trunc && from_pgno > c_data->compact_truncate)
goto next_page;
first_pgno = from_pgno;
last_pagep = NULL;
while (from_pgno != PGNO_INVALID) {
if ((ret = __memp_fget(mpf,
&from_pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &from_pagep)) != 0)
goto err;
if (P_FREESPACE(dbp, to_pagep) >
(dbp->pgsize - HOFFSET(from_pagep)) +
(NUM_ENT(from_pagep) * sizeof(db_indx_t)))
break;
if (check_trunc && from_pgno > c_data->compact_truncate)
break;
from_pgno = NEXT_PGNO(from_pagep);
if (last_pagep != NULL && last_pagep != first_pagep &&
(ret = __memp_fput(mpf,
dbc->thread_info, last_pagep, dbc->priority)) != 0)
goto err;
last_pagep = from_pagep;
if (first_pagep == NULL)
first_pagep = from_pagep;
from_pagep = NULL;
}
if (first_pgno != PGNO_INVALID && first_pgno != from_pgno) {
DB_ASSERT(dbp->env, first_pagep != NULL);
next_pagep = NULL;
if (NEXT_PGNO(to_pagep) != PGNO_INVALID && (ret =
__memp_fget(mpf, &NEXT_PGNO(to_pagep), dbc->thread_info,
dbc->txn, DB_MPOOL_DIRTY, &next_pagep)) != 0)
goto err;
if (last_pagep == NULL)
last_pagep = first_pagep;
DB_ASSERT(dbp->env, last_pagep != NULL);
if (DBC_LOGGING(dbc)) {
if ((ret = __db_relink_log(dbp, dbc->txn,
&LSN(to_pagep), 0, NEXT_PGNO(to_pagep),
first_pgno, to_pgno, &LSN(to_pagep),
PGNO_INVALID, NULL)) != 0)
goto err;
if ((ret = __db_relink_log(dbp, dbc->txn,
&LSN(first_pagep), 0, PREV_PGNO(first_pagep),
to_pgno, PGNO_INVALID, NULL, first_pgno,
&LSN(first_pagep))) != 0)
goto err;
if (next_pagep != NULL) {
if ((ret = __db_relink_log(dbp, dbc->txn,
&LSN(next_pagep), 0, PREV_PGNO(next_pagep),
PGNO(last_pagep), PGNO_INVALID, NULL,
PGNO(next_pagep), &LSN(next_pagep))) != 0)
goto err;
if ((ret = __db_relink_log(dbp, dbc->txn,
&LSN(last_pagep), 0, NEXT_PGNO(last_pagep),
PGNO(next_pagep), PGNO(last_pagep),
&LSN(last_pagep), PGNO_INVALID, NULL)) != 0)
goto err;
} else if (NEXT_PGNO(last_pagep) != PGNO_INVALID &&
(ret = __db_relink_log(dbp, dbc->txn,
&LSN(last_pagep), 0, NEXT_PGNO(last_pagep),
PGNO_INVALID, PGNO(last_pagep),
&LSN(last_pagep), PGNO_INVALID, NULL)) != 0)
goto err;
if (prev_pagep != NULL &&
(ret = __db_relink_log(dbp, dbc->txn,
&LSN(prev_pagep), 0, NEXT_PGNO(prev_pagep),
NEXT_PGNO(last_pagep), PGNO(prev_pagep),
&LSN(prev_pagep), PGNO_INVALID, NULL)) != 0)
goto err;
} else {
LSN_NOT_LOGGED(LSN(to_pagep));
LSN_NOT_LOGGED(LSN(first_pagep));
LSN_NOT_LOGGED(LSN(last_pagep));
if (next_pagep != NULL)
LSN_NOT_LOGGED(LSN(to_pagep));
}
if (prev_pagep != NULL)
NEXT_PGNO(prev_pagep) = NEXT_PGNO(last_pagep);
NEXT_PGNO(last_pagep) = NEXT_PGNO(to_pagep);
NEXT_PGNO(to_pagep) = first_pgno;
PREV_PGNO(first_pagep) = to_pgno;
if (next_pagep != NULL) {
PREV_PGNO(next_pagep) = PGNO(last_pagep);
if ((ret = __memp_fput(mpf,
dbc->thread_info, next_pagep, dbc->priority)) != 0)
goto err;
next_pagep = NULL;
}
if (last_pagep != first_pagep && (ret = __memp_fput(mpf,
dbc->thread_info, last_pagep, dbc->priority)) != 0)
goto err;
last_pagep = NULL;
if ((ret = __memp_fput(mpf,
dbc->thread_info, first_pagep, dbc->priority)) != 0)
goto err;
first_pagep = NULL;
} else if (last_pagep != NULL && (ret = __memp_fput(mpf,
dbc->thread_info, last_pagep, dbc->priority)) != 0)
goto err;
if (from_pagep == NULL) {
from_pagep = first_pagep;
first_pagep = NULL;
}
if (from_pgno != PGNO_INVALID)
goto next_page;
if (prev_pagep != NULL && (ret = __memp_fput(mpf,
dbc->thread_info, prev_pagep, dbc->priority)) != 0)
goto err;
ret = __memp_fput(mpf, dbc->thread_info, to_pagep, dbc->priority);
return (ret);
err: if (last_pagep != NULL && last_pagep != first_pagep)
(void)__memp_fput(mpf,
dbc->thread_info, last_pagep, dbc->priority);
if (first_pagep != NULL && first_pagep != from_pagep)
(void)__memp_fput(mpf,
dbc->thread_info, first_pagep, dbc->priority);
if (next_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, next_pagep, dbc->priority);
if (from_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, from_pagep, dbc->priority);
if (to_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, to_pagep, dbc->priority);
if (prev_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, prev_pagep, dbc->priority);
hcp->page = NULL;
(void)__TLPUT(dbc, tlock);
(void)__TLPUT(dbc, firstlock);
if (carray != NULL)
__os_free(env, carray);
return (ret);
}
int
__ham_split_page(dbc, obucket, nbucket)
DBC *dbc;
u_int32_t obucket, nbucket;
{
DB *dbp;
DBC **carray, *tmp_dbc;
DBT key, page_dbt;
DB_LOCK block;
DB_LSN new_lsn;
DB_MPOOLFILE *mpf;
ENV *env;
HASH_CURSOR *hcp, *cp;
PAGE **pp, *old_pagep, *temp_pagep, *new_pagep, *next_pagep;
db_indx_t n, dest_indx;
db_pgno_t bucket_pgno, npgno, next_pgno;
u_int32_t big_len, len;
int found, i, ret, t_ret;
void *big_buf;
dbp = dbc->dbp;
carray = NULL;
env = dbp->env;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
temp_pagep = old_pagep = new_pagep = NULL;
npgno = PGNO_INVALID;
LOCK_INIT(block);
bucket_pgno = BUCKET_TO_PAGE(hcp, obucket);
if ((ret = __db_lget(dbc,
0, bucket_pgno, DB_LOCK_WRITE, 0, &block)) != 0)
goto err;
if ((ret = __memp_fget(mpf, &bucket_pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &old_pagep)) != 0)
goto err;
if (old_pagep->type == P_HASH_UNSORTED)
if ((ret = __ham_sort_page_cursor(dbc, old_pagep)) != 0)
return (ret);
npgno = BUCKET_TO_PAGE(hcp, nbucket);
if ((ret = __memp_fget(mpf, &npgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &new_pagep)) != 0)
goto err;
P_INIT(new_pagep,
dbp->pgsize, npgno, PGNO_INVALID, PGNO_INVALID, 0, P_HASH);
temp_pagep = hcp->split_buf;
memcpy(temp_pagep, old_pagep, dbp->pgsize);
if (DBC_LOGGING(dbc)) {
page_dbt.size = dbp->pgsize;
page_dbt.data = old_pagep;
if ((ret = __ham_splitdata_log(dbp,
dbc->txn, &new_lsn, 0, SPLITOLD,
PGNO(old_pagep), &page_dbt, &LSN(old_pagep))) != 0)
goto err;
} else
LSN_NOT_LOGGED(new_lsn);
LSN(old_pagep) = new_lsn;
P_INIT(old_pagep, dbp->pgsize, PGNO(old_pagep), PGNO_INVALID,
PGNO_INVALID, 0, P_HASH);
big_len = 0;
big_buf = NULL;
memset(&key, 0, sizeof(key));
while (temp_pagep != NULL) {
if ((ret = __ham_get_clist(dbp,
PGNO(temp_pagep), NDX_INVALID, &carray)) != 0)
goto err;
for (n = 0; n < (db_indx_t)NUM_ENT(temp_pagep); n += 2) {
if ((ret = __db_ret(dbc, temp_pagep, H_KEYINDEX(n),
&key, &big_buf, &big_len)) != 0)
goto err;
if (__ham_call_hash(dbc, key.data, key.size) == obucket)
pp = &old_pagep;
else
pp = &new_pagep;
len = LEN_HITEM(dbp, temp_pagep, dbp->pgsize,
H_DATAINDEX(n)) +
LEN_HITEM(dbp, temp_pagep, dbp->pgsize,
H_KEYINDEX(n)) +
2 * sizeof(db_indx_t);
if (P_FREESPACE(dbp, *pp) < len) {
if (DBC_LOGGING(dbc)) {
page_dbt.size = dbp->pgsize;
page_dbt.data = *pp;
if ((ret = __ham_splitdata_log(dbp,
dbc->txn, &new_lsn, 0,
SPLITNEW, PGNO(*pp), &page_dbt,
&LSN(*pp))) != 0)
goto err;
} else
LSN_NOT_LOGGED(new_lsn);
LSN(*pp) = new_lsn;
next_pagep = *pp;
if ((ret =
__ham_add_ovflpage(dbc, &next_pagep)) != 0)
goto err;
if ((ret = __memp_fput(mpf,
dbc->thread_info, *pp, dbc->priority)) != 0)
goto err;
*pp = next_pagep;
}
dest_indx = NDX_INVALID;
if ((ret = __ham_copypair(dbc, temp_pagep,
H_KEYINDEX(n), *pp, &dest_indx, 0)) != 0)
goto err;
if (PGNO(temp_pagep) != bucket_pgno) {
if ((ret = __db_cursor_int(dbp,
dbc->thread_info, dbc->txn, dbp->type,
PGNO_INVALID, 0, DB_LOCK_INVALIDID,
&tmp_dbc)) != 0)
goto err;
hcp = (HASH_CURSOR*)tmp_dbc->internal;
hcp->pgno = PGNO(*pp);
hcp->indx = dest_indx;
hcp->dup_off = 0;
hcp->order = 0;
if ((ret = __hamc_update(
tmp_dbc, len, DB_HAM_CURADJ_ADD, 0)) != 0)
goto err;
if ((ret = __dbc_close(tmp_dbc)) != 0)
goto err;
}
if (carray != NULL) {
found = 0;
for (i = 0; carray[i] != NULL; i++) {
cp =
(HASH_CURSOR *)carray[i]->internal;
if (cp->pgno == PGNO(temp_pagep) &&
cp->indx == n) {
cp->pgno = PGNO(*pp);
cp->indx = dest_indx;
if (cp->pgno == PGNO(old_pagep))
cp->bucket = obucket;
else
cp->bucket = nbucket;
found = 1;
}
}
if (found && DBC_LOGGING(dbc) &&
IS_SUBTRANSACTION(dbc->txn)) {
if ((ret =
__ham_chgpg_log(dbp,
dbc->txn, &new_lsn, 0,
DB_HAM_SPLIT, PGNO(temp_pagep),
PGNO(*pp), n, dest_indx)) != 0)
goto err;
}
}
}
next_pgno = NEXT_PGNO(temp_pagep);
if (PGNO(temp_pagep) != bucket_pgno && (ret =
__db_free(dbc, temp_pagep, 0)) != 0) {
temp_pagep = NULL;
goto err;
}
if (next_pgno == PGNO_INVALID)
temp_pagep = NULL;
else if ((ret = __memp_fget(mpf,
&next_pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE | DB_MPOOL_DIRTY, &temp_pagep)) != 0)
goto err;
if (temp_pagep != NULL) {
if (DBC_LOGGING(dbc)) {
page_dbt.size = dbp->pgsize;
page_dbt.data = temp_pagep;
if ((ret = __ham_splitdata_log(dbp,
dbc->txn, &new_lsn, 0,
SPLITOLD, PGNO(temp_pagep),
&page_dbt, &LSN(temp_pagep))) != 0)
goto err;
} else
LSN_NOT_LOGGED(new_lsn);
LSN(temp_pagep) = new_lsn;
}
if (carray != NULL)
__os_free(env, carray);
carray = NULL;
}
if (big_buf != NULL)
__os_free(env, big_buf);
if (temp_pagep != NULL && PGNO(temp_pagep) != bucket_pgno &&
(ret = __db_free(dbc, temp_pagep, 0)) != 0) {
temp_pagep = NULL;
goto err;
}
if (DBC_LOGGING(dbc)) {
page_dbt.size = dbp->pgsize;
page_dbt.data = old_pagep;
if ((ret = __ham_splitdata_log(dbp, dbc->txn,
&new_lsn, 0, SPLITNEW, PGNO(old_pagep), &page_dbt,
&LSN(old_pagep))) != 0)
goto err;
LSN(old_pagep) = new_lsn;
page_dbt.data = new_pagep;
if ((ret = __ham_splitdata_log(dbp, dbc->txn, &new_lsn, 0,
SPLITNEW, PGNO(new_pagep), &page_dbt,
&LSN(new_pagep))) != 0)
goto err;
LSN(new_pagep) = new_lsn;
} else {
LSN_NOT_LOGGED(LSN(old_pagep));
LSN_NOT_LOGGED(LSN(new_pagep));
}
ret = __memp_fput(mpf, dbc->thread_info, old_pagep, dbc->priority);
if ((t_ret = __memp_fput(mpf,
dbc->thread_info, new_pagep, dbc->priority)) != 0 && ret == 0)
ret = t_ret;
if (0) {
err: if (old_pagep != NULL)
(void)__memp_fput(mpf,
dbc->thread_info, old_pagep, dbc->priority);
if (new_pagep != NULL) {
P_INIT(new_pagep, dbp->pgsize,
npgno, PGNO_INVALID, PGNO_INVALID, 0, P_HASH);
(void)__memp_fput(mpf,
dbc->thread_info, new_pagep, dbc->priority);
}
if (temp_pagep != NULL && PGNO(temp_pagep) != bucket_pgno)
(void)__memp_fput(mpf,
dbc->thread_info, temp_pagep, dbc->priority);
}
if ((t_ret = __TLPUT(dbc, block)) != 0 && ret == 0)
ret = t_ret;
if (carray != NULL)
__os_free(env, carray);
return (ret);
}
int
__ham_add_el(dbc, key, val, type)
DBC *dbc;
const DBT *key, *val;
u_int32_t type;
{
const DBT *pkey, *pdata;
DB *dbp;
DBT key_dbt, data_dbt;
DB_LSN new_lsn;
DB_MPOOLFILE *mpf;
HASH_CURSOR *hcp;
HOFFPAGE doff, koff;
PAGE *new_pagep;
db_pgno_t next_pgno, pgno;
u_int32_t data_size, data_type, key_size, key_type;
u_int32_t pages, pagespace, pairsize;
int do_expand, is_keybig, is_databig, match, ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
do_expand = 0;
pgno = hcp->seek_found_page != PGNO_INVALID ?
hcp->seek_found_page : hcp->pgno;
if (hcp->page == NULL && (ret = __memp_fget(mpf, &pgno,
dbc->thread_info, dbc->txn, DB_MPOOL_CREATE, &hcp->page)) != 0)
return (ret);
key_size = HKEYDATA_PSIZE(key->size);
data_size = HKEYDATA_PSIZE(val->size);
is_keybig = ISBIG(hcp, key->size);
is_databig = ISBIG(hcp, val->size);
if (is_keybig)
key_size = HOFFPAGE_PSIZE;
if (is_databig)
data_size = HOFFPAGE_PSIZE;
pairsize = key_size + data_size;
while (H_NUMPAIRS(hcp->page) && NEXT_PGNO(hcp->page) != PGNO_INVALID) {
if (P_FREESPACE(dbp, hcp->page) >= pairsize)
break;
next_pgno = NEXT_PGNO(hcp->page);
if ((ret = __ham_next_cpage(dbc, next_pgno)) != 0)
return (ret);
}
if (P_FREESPACE(dbp, hcp->page) < pairsize) {
do_expand = 1;
if ((ret = __memp_dirty(mpf, &hcp->page,
dbc->thread_info, dbc->txn, dbc->priority, 0)) != 0)
return (ret);
new_pagep = hcp->page;
if ((ret = __ham_add_ovflpage(dbc, &new_pagep)) != 0)
return (ret);
if ((ret = __memp_fput(mpf,
dbc->thread_info, hcp->page, dbc->priority)) != 0) {
(void)__memp_fput(mpf,
dbc->thread_info, new_pagep, dbc->priority);
return (ret);
}
hcp->page = new_pagep;
hcp->pgno = PGNO(hcp->page);
}
if (dbc->txn == NULL &&
dbp->mpf->mfp->maxpgno != 0 && (is_keybig || is_databig)) {
pagespace = P_MAXSPACE(dbp, dbp->pgsize);
pages = 0;
if (is_databig)
pages = ((data_size - 1) / pagespace) + 1;
if (is_keybig) {
pages += ((key->size - 1) / pagespace) + 1;
if (pages >
(dbp->mpf->mfp->maxpgno - dbp->mpf->mfp->last_pgno))
return (__db_space_err(dbp));
}
}
if ((ret = __memp_dirty(mpf,
&hcp->page, dbc->thread_info, dbc->txn, dbc->priority, 0)) != 0)
return (ret);
hcp->indx = hcp->seek_found_indx;
F_CLR(hcp, H_DELETED);
if (is_keybig) {
koff.type = H_OFFPAGE;
UMRW_SET(koff.unused[0]);
UMRW_SET(koff.unused[1]);
UMRW_SET(koff.unused[2]);
if ((ret = __db_poff(dbc, key, &koff.pgno)) != 0)
return (ret);
koff.tlen = key->size;
key_dbt.data = &koff;
key_dbt.size = sizeof(koff);
pkey = &key_dbt;
key_type = H_OFFPAGE;
} else {
pkey = key;
key_type = H_KEYDATA;
}
if (is_databig) {
doff.type = H_OFFPAGE;
UMRW_SET(doff.unused[0]);
UMRW_SET(doff.unused[1]);
UMRW_SET(doff.unused[2]);
if ((ret = __db_poff(dbc, val, &doff.pgno)) != 0)
return (ret);
doff.tlen = val->size;
data_dbt.data = &doff;
data_dbt.size = sizeof(doff);
pdata = &data_dbt;
data_type = H_OFFPAGE;
} else {
pdata = val;
data_type = type;
}
if (((PAGE *)hcp->page)->type == P_HASH_UNSORTED)
if ((ret = __ham_sort_page_cursor(dbc, hcp->page)) != 0)
return (ret);
if (PGNO(hcp->page) != hcp->seek_found_page) {
if ((ret = __ham_getindex(dbc, hcp->page, pkey,
key_type, &match, &hcp->seek_found_indx)) != 0)
return (ret);
hcp->seek_found_page = PGNO(hcp->page);
DB_ASSERT(dbp->env, hcp->seek_found_indx <= NUM_ENT(hcp->page));
}
if (DBC_LOGGING(dbc)) {
if ((ret = __ham_insdel_log(dbp, dbc->txn, &new_lsn, 0,
PUTPAIR, PGNO(hcp->page), (u_int32_t)hcp->seek_found_indx,
&LSN(hcp->page), OP_SET(key_type, hcp->page), pkey,
OP_SET(data_type, hcp->page), pdata)) != 0)
return (ret);
} else
LSN_NOT_LOGGED(new_lsn);
LSN(hcp->page) = new_lsn;
if ((ret = __ham_insertpair(dbc, hcp->page,
&hcp->seek_found_indx, pkey, pdata, key_type, data_type)) != 0)
return (ret);
if ((ret = __hamc_update(dbc, pairsize, DB_HAM_CURADJ_ADD, 0)) != 0)
return (ret);
hcp->pgno = PGNO(hcp->page);
hcp->indx = hcp->seek_found_indx;
if (!STD_LOCKING(dbc)) {
if ((ret = __ham_dirty_meta(dbc, 0)) != 0)
return (ret);
hcp->hdr->nelem++;
}
if (do_expand || (hcp->hdr->ffactor != 0 &&
(u_int32_t)H_NUMPAIRS(hcp->page) > hcp->hdr->ffactor))
F_SET(hcp, H_EXPAND);
return (0);
}
int
__ham_copypair(dbc, src_page, src_ndx, dest_page, dest_indx, log)
DBC *dbc;
PAGE *src_page;
u_int32_t src_ndx;
PAGE *dest_page;
db_indx_t *dest_indx;
int log;
{
DB *dbp;
DBT tkey, tdata;
db_indx_t kindx, dindx, dest;
u_int32_t ktype, dtype;
int match, ret;
dbp = dbc->dbp;
ret = 0;
memset(&tkey, 0, sizeof(tkey));
memset(&tdata, 0, sizeof(tdata));
ktype = HPAGE_TYPE(dbp, src_page, H_KEYINDEX(src_ndx));
dtype = HPAGE_TYPE(dbp, src_page, H_DATAINDEX(src_ndx));
kindx = H_KEYINDEX(src_ndx);
dindx = H_DATAINDEX(src_ndx);
if (ktype == H_OFFPAGE) {
tkey.data = P_ENTRY(dbp, src_page, kindx);
tkey.size = LEN_HITEM(dbp, src_page, dbp->pgsize, kindx);
} else {
tkey.data = HKEYDATA_DATA(P_ENTRY(dbp, src_page, kindx));
tkey.size = LEN_HKEYDATA(dbp, src_page, dbp->pgsize, kindx);
}
if (dtype == H_OFFPAGE || dtype == H_OFFDUP) {
tdata.data = P_ENTRY(dbp, src_page, dindx);
tdata.size = LEN_HITEM(dbp, src_page, dbp->pgsize, dindx);
} else {
tdata.data = HKEYDATA_DATA(P_ENTRY(dbp, src_page, dindx));
tdata.size = LEN_HKEYDATA(dbp, src_page, dbp->pgsize, dindx);
}
if (dest_indx != NULL)
dest = *dest_indx;
else
dest = NDX_INVALID;
if (dest == NDX_INVALID) {
if ((ret = __ham_getindex(dbc,
dest_page, &tkey, ktype, &match, &dest)) != 0)
return (ret);
DB_ASSERT(dbp->env, match != 0);
}
if (log == 1) {
if (DBC_LOGGING(dbc)) {
if ((ret = __ham_insdel_log(dbp, dbc->txn,
&LSN(dest_page), 0, PUTPAIR,
PGNO(dest_page), (u_int32_t)dest, &LSN(dest_page),
OP_SET(ktype, dest_page), &tkey,
OP_SET(dtype, dest_page), &tdata)) != 0)
return (ret);
} else
LSN_NOT_LOGGED(LSN(dest_page));
}
if ((ret = __ham_insertpair(dbc, dest_page, &dest,
&tkey, &tdata, ktype, dtype)) != 0)
return (ret);
DB_ASSERT(dbp->env, dtype != H_DUPLICATE ||
HPAGE_TYPE(dbp, dest_page, H_DATAINDEX(dest)) == dtype);
if (dest_indx != NULL)
*dest_indx = dest;
return (ret);
}
int
__ham_add_ovflpage(dbc, pp)
DBC *dbc;
PAGE **pp;
{
DB *dbp;
DB_LSN new_lsn;
DB_MPOOLFILE *mpf;
PAGE *new_pagep, *pagep;
int ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
pagep = *pp;
*pp = NULL;
DB_ASSERT(dbp->env, IS_DIRTY(pagep));
if ((ret = __db_new(dbc, P_HASH, NULL, &new_pagep)) != 0)
return (ret);
if (DBC_LOGGING(dbc)) {
if ((ret = __ham_newpage_log(dbp, dbc->txn, &new_lsn, 0,
PUTOVFL, PGNO(pagep), &LSN(pagep), PGNO(new_pagep),
&LSN(new_pagep), PGNO_INVALID, NULL)) != 0) {
(void)__memp_fput(mpf,
dbc->thread_info, new_pagep, dbc->priority);
return (ret);
}
} else
LSN_NOT_LOGGED(new_lsn);
LSN(pagep) = LSN(new_pagep) = new_lsn;
NEXT_PGNO(pagep) = PGNO(new_pagep);
PREV_PGNO(new_pagep) = PGNO(pagep);
*pp = new_pagep;
return (0);
}
int
__ham_get_cpage(dbc, mode)
DBC *dbc;
db_lockmode_t mode;
{
DB *dbp;
DB_LOCK tmp_lock;
DB_MPOOLFILE *mpf;
HASH_CURSOR *hcp;
int ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
ret = 0;
LOCK_INIT(tmp_lock);
if (STD_LOCKING(dbc)) {
if (hcp->lbucket != hcp->bucket) {
if ((ret = __TLPUT(dbc, hcp->lock)) != 0)
return (ret);
LOCK_INIT(hcp->lock);
hcp->stream_start_pgno = PGNO_INVALID;
}
if ((LOCK_ISSET(hcp->lock) &&
((hcp->lock_mode == DB_LOCK_READ ||
F_ISSET(dbp, DB_AM_READ_UNCOMMITTED)) &&
mode == DB_LOCK_WRITE))) {
tmp_lock = hcp->lock;
LOCK_INIT(hcp->lock);
}
if (!LOCK_ISSET(hcp->lock))
if ((ret = __ham_lock_bucket(dbc, mode)) != 0)
return (ret);
if (ret == 0) {
hcp->lock_mode = mode;
hcp->lbucket = hcp->bucket;
if ((ret = __ENV_LPUT(dbp->env, tmp_lock)) != 0)
return (ret);
} else if (LOCK_ISSET(tmp_lock))
hcp->lock = tmp_lock;
}
if (ret == 0 && hcp->page == NULL) {
if (hcp->pgno == PGNO_INVALID)
hcp->pgno = BUCKET_TO_PAGE(hcp, hcp->bucket);
if ((ret = __memp_fget(mpf,
&hcp->pgno, dbc->thread_info, dbc->txn,
(mode == DB_LOCK_WRITE ? DB_MPOOL_DIRTY : 0) |
DB_MPOOL_CREATE, &hcp->page)) != 0)
return (ret);
}
return (0);
}
int
__ham_next_cpage(dbc, pgno)
DBC *dbc;
db_pgno_t pgno;
{
DB *dbp;
DB_MPOOLFILE *mpf;
HASH_CURSOR *hcp;
PAGE *p;
int ret;
dbp = dbc->dbp;
mpf = dbp->mpf;
hcp = (HASH_CURSOR *)dbc->internal;
if (hcp->page != NULL && (ret = __memp_fput(mpf,
dbc->thread_info, hcp->page, dbc->priority)) != 0)
return (ret);
hcp->stream_start_pgno = PGNO_INVALID;
hcp->page = NULL;
if ((ret = __memp_fget(mpf, &pgno, dbc->thread_info, dbc->txn,
DB_MPOOL_CREATE, &p)) != 0)
return (ret);
hcp->page = p;
hcp->pgno = pgno;
hcp->indx = 0;
return (0);
}
int
__ham_lock_bucket(dbc, mode)
DBC *dbc;
db_lockmode_t mode;
{
HASH_CURSOR *hcp;
db_pgno_t pgno;
int gotmeta, ret;
hcp = (HASH_CURSOR *)dbc->internal;
gotmeta = hcp->hdr == NULL ? 1 : 0;
if (gotmeta)
if ((ret = __ham_get_meta(dbc)) != 0)
return (ret);
pgno = BUCKET_TO_PAGE(hcp, hcp->bucket);
if (gotmeta)
if ((ret = __ham_release_meta(dbc)) != 0)
return (ret);
ret = __db_lget(dbc, 0, pgno, mode, 0, &hcp->lock);
hcp->lock_mode = mode;
return (ret);
}
void
__ham_dpair(dbp, p, indx)
DB *dbp;
PAGE *p;
u_int32_t indx;
{
db_indx_t delta, n, *inp;
u_int8_t *dest, *src;
inp = P_INP(dbp, p);
delta = H_PAIRSIZE(dbp, p, dbp->pgsize, indx);
if ((db_indx_t)indx != NUM_ENT(p) - 2) {
src = (u_int8_t *)p + HOFFSET(p);
dest = src + delta;
memmove(dest, src, inp[H_DATAINDEX(indx)] - HOFFSET(p));
}
HOFFSET(p) = HOFFSET(p) + delta;
NUM_ENT(p) = NUM_ENT(p) - 2;
for (n = (db_indx_t)indx; n < (db_indx_t)(NUM_ENT(p)); n++)
inp[n] = inp[n + 2] + delta;
}
static int
__hamc_delpg_getorder(cp, my_dbc, orderp, new_pgno, indx, args)
DBC *cp, *my_dbc;
u_int32_t *orderp;
db_pgno_t new_pgno;
u_int32_t indx;
void *args;
{
HASH_CURSOR *hcp;
COMPQUIET(args, NULL);
if (cp == my_dbc || cp->dbtype != DB_HASH)
return (0);
hcp = (HASH_CURSOR *)cp->internal;
if (hcp->pgno == new_pgno &&
!MVCC_SKIP_CURADJ(cp, new_pgno)) {
if (hcp->indx == indx &&
F_ISSET(hcp, H_DELETED) &&
hcp->order > *orderp)
*orderp = hcp->order;
}
return (0);
}
struct __hamc_delpg_setorder_args {
db_pgno_t new_pgno;
u_int32_t order;
db_ham_mode op;
DB_TXN *my_txn;
};
static int
__hamc_delpg_setorder(cp, my_dbc, foundp, old_pgno, indx, vargs)
DBC *cp, *my_dbc;
u_int32_t *foundp;
db_pgno_t old_pgno;
u_int32_t indx;
void *vargs;
{
HASH_CURSOR *hcp;
struct __hamc_delpg_setorder_args *args;
if (cp == my_dbc || cp->dbtype != DB_HASH)
return (0);
hcp = (HASH_CURSOR *)cp->internal;
args = vargs;
if (hcp->pgno == old_pgno &&
!MVCC_SKIP_CURADJ(cp, old_pgno)) {
switch (args->op) {
case DB_HAM_DELFIRSTPG:
hcp->pgno = args->new_pgno;
if (hcp->indx == indx)
hcp->order += args->order;
break;
case DB_HAM_DELMIDPG:
hcp->pgno = args->new_pgno;
DB_ASSERT(cp->dbp->env, hcp->indx == 0 &&
F_ISSET(hcp, H_DELETED));
hcp->order += args->order;
break;
case DB_HAM_DELLASTPG:
hcp->pgno = args->new_pgno;
DB_ASSERT(cp->dbp->env, hcp->indx == 0 &&
F_ISSET(hcp, H_DELETED));
hcp->indx = indx;
hcp->order += args->order;
break;
default:
return (__db_unknown_path(
cp->dbp->env, "__hamc_delpg"));
}
if (args->my_txn != NULL && cp->txn != args->my_txn)
*foundp = 1;
}
return (0);
}
static int
__hamc_delpg(dbc, old_pgno, new_pgno, num_ent, op, orderp)
DBC *dbc;
db_pgno_t old_pgno, new_pgno;
u_int32_t num_ent;
db_ham_mode op;
u_int32_t *orderp;
{
DB *dbp;
DB_LSN lsn;
db_indx_t indx;
int ret;
u_int32_t found;
struct __hamc_delpg_setorder_args args;
indx = (op == DB_HAM_DELLASTPG) ? num_ent : 0;
dbp = dbc->dbp;
if ((ret = __db_walk_cursors(dbp, dbc,
__hamc_delpg_getorder, &args.order, new_pgno, indx, NULL)) != 0)
return (ret);
args.order++;
args.my_txn = IS_SUBTRANSACTION(dbc->txn) ? dbc->txn : NULL;
args.op = op;
args.new_pgno = new_pgno;
if ((ret = __db_walk_cursors(dbp, dbc,
__hamc_delpg_setorder, &found, old_pgno, indx, &args)) != 0)
return (ret);
if (found != 0 && DBC_LOGGING(dbc)) {
if ((ret = __ham_chgpg_log(dbp, args.my_txn, &lsn, 0, op,
old_pgno, new_pgno, indx, args.order)) != 0)
return (ret);
}
*orderp = args.order;
return (0);
}