mirror of
https://github.com/berkeleydb/libdb.git
synced 2024-11-17 01:26:25 +00:00
5659 lines
144 KiB
Text
5659 lines
144 KiB
Text
|
/*
|
||
|
|
||
|
BerkeleyDB.xs -- Perl 5 interface to Berkeley DB version 2, 3 & 4
|
||
|
|
||
|
written by Paul Marquess <pmqs@cpan.org>
|
||
|
|
||
|
All comments/suggestions/problems are welcome
|
||
|
|
||
|
Copyright (c) 1997-2011 Paul Marquess. All rights reserved.
|
||
|
This program is free software; you can redistribute it and/or
|
||
|
modify it under the same terms as Perl itself.
|
||
|
|
||
|
Please refer to the COPYRIGHT section in
|
||
|
|
||
|
Changes:
|
||
|
0.01 - First Alpha Release
|
||
|
0.02 -
|
||
|
|
||
|
*/
|
||
|
|
||
|
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
#define PERL_POLLUTE
|
||
|
#include "EXTERN.h"
|
||
|
#include "perl.h"
|
||
|
#include "XSUB.h"
|
||
|
#include "ppport.h"
|
||
|
|
||
|
|
||
|
/* XSUB.h defines a macro called abort */
|
||
|
/* This clashes with the txn abort method in Berkeley DB 4.x */
|
||
|
/* This is a problem with ActivePerl (at least) */
|
||
|
|
||
|
#ifdef _WIN32
|
||
|
# ifdef abort
|
||
|
# undef abort
|
||
|
# endif
|
||
|
# ifdef fopen
|
||
|
# undef fopen
|
||
|
# endif
|
||
|
# ifdef fclose
|
||
|
# undef fclose
|
||
|
# endif
|
||
|
# ifdef rename
|
||
|
# undef rename
|
||
|
# endif
|
||
|
# ifdef open
|
||
|
# undef open
|
||
|
# endif
|
||
|
#endif
|
||
|
|
||
|
#ifndef SvUTF8_off
|
||
|
# define SvUTF8_off(x)
|
||
|
#endif
|
||
|
|
||
|
/* Being the Berkeley DB we prefer the <sys/cdefs.h> (which will be
|
||
|
* shortly #included by the <db.h>) __attribute__ to the possibly
|
||
|
* already defined __attribute__, for example by GNUC or by Perl. */
|
||
|
|
||
|
#undef __attribute__
|
||
|
|
||
|
#ifdef USE_PERLIO
|
||
|
# define GetFILEptr(sv) PerlIO_findFILE(IoIFP(sv_2io(sv)))
|
||
|
#else
|
||
|
# define GetFILEptr(sv) IoIFP(sv_2io(sv))
|
||
|
#endif
|
||
|
|
||
|
#include <db.h>
|
||
|
|
||
|
/* Check the version of Berkeley DB */
|
||
|
|
||
|
#ifndef DB_VERSION_MAJOR
|
||
|
#ifdef HASHMAGIC
|
||
|
#error db.h is from Berkeley DB 1.x - need at least Berkeley DB 2.6.4
|
||
|
#else
|
||
|
#error db.h is not for Berkeley DB at all.
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
#if (DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 6) ||\
|
||
|
(DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR == 6 && DB_VERSION_PATCH < 4)
|
||
|
# error db.h is from Berkeley DB 2.0-2.5 - need at least Berkeley DB 2.6.4
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#if (DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR == 0)
|
||
|
# define IS_DB_3_0_x
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR >= 3
|
||
|
# define AT_LEAST_DB_3
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 3 || (DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR >= 1)
|
||
|
# define AT_LEAST_DB_3_1
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 3 || (DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR >= 2)
|
||
|
# define AT_LEAST_DB_3_2
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 3 || \
|
||
|
(DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR > 2) ||\
|
||
|
(DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR == 2 && DB_VERSION_PATCH >= 6)
|
||
|
# define AT_LEAST_DB_3_2_6
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 3 || (DB_VERSION_MAJOR == 3 && DB_VERSION_MINOR >= 3)
|
||
|
# define AT_LEAST_DB_3_3
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR >= 4
|
||
|
# define AT_LEAST_DB_4
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 1)
|
||
|
# define AT_LEAST_DB_4_1
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 2)
|
||
|
# define AT_LEAST_DB_4_2
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 3)
|
||
|
# define AT_LEAST_DB_4_3
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 4)
|
||
|
# define AT_LEAST_DB_4_4
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 5)
|
||
|
# define AT_LEAST_DB_4_5
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 6)
|
||
|
# define AT_LEAST_DB_4_6
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 7)
|
||
|
# define AT_LEAST_DB_4_7
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 4 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR >= 8)
|
||
|
# define AT_LEAST_DB_4_8
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 5 || (DB_VERSION_MAJOR == 5 && DB_VERSION_MINOR >= 1)
|
||
|
# define AT_LEAST_DB_5_1
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 5 || (DB_VERSION_MAJOR == 5 && DB_VERSION_MINOR >= 2)
|
||
|
# define AT_LEAST_DB_5_2
|
||
|
#endif
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#define DBM_FILTERING
|
||
|
#define STRICT_CLOSE
|
||
|
/* #define ALLOW_RECNO_OFFSET */
|
||
|
/* #define TRACE */
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2 && ! defined(DB_LOCK_DEADLOCK)
|
||
|
# define DB_LOCK_DEADLOCK EAGAIN
|
||
|
#endif /* DB_VERSION_MAJOR == 2 */
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
# define DB_QUEUE 4
|
||
|
#endif /* DB_VERSION_MAJOR == 2 */
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
# define BackRef internal
|
||
|
#else
|
||
|
# if DB_VERSION_MAJOR == 3 || (DB_VERSION_MAJOR == 4 && DB_VERSION_MINOR == 0)
|
||
|
# define BackRef cj_internal
|
||
|
# else
|
||
|
# define BackRef api_internal
|
||
|
# endif
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
# define DB_callback DB * db,
|
||
|
#else
|
||
|
# define DB_callback
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
typedef struct {
|
||
|
int db_lorder;
|
||
|
size_t db_cachesize;
|
||
|
size_t db_pagesize;
|
||
|
|
||
|
|
||
|
void *(*db_malloc) __P((size_t));
|
||
|
int (*dup_compare)
|
||
|
__P((DB_callback const DBT *, const DBT *));
|
||
|
|
||
|
u_int32_t bt_maxkey;
|
||
|
u_int32_t bt_minkey;
|
||
|
int (*bt_compare)
|
||
|
__P((DB_callback const DBT *, const DBT *));
|
||
|
size_t (*bt_prefix)
|
||
|
__P((DB_callback const DBT *, const DBT *));
|
||
|
|
||
|
u_int32_t h_ffactor;
|
||
|
u_int32_t h_nelem;
|
||
|
u_int32_t (*h_hash)
|
||
|
__P((DB_callback const void *, u_int32_t));
|
||
|
|
||
|
int re_pad;
|
||
|
int re_delim;
|
||
|
u_int32_t re_len;
|
||
|
char *re_source;
|
||
|
|
||
|
#define DB_DELIMITER 0x0001
|
||
|
#define DB_FIXEDLEN 0x0008
|
||
|
#define DB_PAD 0x0010
|
||
|
u_int32_t flags;
|
||
|
u_int32_t q_extentsize;
|
||
|
|
||
|
u_int32_t heapsize_gbytes;
|
||
|
u_int32_t heapsize_bytes;
|
||
|
} DB_INFO ;
|
||
|
|
||
|
#endif /* DB_VERSION_MAJOR > 2 */
|
||
|
|
||
|
typedef struct {
|
||
|
int Status ;
|
||
|
/* char ErrBuff[1000] ; */
|
||
|
SV * ErrPrefix ;
|
||
|
SV * ErrHandle ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
SV * MsgHandle ;
|
||
|
#endif
|
||
|
DB_ENV * Env ;
|
||
|
int open_dbs ;
|
||
|
int TxnMgrStatus ;
|
||
|
int active ;
|
||
|
bool txn_enabled ;
|
||
|
bool opened ;
|
||
|
bool cds_enabled;
|
||
|
} BerkeleyDB_ENV_type ;
|
||
|
|
||
|
|
||
|
typedef struct {
|
||
|
DBTYPE type ;
|
||
|
bool recno_or_queue ;
|
||
|
char * filename ;
|
||
|
BerkeleyDB_ENV_type * parent_env ;
|
||
|
DB * dbp ;
|
||
|
SV * compare ;
|
||
|
bool in_compare ;
|
||
|
SV * dup_compare ;
|
||
|
bool in_dup_compare ;
|
||
|
SV * prefix ;
|
||
|
bool in_prefix ;
|
||
|
SV * hash ;
|
||
|
bool in_hash ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
SV * associated ;
|
||
|
bool secondary_db ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
SV * associated_foreign ;
|
||
|
SV * bt_compress ;
|
||
|
SV * bt_uncompress ;
|
||
|
#endif
|
||
|
bool primary_recno_or_queue ;
|
||
|
int Status ;
|
||
|
DB_INFO * info ;
|
||
|
DBC * cursor ;
|
||
|
DB_TXN * txn ;
|
||
|
int open_cursors ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
int open_sequences ;
|
||
|
#endif
|
||
|
u_int32_t partial ;
|
||
|
u_int32_t dlen ;
|
||
|
u_int32_t doff ;
|
||
|
int active ;
|
||
|
bool cds_enabled;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
int array_base ;
|
||
|
#endif
|
||
|
#ifdef DBM_FILTERING
|
||
|
SV * filter_fetch_key ;
|
||
|
SV * filter_store_key ;
|
||
|
SV * filter_fetch_value ;
|
||
|
SV * filter_store_value ;
|
||
|
int filtering ;
|
||
|
#endif
|
||
|
} BerkeleyDB_type;
|
||
|
|
||
|
|
||
|
typedef struct {
|
||
|
DBTYPE type ;
|
||
|
bool recno_or_queue ;
|
||
|
char * filename ;
|
||
|
DB * dbp ;
|
||
|
SV * compare ;
|
||
|
SV * dup_compare ;
|
||
|
SV * prefix ;
|
||
|
SV * hash ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
SV * associated ;
|
||
|
bool secondary_db ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
SV * associated_foreign ;
|
||
|
#endif
|
||
|
bool primary_recno_or_queue ;
|
||
|
int Status ;
|
||
|
DB_INFO * info ;
|
||
|
DBC * cursor ;
|
||
|
DB_TXN * txn ;
|
||
|
BerkeleyDB_type * parent_db ;
|
||
|
u_int32_t partial ;
|
||
|
u_int32_t dlen ;
|
||
|
u_int32_t doff ;
|
||
|
int active ;
|
||
|
bool cds_enabled;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
int array_base ;
|
||
|
#endif
|
||
|
#ifdef DBM_FILTERING
|
||
|
SV * filter_fetch_key ;
|
||
|
SV * filter_store_key ;
|
||
|
SV * filter_fetch_value ;
|
||
|
SV * filter_store_value ;
|
||
|
int filtering ;
|
||
|
#endif
|
||
|
} BerkeleyDB_Cursor_type;
|
||
|
|
||
|
typedef struct {
|
||
|
BerkeleyDB_ENV_type * env ;
|
||
|
} BerkeleyDB_TxnMgr_type ;
|
||
|
|
||
|
#if 1
|
||
|
typedef struct {
|
||
|
int Status ;
|
||
|
DB_TXN * txn ;
|
||
|
int active ;
|
||
|
} BerkeleyDB_Txn_type ;
|
||
|
#else
|
||
|
typedef DB_TXN BerkeleyDB_Txn_type ;
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
typedef struct {
|
||
|
int active;
|
||
|
BerkeleyDB_type *db;
|
||
|
DB_SEQUENCE *seq;
|
||
|
} BerkeleyDB_Sequence_type;
|
||
|
#else
|
||
|
typedef int BerkeleyDB_Sequence_type;
|
||
|
typedef SV* db_seq_t;
|
||
|
#endif
|
||
|
|
||
|
|
||
|
typedef BerkeleyDB_ENV_type * BerkeleyDB__Env ;
|
||
|
typedef BerkeleyDB_ENV_type * BerkeleyDB__Env__Raw ;
|
||
|
typedef BerkeleyDB_ENV_type * BerkeleyDB__Env__Inner ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB ;
|
||
|
typedef void * BerkeleyDB__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Common ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Common__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Common__Inner ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Hash ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Hash__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Btree ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Btree__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Recno ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Recno__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Queue ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Queue__Raw ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Heap ;
|
||
|
typedef BerkeleyDB_type * BerkeleyDB__Heap__Raw ;
|
||
|
typedef BerkeleyDB_Cursor_type BerkeleyDB__Cursor_type ;
|
||
|
typedef BerkeleyDB_Cursor_type * BerkeleyDB__Cursor ;
|
||
|
typedef BerkeleyDB_Cursor_type * BerkeleyDB__Cursor__Raw ;
|
||
|
typedef BerkeleyDB_TxnMgr_type * BerkeleyDB__TxnMgr ;
|
||
|
typedef BerkeleyDB_TxnMgr_type * BerkeleyDB__TxnMgr__Raw ;
|
||
|
typedef BerkeleyDB_TxnMgr_type * BerkeleyDB__TxnMgr__Inner ;
|
||
|
typedef BerkeleyDB_Txn_type * BerkeleyDB__Txn ;
|
||
|
typedef BerkeleyDB_Txn_type * BerkeleyDB__Txn__Raw ;
|
||
|
typedef BerkeleyDB_Txn_type * BerkeleyDB__Txn__Inner ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
typedef BerkeleyDB_Sequence_type * BerkeleyDB__Sequence ;
|
||
|
#else
|
||
|
typedef int * BerkeleyDB__Sequence ;
|
||
|
#endif
|
||
|
#if 0
|
||
|
typedef DB_LOG * BerkeleyDB__Log ;
|
||
|
typedef DB_LOCKTAB * BerkeleyDB__Lock ;
|
||
|
#endif
|
||
|
typedef DBT DBTKEY ;
|
||
|
typedef DBT DBT_OPT ;
|
||
|
typedef DBT DBT_B ;
|
||
|
typedef DBT DBTKEY_B ;
|
||
|
typedef DBT DBTKEY_Br ;
|
||
|
typedef DBT DBTKEY_Bpr ;
|
||
|
typedef DBT DBTKEY_seq ;
|
||
|
typedef DBT DBTVALUE ;
|
||
|
typedef void * PV_or_NULL ;
|
||
|
typedef PerlIO * IO_or_NULL ;
|
||
|
typedef int DualType ;
|
||
|
typedef SV SVnull;
|
||
|
|
||
|
static void
|
||
|
hash_delete(char * hash, char * key);
|
||
|
|
||
|
#ifdef TRACE
|
||
|
# define Trace(x) (printf("# "), printf x)
|
||
|
#else
|
||
|
# define Trace(x)
|
||
|
#endif
|
||
|
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
# define RECNO_BASE db->array_base
|
||
|
#else
|
||
|
# define RECNO_BASE 1
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
# define flagSet_DB2(i, f) i |= f
|
||
|
#else
|
||
|
# define flagSet_DB2(i, f)
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 5
|
||
|
# define flagSet(bitmask) (flags & (bitmask))
|
||
|
#else
|
||
|
# define flagSet(bitmask) ((flags & DB_OPFLAGS_MASK) == (bitmask))
|
||
|
#endif
|
||
|
|
||
|
#ifdef DB_GET_BOTH_RANGE
|
||
|
# define flagSetBoth() (flagSet(DB_GET_BOTH) || flagSet(DB_GET_BOTH_RANGE))
|
||
|
#else
|
||
|
# define flagSetBoth() (flagSet(DB_GET_BOTH))
|
||
|
#endif
|
||
|
|
||
|
#ifndef AT_LEAST_DB_4
|
||
|
typedef int db_timeout_t ;
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_5_2
|
||
|
|
||
|
# define isHeapDb(db) ((db)->type == DB_HEAP)
|
||
|
#else
|
||
|
# define isHeapDb(db) (0)
|
||
|
|
||
|
int __heap_exist __P((void));
|
||
|
# define DB_HEAP_RID_SZ 1
|
||
|
|
||
|
|
||
|
#endif
|
||
|
|
||
|
#define ERR_BUFF "BerkeleyDB::Error"
|
||
|
|
||
|
#define ZMALLOC(to, typ) ((to = (typ *)safemalloc(sizeof(typ))), \
|
||
|
Zero(to,1,typ))
|
||
|
|
||
|
#define DBT_clear(x) Zero(&x, 1, DBT) ;
|
||
|
|
||
|
#if 1
|
||
|
#define getInnerObject(x) (*av_fetch((AV*)SvRV(x), 0, FALSE))
|
||
|
#else
|
||
|
#define getInnerObject(x) ((SV*)SvRV(sv))
|
||
|
#endif
|
||
|
|
||
|
#define my_sv_setpvn(sv, d, s) do { \
|
||
|
s ? sv_setpvn(sv, d, s) : sv_setpv(sv, ""); \
|
||
|
SvUTF8_off(sv); \
|
||
|
} while(0)
|
||
|
|
||
|
#define GetValue_iv(h,k) (((sv = readHash(h, k)) && sv != &PL_sv_undef) \
|
||
|
? SvIV(sv) : 0)
|
||
|
#define SetValue_iv(i, k) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) \
|
||
|
i = SvIV(sv)
|
||
|
#define SetValue_io(i, k) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) \
|
||
|
i = GetFILEptr(sv)
|
||
|
#define SetValue_sv(i, k) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) \
|
||
|
i = sv
|
||
|
#define SetValue_pv(i, k,t) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) \
|
||
|
i = (t)SvPV(sv,PL_na)
|
||
|
#define SetValue_pvx(i, k, t) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) \
|
||
|
i = (t)SvPVX(sv)
|
||
|
#define SetValue_ov(i,k,t) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) {\
|
||
|
IV tmp = SvIV(getInnerObject(sv)) ; \
|
||
|
i = INT2PTR(t, tmp) ; \
|
||
|
}
|
||
|
|
||
|
#define SetValue_ovx(i,k,t) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) {\
|
||
|
HV * hv = (HV *)GetInternalObject(sv); \
|
||
|
SV ** svp = hv_fetch(hv, "db", 2, FALSE);\
|
||
|
IV tmp = SvIV(*svp); \
|
||
|
i = INT2PTR(t, tmp) ; \
|
||
|
}
|
||
|
|
||
|
#define SetValue_ovX(i,k,t) if ((sv = readHash(hash, k)) && sv != &PL_sv_undef) {\
|
||
|
IV tmp = SvIV(GetInternalObject(sv));\
|
||
|
i = INT2PTR(t, tmp) ; \
|
||
|
}
|
||
|
|
||
|
#define LastDBerror DB_RUNRECOVERY
|
||
|
|
||
|
#define setDUALerrno(var, err) \
|
||
|
sv_setnv(var, (double)err) ; \
|
||
|
sv_setpv(var, ((err) ? db_strerror(err) : "")) ;\
|
||
|
SvNOK_on(var);
|
||
|
|
||
|
#define OutputValue(arg, name) \
|
||
|
{ if (RETVAL == 0) { \
|
||
|
my_sv_setpvn(arg, name.data, name.size) ; \
|
||
|
DBM_ckFilter(arg, filter_fetch_value,"filter_fetch_value") ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define OutputValue_B(arg, name) \
|
||
|
{ if (RETVAL == 0) { \
|
||
|
if (db->type == DB_BTREE && \
|
||
|
flagSet(DB_GET_RECNO)){ \
|
||
|
sv_setiv(arg, (I32)(*(I32*)name.data) - RECNO_BASE); \
|
||
|
} \
|
||
|
else { \
|
||
|
my_sv_setpvn(arg, name.data, name.size) ; \
|
||
|
} \
|
||
|
DBM_ckFilter(arg, filter_fetch_value, "filter_fetch_value"); \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define OutputKey(arg, name) \
|
||
|
{ if (RETVAL == 0) \
|
||
|
{ \
|
||
|
if (!db->recno_or_queue) { \
|
||
|
my_sv_setpvn(arg, name.data, name.size); \
|
||
|
} \
|
||
|
else \
|
||
|
sv_setiv(arg, (I32)*(I32*)name.data - RECNO_BASE); \
|
||
|
if (! isHeapDb(db)) \
|
||
|
DBM_ckFilter(arg, filter_fetch_key, "filter_fetch_key") ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
|
||
|
#define InputKey_seq(arg, var) \
|
||
|
{ \
|
||
|
SV* my_sv = arg ; \
|
||
|
/* DBM_ckFilter(my_sv, filter_store_key, "filter_store_key"); */ \
|
||
|
DBT_clear(var) ; \
|
||
|
SvGETMAGIC(arg) ; \
|
||
|
if (seq->db->recno_or_queue) { \
|
||
|
Value = GetRecnoKey(seq->db, SvIV(my_sv)) ; \
|
||
|
var.data = & Value; \
|
||
|
var.size = (int)sizeof(db_recno_t); \
|
||
|
} \
|
||
|
else { \
|
||
|
STRLEN len; \
|
||
|
var.data = SvPV(my_sv, len); \
|
||
|
var.size = (int)len; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define OutputKey_seq(arg, name) \
|
||
|
{ if (RETVAL == 0) \
|
||
|
{ \
|
||
|
if (!seq->db->recno_or_queue) { \
|
||
|
my_sv_setpvn(arg, name.data, name.size); \
|
||
|
} \
|
||
|
else \
|
||
|
sv_setiv(arg, (I32)*(I32*)name.data - RECNO_BASE); \
|
||
|
} \
|
||
|
}
|
||
|
#else
|
||
|
#define InputKey_seq(arg, var)
|
||
|
#define OutputKey_seq(arg, name)
|
||
|
#endif
|
||
|
|
||
|
#define OutputKey_B(arg, name) \
|
||
|
{ if (RETVAL == 0) \
|
||
|
{ \
|
||
|
if (db->recno_or_queue \
|
||
|
|| (db->type == DB_BTREE && \
|
||
|
flagSet(DB_GET_RECNO))){ \
|
||
|
sv_setiv(arg, (I32)(*(I32*)name.data) - RECNO_BASE); \
|
||
|
} \
|
||
|
else { \
|
||
|
my_sv_setpvn(arg, name.data, name.size); \
|
||
|
} \
|
||
|
DBM_ckFilter(arg, filter_fetch_key, "filter_fetch_key") ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define OutputKey_Br(arg, name) \
|
||
|
{ if (RETVAL == 0) \
|
||
|
{ \
|
||
|
if (db->recno_or_queue || db->primary_recno_or_queue \
|
||
|
|| (db->type == DB_BTREE && \
|
||
|
flagSet(DB_GET_RECNO))){ \
|
||
|
sv_setiv(arg, (I32)(*(I32*)name.data) - RECNO_BASE); \
|
||
|
} \
|
||
|
else { \
|
||
|
my_sv_setpvn(arg, name.data, name.size); \
|
||
|
} \
|
||
|
DBM_ckFilter(arg, filter_fetch_key, "filter_fetch_key") ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define OutputKey_Bpr(arg, name) \
|
||
|
{ if (RETVAL == 0) \
|
||
|
{ \
|
||
|
if (db->primary_recno_or_queue \
|
||
|
|| (db->type == DB_BTREE && \
|
||
|
flagSet(DB_GET_RECNO))){ \
|
||
|
sv_setiv(arg, (I32)(*(I32*)name.data) - RECNO_BASE); \
|
||
|
} \
|
||
|
else { \
|
||
|
my_sv_setpvn(arg, name.data, name.size); \
|
||
|
} \
|
||
|
DBM_ckFilter(arg, filter_fetch_key, "filter_fetch_key") ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define SetPartial(data,db) \
|
||
|
data.flags = db->partial ; \
|
||
|
data.dlen = db->dlen ; \
|
||
|
data.doff = db->doff ;
|
||
|
|
||
|
#define ckActive(active, type) \
|
||
|
{ \
|
||
|
if (!active) \
|
||
|
softCrash("%s is already closed", type) ; \
|
||
|
}
|
||
|
|
||
|
#define ckActive_Environment(a) ckActive(a, "Environment")
|
||
|
#define ckActive_TxnMgr(a) ckActive(a, "Transaction Manager")
|
||
|
#define ckActive_Transaction(a) ckActive(a, "Transaction")
|
||
|
#define ckActive_Database(a) ckActive(a, "Database")
|
||
|
#define ckActive_Cursor(a) ckActive(a, "Cursor")
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
#define ckActive_Sequence(a) ckActive(a, "Sequence")
|
||
|
#else
|
||
|
#define ckActive_Sequence(a)
|
||
|
#endif
|
||
|
|
||
|
#define dieIfEnvOpened(e, m) if (e->opened) softCrash("Cannot call method BerkeleyDB::Env::%s after environment has been opened", m);
|
||
|
|
||
|
#define isSTDOUT_ERR(f) ((f) == stdout || (f) == stderr)
|
||
|
|
||
|
|
||
|
/* Internal Global Data */
|
||
|
#define MY_CXT_KEY "BerkeleyDB::_guts" XS_VERSION
|
||
|
|
||
|
typedef struct {
|
||
|
db_recno_t x_Value;
|
||
|
db_recno_t x_zero;
|
||
|
DBTKEY x_empty;
|
||
|
#ifndef AT_LEAST_DB_3_2
|
||
|
BerkeleyDB x_CurrentDB;
|
||
|
#endif
|
||
|
} my_cxt_t;
|
||
|
|
||
|
START_MY_CXT
|
||
|
|
||
|
#define Value (MY_CXT.x_Value)
|
||
|
#define zero (MY_CXT.x_zero)
|
||
|
#define empty (MY_CXT.x_empty)
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
# define CurrentDB ((BerkeleyDB)db->BackRef)
|
||
|
#else
|
||
|
# define CurrentDB (MY_CXT.x_CurrentDB)
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
# define getCurrentDB ((BerkeleyDB)db->BackRef)
|
||
|
# define saveCurrentDB(db)
|
||
|
#else
|
||
|
# define getCurrentDB (MY_CXT.x_CurrentDB)
|
||
|
# define saveCurrentDB(db) (MY_CXT.x_CurrentDB) = db
|
||
|
#endif
|
||
|
|
||
|
#if 0
|
||
|
static char ErrBuff[1000] ;
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
# if PERL_REVISION == 5 && PERL_VERSION <= 4
|
||
|
|
||
|
/* saferealloc in perl5.004 will croak if it is given a NULL pointer*/
|
||
|
void *
|
||
|
MyRealloc(void * ptr, size_t size)
|
||
|
{
|
||
|
if (ptr == NULL )
|
||
|
return safemalloc(size) ;
|
||
|
else
|
||
|
return saferealloc(ptr, size) ;
|
||
|
}
|
||
|
|
||
|
# else
|
||
|
# define MyRealloc saferealloc
|
||
|
# endif
|
||
|
#endif
|
||
|
|
||
|
static char *
|
||
|
my_strdup(const char *s)
|
||
|
{
|
||
|
if (s == NULL)
|
||
|
return NULL ;
|
||
|
|
||
|
{
|
||
|
MEM_SIZE l = strlen(s) + 1;
|
||
|
char *s1 = (char *)safemalloc(l);
|
||
|
|
||
|
Copy(s, s1, (MEM_SIZE)l, char);
|
||
|
return s1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
static char *
|
||
|
db_strerror(int err)
|
||
|
{
|
||
|
if (err == 0)
|
||
|
return "" ;
|
||
|
|
||
|
if (err > 0)
|
||
|
return Strerror(err) ;
|
||
|
|
||
|
switch (err) {
|
||
|
case DB_INCOMPLETE:
|
||
|
return ("DB_INCOMPLETE: Sync was unable to complete");
|
||
|
case DB_KEYEMPTY:
|
||
|
return ("DB_KEYEMPTY: Non-existent key/data pair");
|
||
|
case DB_KEYEXIST:
|
||
|
return ("DB_KEYEXIST: Key/data pair already exists");
|
||
|
case DB_LOCK_DEADLOCK:
|
||
|
return (
|
||
|
"DB_LOCK_DEADLOCK: Locker killed to resolve a deadlock");
|
||
|
case DB_LOCK_NOTGRANTED:
|
||
|
return ("DB_LOCK_NOTGRANTED: Lock not granted");
|
||
|
case DB_LOCK_NOTHELD:
|
||
|
return ("DB_LOCK_NOTHELD: Lock not held by locker");
|
||
|
case DB_NOTFOUND:
|
||
|
return ("DB_NOTFOUND: No matching key/data pair found");
|
||
|
case DB_RUNRECOVERY:
|
||
|
return ("DB_RUNRECOVERY: Fatal error, run database recovery");
|
||
|
default:
|
||
|
return "Unknown Error" ;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
#endif /* DB_VERSION_MAJOR == 2 */
|
||
|
|
||
|
#ifdef TRACE
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
static char *
|
||
|
my_db_strerror(int err)
|
||
|
{
|
||
|
static char buffer[1000] ;
|
||
|
SV * sv = perl_get_sv(ERR_BUFF, FALSE) ;
|
||
|
sprintf(buffer, "%d: %s", err, db_strerror(err)) ;
|
||
|
if (err && sv) {
|
||
|
strcat(buffer, ", ") ;
|
||
|
strcat(buffer, SvPVX(sv)) ;
|
||
|
}
|
||
|
return buffer;
|
||
|
}
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
static void
|
||
|
close_everything(void)
|
||
|
{
|
||
|
dTHR;
|
||
|
Trace(("close_everything\n")) ;
|
||
|
/* Abort All Transactions */
|
||
|
{
|
||
|
BerkeleyDB__Txn__Raw tid ;
|
||
|
HE * he ;
|
||
|
I32 len ;
|
||
|
HV * hv = perl_get_hv("BerkeleyDB::Term::Txn", TRUE);
|
||
|
int all = 0 ;
|
||
|
int closed = 0 ;
|
||
|
(void)hv_iterinit(hv) ;
|
||
|
Trace(("BerkeleyDB::Term::close_all_txns dirty=%d\n", PL_dirty)) ;
|
||
|
while ( (he = hv_iternext(hv)) ) {
|
||
|
tid = * (BerkeleyDB__Txn__Raw *) hv_iterkey(he, &len) ;
|
||
|
Trace((" Aborting Transaction [%d] in [%d] Active [%d]\n", tid->txn, tid, tid->active));
|
||
|
if (tid->active) {
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
tid->txn->abort(tid->txn) ;
|
||
|
#else
|
||
|
txn_abort(tid->txn);
|
||
|
#endif
|
||
|
++ closed ;
|
||
|
}
|
||
|
tid->active = FALSE ;
|
||
|
++ all ;
|
||
|
}
|
||
|
Trace(("End of BerkeleyDB::Term::close_all_txns aborted %d of %d transactios\n",closed, all)) ;
|
||
|
}
|
||
|
|
||
|
/* Close All Cursors */
|
||
|
{
|
||
|
BerkeleyDB__Cursor db ;
|
||
|
HE * he ;
|
||
|
I32 len ;
|
||
|
HV * hv = perl_get_hv("BerkeleyDB::Term::Cursor", TRUE);
|
||
|
int all = 0 ;
|
||
|
int closed = 0 ;
|
||
|
(void) hv_iterinit(hv) ;
|
||
|
Trace(("BerkeleyDB::Term::close_all_cursors \n")) ;
|
||
|
while ( (he = hv_iternext(hv)) ) {
|
||
|
db = * (BerkeleyDB__Cursor*) hv_iterkey(he, &len) ;
|
||
|
Trace((" Closing Cursor [%d] in [%d] Active [%d]\n", db->cursor, db, db->active));
|
||
|
if (db->active) {
|
||
|
((db->cursor)->c_close)(db->cursor) ;
|
||
|
++ closed ;
|
||
|
}
|
||
|
db->active = FALSE ;
|
||
|
++ all ;
|
||
|
}
|
||
|
Trace(("End of BerkeleyDB::Term::close_all_cursors closed %d of %d cursors\n",closed, all)) ;
|
||
|
}
|
||
|
|
||
|
/* Close All Databases */
|
||
|
{
|
||
|
BerkeleyDB db ;
|
||
|
HE * he ;
|
||
|
I32 len ;
|
||
|
HV * hv = perl_get_hv("BerkeleyDB::Term::Db", TRUE);
|
||
|
int all = 0 ;
|
||
|
int closed = 0 ;
|
||
|
(void)hv_iterinit(hv) ;
|
||
|
Trace(("BerkeleyDB::Term::close_all_dbs\n" )) ;
|
||
|
while ( (he = hv_iternext(hv)) ) {
|
||
|
db = * (BerkeleyDB*) hv_iterkey(he, &len) ;
|
||
|
Trace((" Closing Database [%d] in [%d] Active [%d]\n", db->dbp, db, db->active));
|
||
|
if (db->active) {
|
||
|
(db->dbp->close)(db->dbp, 0) ;
|
||
|
++ closed ;
|
||
|
}
|
||
|
db->active = FALSE ;
|
||
|
++ all ;
|
||
|
}
|
||
|
Trace(("End of BerkeleyDB::Term::close_all_dbs closed %d of %d dbs\n",closed, all)) ;
|
||
|
}
|
||
|
|
||
|
/* Close All Environments */
|
||
|
{
|
||
|
BerkeleyDB__Env env ;
|
||
|
HE * he ;
|
||
|
I32 len ;
|
||
|
HV * hv = perl_get_hv("BerkeleyDB::Term::Env", TRUE);
|
||
|
int all = 0 ;
|
||
|
int closed = 0 ;
|
||
|
(void)hv_iterinit(hv) ;
|
||
|
Trace(("BerkeleyDB::Term::close_all_envs\n")) ;
|
||
|
while ( (he = hv_iternext(hv)) ) {
|
||
|
env = * (BerkeleyDB__Env*) hv_iterkey(he, &len) ;
|
||
|
Trace((" Closing Environment [%d] in [%d] Active [%d]\n", env->Env, env, env->active));
|
||
|
if (env->active) {
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
db_appexit(env->Env) ;
|
||
|
#else
|
||
|
(env->Env->close)(env->Env, 0) ;
|
||
|
#endif
|
||
|
++ closed ;
|
||
|
}
|
||
|
env->active = FALSE ;
|
||
|
++ all ;
|
||
|
}
|
||
|
Trace(("End of BerkeleyDB::Term::close_all_envs closed %d of %d dbs\n",closed, all)) ;
|
||
|
}
|
||
|
|
||
|
Trace(("end close_everything\n")) ;
|
||
|
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
destroyDB(BerkeleyDB db)
|
||
|
{
|
||
|
dTHR;
|
||
|
if (! PL_dirty && db->active) {
|
||
|
if (db->parent_env && db->parent_env->open_dbs)
|
||
|
-- db->parent_env->open_dbs ;
|
||
|
-- db->open_cursors ;
|
||
|
((db->dbp)->close)(db->dbp, 0) ;
|
||
|
}
|
||
|
if (db->hash)
|
||
|
SvREFCNT_dec(db->hash) ;
|
||
|
if (db->compare)
|
||
|
SvREFCNT_dec(db->compare) ;
|
||
|
if (db->dup_compare)
|
||
|
SvREFCNT_dec(db->dup_compare) ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
if (db->associated && !db->secondary_db)
|
||
|
SvREFCNT_dec(db->associated) ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
if (db->associated_foreign)
|
||
|
SvREFCNT_dec(db->associated_foreign) ;
|
||
|
#endif
|
||
|
if (db->prefix)
|
||
|
SvREFCNT_dec(db->prefix) ;
|
||
|
#ifdef DBM_FILTERING
|
||
|
if (db->filter_fetch_key)
|
||
|
SvREFCNT_dec(db->filter_fetch_key) ;
|
||
|
if (db->filter_store_key)
|
||
|
SvREFCNT_dec(db->filter_store_key) ;
|
||
|
if (db->filter_fetch_value)
|
||
|
SvREFCNT_dec(db->filter_fetch_value) ;
|
||
|
if (db->filter_store_value)
|
||
|
SvREFCNT_dec(db->filter_store_value) ;
|
||
|
#endif
|
||
|
hash_delete("BerkeleyDB::Term::Db", (char *)db) ;
|
||
|
if (db->filename)
|
||
|
Safefree(db->filename) ;
|
||
|
Safefree(db) ;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
softCrash(const char *pat, ...)
|
||
|
{
|
||
|
char buffer1 [500] ;
|
||
|
char buffer2 [500] ;
|
||
|
va_list args;
|
||
|
va_start(args, pat);
|
||
|
|
||
|
Trace(("softCrash: %s\n", pat)) ;
|
||
|
|
||
|
#define ABORT_PREFIX "BerkeleyDB Aborting: "
|
||
|
|
||
|
/* buffer = (char*) safemalloc(strlen(pat) + strlen(ABORT_PREFIX) + 1) ; */
|
||
|
strcpy(buffer1, ABORT_PREFIX) ;
|
||
|
strcat(buffer1, pat) ;
|
||
|
|
||
|
vsprintf(buffer2, buffer1, args) ;
|
||
|
|
||
|
croak(buffer2);
|
||
|
|
||
|
/* NOTREACHED */
|
||
|
va_end(args);
|
||
|
return 1 ;
|
||
|
}
|
||
|
|
||
|
|
||
|
static I32
|
||
|
GetArrayLength(BerkeleyDB db)
|
||
|
{
|
||
|
DBT key ;
|
||
|
DBT value ;
|
||
|
int RETVAL = 0 ;
|
||
|
DBC * cursor ;
|
||
|
|
||
|
DBT_clear(key) ;
|
||
|
DBT_clear(value) ;
|
||
|
#if DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 6
|
||
|
if ( ((db->dbp)->cursor)(db->dbp, db->txn, &cursor) == 0 )
|
||
|
#else
|
||
|
if ( ((db->dbp)->cursor)(db->dbp, db->txn, &cursor, 0) == 0 )
|
||
|
#endif
|
||
|
{
|
||
|
RETVAL = cursor->c_get(cursor, &key, &value, DB_LAST) ;
|
||
|
if (RETVAL == 0)
|
||
|
RETVAL = *(I32 *)key.data ;
|
||
|
else /* No key means empty file */
|
||
|
RETVAL = 0 ;
|
||
|
cursor->c_close(cursor) ;
|
||
|
}
|
||
|
|
||
|
Trace(("GetArrayLength got %d\n", RETVAL)) ;
|
||
|
return ((I32)RETVAL) ;
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
|
||
|
#define GetRecnoKey(db, value) _GetRecnoKey(db, value)
|
||
|
|
||
|
static db_recno_t
|
||
|
_GetRecnoKey(BerkeleyDB db, I32 value)
|
||
|
{
|
||
|
Trace(("GetRecnoKey start value = %d\n", value)) ;
|
||
|
if (db->recno_or_queue && value < 0) {
|
||
|
/* Get the length of the array */
|
||
|
I32 length = GetArrayLength(db) ;
|
||
|
|
||
|
/* check for attempt to write before start of array */
|
||
|
if (length + value + RECNO_BASE <= 0)
|
||
|
softCrash("Modification of non-creatable array value attempted, subscript %ld", (long)value) ;
|
||
|
|
||
|
value = length + value + RECNO_BASE ;
|
||
|
}
|
||
|
else
|
||
|
++ value ;
|
||
|
|
||
|
Trace(("GetRecnoKey end value = %d\n", value)) ;
|
||
|
|
||
|
return value ;
|
||
|
}
|
||
|
|
||
|
#else /* ! 0 */
|
||
|
|
||
|
#if 0
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
#define GetRecnoKey(db, value) _GetRecnoKey(db, value)
|
||
|
|
||
|
static db_recno_t
|
||
|
_GetRecnoKey(BerkeleyDB db, I32 value)
|
||
|
{
|
||
|
if (value + RECNO_BASE < 1)
|
||
|
softCrash("key value %d < base (%d)", (value), RECNO_BASE?0:1) ;
|
||
|
return value + RECNO_BASE ;
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
#endif /* 0 */
|
||
|
|
||
|
#define GetRecnoKey(db, value) ((value) + RECNO_BASE )
|
||
|
|
||
|
#endif /* 0 */
|
||
|
|
||
|
#if 0
|
||
|
static SV *
|
||
|
GetInternalObject(SV * sv)
|
||
|
{
|
||
|
SV * info = (SV*) NULL ;
|
||
|
SV * s ;
|
||
|
MAGIC * mg ;
|
||
|
|
||
|
Trace(("in GetInternalObject %d\n", sv)) ;
|
||
|
if (sv == NULL || !SvROK(sv))
|
||
|
return NULL ;
|
||
|
|
||
|
s = SvRV(sv) ;
|
||
|
if (SvMAGICAL(s))
|
||
|
{
|
||
|
if (SvTYPE(s) == SVt_PVHV || SvTYPE(s) == SVt_PVAV)
|
||
|
mg = mg_find(s, 'P') ;
|
||
|
else
|
||
|
mg = mg_find(s, 'q') ;
|
||
|
|
||
|
/* all this testing is probably overkill, but till I know more
|
||
|
about global destruction it stays.
|
||
|
*/
|
||
|
/* if (mg && mg->mg_obj && SvRV(mg->mg_obj) && SvPVX(SvRV(mg->mg_obj))) */
|
||
|
if (mg && mg->mg_obj && SvRV(mg->mg_obj) )
|
||
|
info = SvRV(mg->mg_obj) ;
|
||
|
else
|
||
|
info = s ;
|
||
|
}
|
||
|
|
||
|
Trace(("end of GetInternalObject %d\n", info)) ;
|
||
|
return info ;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int
|
||
|
btree_compare(DB_callback const DBT * key1, const DBT * key2 )
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * data1, * data2 ;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
/* BerkeleyDB keepDB = getCurrentDB ; */
|
||
|
|
||
|
Trace(("In btree_compare \n")) ;
|
||
|
data1 = (char*) key1->data ;
|
||
|
data2 = (char*) key2->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (key1->size == 0)
|
||
|
data1 = "" ;
|
||
|
if (key2->size == 0)
|
||
|
data2 = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
/* SAVESPTR(CurrentDB); */
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,2) ;
|
||
|
PUSHs(sv_2mortal(newSVpvn(data1,key1->size)));
|
||
|
PUSHs(sv_2mortal(newSVpvn(data2,key2->size)));
|
||
|
PUTBACK ;
|
||
|
|
||
|
count = perl_call_sv(getCurrentDB->compare, G_SCALAR);
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("in btree_compare - expected 1 return value from compare sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
/* CurrentDB = keepDB ; */
|
||
|
return (retval) ;
|
||
|
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
dup_compare(DB_callback const DBT * key1, const DBT * key2 )
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * data1, * data2 ;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
/* BerkeleyDB keepDB = CurrentDB ; */
|
||
|
|
||
|
Trace(("In dup_compare \n")) ;
|
||
|
if (!getCurrentDB)
|
||
|
softCrash("Internal Error - No CurrentDB in dup_compare") ;
|
||
|
if (getCurrentDB->dup_compare == NULL)
|
||
|
|
||
|
|
||
|
softCrash("in dup_compare: no callback specified for database '%s'", getCurrentDB->filename) ;
|
||
|
|
||
|
data1 = (char*) key1->data ;
|
||
|
data2 = (char*) key2->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (key1->size == 0)
|
||
|
data1 = "" ;
|
||
|
if (key2->size == 0)
|
||
|
data2 = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,2) ;
|
||
|
PUSHs(sv_2mortal(newSVpvn(data1,key1->size)));
|
||
|
PUSHs(sv_2mortal(newSVpvn(data2,key2->size)));
|
||
|
PUTBACK ;
|
||
|
|
||
|
count = perl_call_sv(getCurrentDB->dup_compare, G_SCALAR);
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("dup_compare: expected 1 return value from compare sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
/* CurrentDB = keepDB ; */
|
||
|
return (retval) ;
|
||
|
|
||
|
}
|
||
|
|
||
|
static size_t
|
||
|
btree_prefix(DB_callback const DBT * key1, const DBT * key2 )
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * data1, * data2 ;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
/* BerkeleyDB keepDB = CurrentDB ; */
|
||
|
|
||
|
Trace(("In btree_prefix \n")) ;
|
||
|
data1 = (char*) key1->data ;
|
||
|
data2 = (char*) key2->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (key1->size == 0)
|
||
|
data1 = "" ;
|
||
|
if (key2->size == 0)
|
||
|
data2 = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,2) ;
|
||
|
PUSHs(sv_2mortal(newSVpvn(data1,key1->size)));
|
||
|
PUSHs(sv_2mortal(newSVpvn(data2,key2->size)));
|
||
|
PUTBACK ;
|
||
|
|
||
|
count = perl_call_sv(getCurrentDB->prefix, G_SCALAR);
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("btree_prefix: expected 1 return value from prefix sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
/* CurrentDB = keepDB ; */
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
static u_int32_t
|
||
|
hash_cb(DB_callback const void * data, u_int32_t size)
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
/* BerkeleyDB keepDB = CurrentDB ; */
|
||
|
|
||
|
Trace(("In hash_cb \n")) ;
|
||
|
#ifndef newSVpvn
|
||
|
if (size == 0)
|
||
|
data = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
|
||
|
XPUSHs(sv_2mortal(newSVpvn((char*)data,size)));
|
||
|
PUTBACK ;
|
||
|
|
||
|
count = perl_call_sv(getCurrentDB->hash, G_SCALAR);
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("hash_cb: expected 1 return value from hash sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
/* CurrentDB = keepDB ; */
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
|
||
|
static int
|
||
|
associate_cb(DB_callback const DBT * pkey, const DBT * pdata, DBT * skey)
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * pk_dat, * pd_dat ;
|
||
|
/* char *sk_dat ; */
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
int i ;
|
||
|
SV * skey_SV ;
|
||
|
STRLEN skey_len;
|
||
|
char * skey_ptr ;
|
||
|
AV * skey_AV;
|
||
|
DBT * tkey;
|
||
|
|
||
|
Trace(("In associate_cb \n")) ;
|
||
|
if (getCurrentDB->associated == NULL){
|
||
|
Trace(("No Callback registered\n")) ;
|
||
|
return EINVAL ;
|
||
|
}
|
||
|
|
||
|
skey_SV = newSVpv("",0);
|
||
|
|
||
|
|
||
|
pk_dat = (char*) pkey->data ;
|
||
|
pd_dat = (char*) pdata->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (pkey->size == 0)
|
||
|
pk_dat = "" ;
|
||
|
if (pdata->size == 0)
|
||
|
pd_dat = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,3) ;
|
||
|
PUSHs(sv_2mortal(newSVpvn(pk_dat,pkey->size)));
|
||
|
PUSHs(sv_2mortal(newSVpvn(pd_dat,pdata->size)));
|
||
|
PUSHs(sv_2mortal(skey_SV));
|
||
|
PUTBACK ;
|
||
|
|
||
|
Trace(("calling associated cb\n"));
|
||
|
count = perl_call_sv(getCurrentDB->associated, G_SCALAR);
|
||
|
Trace(("called associated cb\n"));
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("associate: expected 1 return value from prefix sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
|
||
|
if (retval != DB_DONOTINDEX)
|
||
|
{
|
||
|
/* retrieve the secondary key */
|
||
|
DBT_clear(*skey);
|
||
|
|
||
|
skey->flags = DB_DBT_APPMALLOC;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_6
|
||
|
if ( SvROK(skey_SV) ) {
|
||
|
SV *rv = SvRV(skey_SV);
|
||
|
|
||
|
if ( SvTYPE(rv) == SVt_PVAV ) {
|
||
|
AV *av = (AV *)rv;
|
||
|
SV **svs = AvARRAY(av);
|
||
|
I32 len = av_len(av) + 1;
|
||
|
I32 i;
|
||
|
DBT *dbts;
|
||
|
|
||
|
if ( len == 0 ) {
|
||
|
retval = DB_DONOTINDEX;
|
||
|
} else if ( len == 1 ) {
|
||
|
skey_ptr = SvPV(svs[0], skey_len);
|
||
|
skey->size = skey_len;
|
||
|
skey->data = (char*)safemalloc(skey_len);
|
||
|
memcpy(skey->data, skey_ptr, skey_len);
|
||
|
Trace(("key is %d -- %.*s\n", skey->size, skey->size, skey->data));
|
||
|
} else {
|
||
|
skey->flags |= DB_DBT_MULTIPLE ;
|
||
|
|
||
|
/* FIXME this will leak if safemalloc fails later... do we care? */
|
||
|
dbts = (DBT *) safemalloc(sizeof(DBT) * len);
|
||
|
skey->size = len;
|
||
|
skey->data = (char *)dbts;
|
||
|
|
||
|
for ( i = 0; i < skey->size; i ++ ) {
|
||
|
skey_ptr = SvPV(svs[i], skey_len);
|
||
|
|
||
|
dbts[i].flags = DB_DBT_APPMALLOC;
|
||
|
dbts[i].size = skey_len;
|
||
|
dbts[i].data = (char *)safemalloc(skey_len);
|
||
|
memcpy(dbts[i].data, skey_ptr, skey_len);
|
||
|
|
||
|
Trace(("key is %d -- %.*s\n", dbts[i].size, dbts[i].size, dbts[i].data));
|
||
|
}
|
||
|
Trace(("mkey has %d subkeys\n", skey->size));
|
||
|
}
|
||
|
} else {
|
||
|
croak("Not an array reference");
|
||
|
}
|
||
|
} else
|
||
|
#endif
|
||
|
{
|
||
|
skey_ptr = SvPV(skey_SV, skey_len);
|
||
|
/* skey->size = SvCUR(skey_SV); */
|
||
|
/* skey->data = (char*)safemalloc(skey->size); */
|
||
|
skey->size = skey_len;
|
||
|
skey->data = (char*)safemalloc(skey_len);
|
||
|
memcpy(skey->data, skey_ptr, skey_len);
|
||
|
}
|
||
|
}
|
||
|
Trace(("key is %d -- %.*s\n", skey->size, skey->size, skey->data));
|
||
|
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
associate_cb_recno(DB_callback const DBT * pkey, const DBT * pdata, DBT * skey)
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * pk_dat, * pd_dat ;
|
||
|
/* char *sk_dat ; */
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
SV * skey_SV ;
|
||
|
STRLEN skey_len;
|
||
|
char * skey_ptr ;
|
||
|
/* db_recno_t Value; */
|
||
|
|
||
|
Trace(("In associate_cb_recno \n")) ;
|
||
|
if (getCurrentDB->associated == NULL){
|
||
|
Trace(("No Callback registered\n")) ;
|
||
|
return EINVAL ;
|
||
|
}
|
||
|
|
||
|
skey_SV = newSVpv("",0);
|
||
|
|
||
|
|
||
|
pk_dat = (char*) pkey->data ;
|
||
|
pd_dat = (char*) pdata->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (pkey->size == 0)
|
||
|
pk_dat = "" ;
|
||
|
if (pdata->size == 0)
|
||
|
pd_dat = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,2) ;
|
||
|
PUSHs(sv_2mortal(newSVpvn(pk_dat,pkey->size)));
|
||
|
PUSHs(sv_2mortal(newSVpvn(pd_dat,pdata->size)));
|
||
|
PUSHs(sv_2mortal(skey_SV));
|
||
|
PUTBACK ;
|
||
|
|
||
|
Trace(("calling associated cb\n"));
|
||
|
count = perl_call_sv(getCurrentDB->associated, G_SCALAR);
|
||
|
Trace(("called associated cb\n"));
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("associate: expected 1 return value from prefix sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
|
||
|
/* retrieve the secondary key */
|
||
|
DBT_clear(*skey);
|
||
|
|
||
|
if (retval != DB_DONOTINDEX)
|
||
|
{
|
||
|
Value = GetRecnoKey(getCurrentDB, SvIV(skey_SV)) ;
|
||
|
skey->flags = DB_DBT_APPMALLOC;
|
||
|
skey->size = (int)sizeof(db_recno_t);
|
||
|
skey->data = (char*)safemalloc(skey->size);
|
||
|
memcpy(skey->data, &Value, skey->size);
|
||
|
}
|
||
|
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
#endif /* AT_LEAST_DB_3_3 */
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
|
||
|
typedef int (*bt_compress_fcn_type)(DB *db, const DBT *prevKey,
|
||
|
const DBT *prevData, const DBT *key, const DBT *data, DBT *dest);
|
||
|
|
||
|
typedef int (*bt_decompress_fcn_type)(DB *db, const DBT *prevKey,
|
||
|
const DBT *prevData, DBT *compressed, DBT *destKey, DBT *destData);
|
||
|
|
||
|
#endif /* AT_LEAST_DB_4_8 */
|
||
|
|
||
|
typedef int (*foreign_cb_type)(DB *, const DBT *, DBT *, const DBT *, int *) ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
|
||
|
static int
|
||
|
associate_foreign_cb(DB* db, const DBT * key, DBT * data, const DBT * foreignkey, int* changed)
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * k_dat, * d_dat, * f_dat;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
int i ;
|
||
|
SV * changed_SV ;
|
||
|
STRLEN skey_len;
|
||
|
char * skey_ptr ;
|
||
|
AV * skey_AV;
|
||
|
DBT * tkey;
|
||
|
SV* data_sv ;
|
||
|
|
||
|
Trace(("In associate_foreign_cb \n")) ;
|
||
|
if (getCurrentDB->associated_foreign == NULL){
|
||
|
Trace(("No Callback registered\n")) ;
|
||
|
return EINVAL ;
|
||
|
}
|
||
|
|
||
|
changed_SV = newSViv(*changed);
|
||
|
|
||
|
|
||
|
k_dat = (char*) key->data ;
|
||
|
d_dat = (char*) data->data ;
|
||
|
f_dat = (char*) foreignkey->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (key->size == 0)
|
||
|
k_dat = "" ;
|
||
|
if (data->size == 0)
|
||
|
d_dat = "" ;
|
||
|
if (foreignkey->size == 0)
|
||
|
f_dat = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,4) ;
|
||
|
|
||
|
PUSHs(sv_2mortal(newSVpvn(k_dat,key->size)));
|
||
|
data_sv = newSVpv(d_dat, data->size);
|
||
|
PUSHs(sv_2mortal(data_sv));
|
||
|
PUSHs(sv_2mortal(newSVpvn(f_dat,foreignkey->size)));
|
||
|
PUSHs(sv_2mortal(changed_SV));
|
||
|
PUTBACK ;
|
||
|
|
||
|
Trace(("calling associated cb\n"));
|
||
|
count = perl_call_sv(getCurrentDB->associated_foreign, G_SCALAR);
|
||
|
Trace(("called associated cb\n"));
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("associate_foreign: expected 1 return value from prefix sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
|
||
|
*changed = SvIV(changed_SV);
|
||
|
|
||
|
if (*changed)
|
||
|
{
|
||
|
DBT_clear(*data);
|
||
|
data->flags = DB_DBT_APPMALLOC;
|
||
|
skey_ptr = SvPV(data_sv, skey_len);
|
||
|
data->size = skey_len;
|
||
|
data->data = (char*)safemalloc(skey_len);
|
||
|
memcpy(data->data, skey_ptr, skey_len);
|
||
|
}
|
||
|
/*Trace(("key is %d -- %.*s\n", skey->size, skey->size, skey->data));*/
|
||
|
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
associate_foreign_cb_recno(DB* db, const DBT * key, DBT * data, const DBT * foreignkey, int* changed)
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
dSP ;
|
||
|
dMY_CXT ;
|
||
|
char * k_dat, * d_dat, * f_dat;
|
||
|
int retval ;
|
||
|
int count ;
|
||
|
int i ;
|
||
|
SV * changed_SV ;
|
||
|
STRLEN skey_len;
|
||
|
char * skey_ptr ;
|
||
|
AV * skey_AV;
|
||
|
DBT * tkey;
|
||
|
SV* data_sv ;
|
||
|
|
||
|
Trace(("In associate_foreign_cb \n")) ;
|
||
|
if (getCurrentDB->associated_foreign == NULL){
|
||
|
Trace(("No Callback registered\n")) ;
|
||
|
return EINVAL ;
|
||
|
}
|
||
|
|
||
|
changed_SV = newSViv(*changed);
|
||
|
|
||
|
|
||
|
k_dat = (char*) key->data ;
|
||
|
d_dat = (char*) data->data ;
|
||
|
f_dat = (char*) foreignkey->data ;
|
||
|
|
||
|
#ifndef newSVpvn
|
||
|
/* As newSVpv will assume that the data pointer is a null terminated C
|
||
|
string if the size parameter is 0, make sure that data points to an
|
||
|
empty string if the length is 0
|
||
|
*/
|
||
|
if (key->size == 0)
|
||
|
k_dat = "" ;
|
||
|
if (data->size == 0)
|
||
|
d_dat = "" ;
|
||
|
if (foreignkey->size == 0)
|
||
|
f_dat = "" ;
|
||
|
#endif
|
||
|
|
||
|
ENTER ;
|
||
|
SAVETMPS;
|
||
|
|
||
|
PUSHMARK(SP) ;
|
||
|
EXTEND(SP,4) ;
|
||
|
|
||
|
PUSHs(sv_2mortal(newSVpvn(k_dat,key->size)));
|
||
|
data_sv = newSVpv(d_dat, data->size);
|
||
|
PUSHs(sv_2mortal(data_sv));
|
||
|
PUSHs(sv_2mortal(newSVpvn(f_dat,foreignkey->size)));
|
||
|
PUSHs(sv_2mortal(changed_SV));
|
||
|
PUTBACK ;
|
||
|
|
||
|
Trace(("calling associated cb\n"));
|
||
|
count = perl_call_sv(getCurrentDB->associated_foreign, G_SCALAR);
|
||
|
Trace(("called associated cb\n"));
|
||
|
|
||
|
SPAGAIN ;
|
||
|
|
||
|
if (count != 1)
|
||
|
softCrash ("associate_foreign: expected 1 return value from prefix sub, got %d", count) ;
|
||
|
|
||
|
retval = POPi ;
|
||
|
|
||
|
PUTBACK ;
|
||
|
|
||
|
*changed = SvIV(changed_SV);
|
||
|
|
||
|
if (*changed)
|
||
|
{
|
||
|
DBT_clear(*data);
|
||
|
Value = GetRecnoKey(getCurrentDB, SvIV(data_sv)) ;
|
||
|
data->flags = DB_DBT_APPMALLOC;
|
||
|
data->size = (int)sizeof(db_recno_t);
|
||
|
data->data = (char*)safemalloc(data->size);
|
||
|
memcpy(data->data, &Value, data->size);
|
||
|
}
|
||
|
/*Trace(("key is %d -- %.*s\n", skey->size, skey->size, skey->data));*/
|
||
|
|
||
|
FREETMPS ;
|
||
|
LEAVE ;
|
||
|
|
||
|
return (retval) ;
|
||
|
}
|
||
|
|
||
|
#endif /* AT_LEAST_DB_3_3 */
|
||
|
|
||
|
static void
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
db_errcall_cb(const DB_ENV* dbenv, const char * db_errpfx, const char * buffer)
|
||
|
#else
|
||
|
db_errcall_cb(const char * db_errpfx, char * buffer)
|
||
|
#endif
|
||
|
{
|
||
|
SV * sv;
|
||
|
|
||
|
Trace(("In errcall_cb \n")) ;
|
||
|
#if 0
|
||
|
|
||
|
if (db_errpfx == NULL)
|
||
|
db_errpfx = "" ;
|
||
|
if (buffer == NULL )
|
||
|
buffer = "" ;
|
||
|
ErrBuff[0] = '\0';
|
||
|
if (strlen(db_errpfx) + strlen(buffer) + 3 <= 1000) {
|
||
|
if (*db_errpfx != '\0') {
|
||
|
strcat(ErrBuff, db_errpfx) ;
|
||
|
strcat(ErrBuff, ": ") ;
|
||
|
}
|
||
|
strcat(ErrBuff, buffer) ;
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
sv = perl_get_sv(ERR_BUFF, FALSE) ;
|
||
|
if (sv) {
|
||
|
if (db_errpfx)
|
||
|
sv_setpvf(sv, "%s: %s", db_errpfx, buffer) ;
|
||
|
else
|
||
|
sv_setpv(sv, buffer) ;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if defined(AT_LEAST_DB_4_4) && ! defined(_WIN32)
|
||
|
|
||
|
int
|
||
|
db_isalive_cb(DB_ENV *dbenv, pid_t pid, db_threadid_t tid, u_int32_t flags)
|
||
|
{
|
||
|
bool processAlive = ( kill(pid, 0) == 0 ) || ( errno != ESRCH );
|
||
|
return processAlive;
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
|
||
|
static SV *
|
||
|
readHash(HV * hash, char * key)
|
||
|
{
|
||
|
SV ** svp;
|
||
|
svp = hv_fetch(hash, key, strlen(key), FALSE);
|
||
|
|
||
|
if (svp) {
|
||
|
if (SvGMAGICAL(*svp))
|
||
|
mg_get(*svp);
|
||
|
if (SvOK(*svp))
|
||
|
return *svp;
|
||
|
}
|
||
|
|
||
|
return NULL ;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
hash_delete(char * hash, char * key)
|
||
|
{
|
||
|
HV * hv = perl_get_hv(hash, TRUE);
|
||
|
(void) hv_delete(hv, (char*)&key, sizeof(key), G_DISCARD);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
hash_store_iv(char * hash, char * key, IV value)
|
||
|
{
|
||
|
HV * hv = perl_get_hv(hash, TRUE);
|
||
|
(void)hv_store(hv, (char*)&key, sizeof(key), newSViv(value), 0);
|
||
|
/* printf("hv_store returned %d\n", ret) ; */
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
hv_store_iv(HV * hash, char * key, IV value)
|
||
|
{
|
||
|
hv_store(hash, key, strlen(key), newSViv(value), 0);
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
static void
|
||
|
hv_store_uv(HV * hash, char * key, UV value)
|
||
|
{
|
||
|
hv_store(hash, key, strlen(key), newSVuv(value), 0);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static void
|
||
|
GetKey(BerkeleyDB_type * db, SV * sv, DBTKEY * key)
|
||
|
{
|
||
|
dMY_CXT ;
|
||
|
if (db->recno_or_queue) {
|
||
|
Value = GetRecnoKey(db, SvIV(sv)) ;
|
||
|
key->data = & Value;
|
||
|
key->size = (int)sizeof(db_recno_t);
|
||
|
}
|
||
|
else {
|
||
|
key->data = SvPV(sv, PL_na);
|
||
|
key->size = (int)PL_na;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static BerkeleyDB
|
||
|
my_db_open(
|
||
|
BerkeleyDB db ,
|
||
|
SV * ref,
|
||
|
SV * ref_dbenv ,
|
||
|
BerkeleyDB__Env dbenv ,
|
||
|
BerkeleyDB__Txn txn,
|
||
|
const char * file,
|
||
|
const char * subname,
|
||
|
DBTYPE type,
|
||
|
int flags,
|
||
|
int mode,
|
||
|
DB_INFO * info,
|
||
|
char * password,
|
||
|
int enc_flags,
|
||
|
HV* hash
|
||
|
)
|
||
|
{
|
||
|
DB_ENV * env = NULL ;
|
||
|
BerkeleyDB RETVAL = NULL ;
|
||
|
DB * dbp ;
|
||
|
int Status ;
|
||
|
DB_TXN* txnid = NULL ;
|
||
|
dMY_CXT;
|
||
|
|
||
|
Trace(("_db_open(dbenv[%p] ref_dbenv [%p] file[%s] subname [%s] type[%d] flags[%d] mode[%d]\n",
|
||
|
dbenv, ref_dbenv, file, subname, type, flags, mode)) ;
|
||
|
|
||
|
|
||
|
if (dbenv)
|
||
|
env = dbenv->Env ;
|
||
|
|
||
|
if (txn)
|
||
|
txnid = txn->txn;
|
||
|
|
||
|
Trace(("_db_open(dbenv[%p] ref_dbenv [%p] txn [%p] file[%s] subname [%s] type[%d] flags[%d] mode[%d]\n",
|
||
|
dbenv, ref_dbenv, txn, file, subname, type, flags, mode)) ;
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
if (subname)
|
||
|
softCrash("Subname needs Berkeley DB 3 or better") ;
|
||
|
#endif
|
||
|
|
||
|
#ifndef AT_LEAST_DB_4_1
|
||
|
if (password)
|
||
|
softCrash("-Encrypt needs Berkeley DB 4.x or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_4_1 */
|
||
|
|
||
|
#ifndef AT_LEAST_DB_3_2
|
||
|
CurrentDB = db ;
|
||
|
#endif
|
||
|
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
Trace(("creating\n"));
|
||
|
Status = db_create(&dbp, env, 0) ;
|
||
|
Trace(("db_create returned %s\n", my_db_strerror(Status))) ;
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
dbp->BackRef = db;
|
||
|
#endif
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
if (! env) {
|
||
|
dbp->set_alloc(dbp, safemalloc, MyRealloc, safefree) ;
|
||
|
dbp->set_errcall(dbp, db_errcall_cb) ;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
{
|
||
|
/* Btree Compression */
|
||
|
SV* sv;
|
||
|
SV* wanted = NULL;
|
||
|
|
||
|
SetValue_sv(wanted, "set_bt_compress") ;
|
||
|
|
||
|
if (wanted)
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_8
|
||
|
softCrash("set_bt_compress needs Berkeley DB 4.8 or better") ;
|
||
|
#else
|
||
|
bt_compress_fcn_type c = NULL;
|
||
|
bt_decompress_fcn_type u = NULL;
|
||
|
/*
|
||
|
SV* compress = NULL;
|
||
|
SV* uncompress = NULL;
|
||
|
|
||
|
SetValue_sv(compress, "_btcompress1") ;
|
||
|
SetValue_sv(uncompress, "_btcompress2") ;
|
||
|
if (compress)
|
||
|
{
|
||
|
c = ;
|
||
|
db->bt_compress = newSVsv(compress) ;
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
Status = dbp->set_bt_compress(dbp, c, u);
|
||
|
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
#endif /* AT_LEAST_DB_4_8 */
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
/* set encryption */
|
||
|
if (password)
|
||
|
{
|
||
|
Status = dbp->set_encrypt(dbp, password, enc_flags);
|
||
|
Trace(("DB->set_encrypt passwd = %s, flags %d returned %s\n",
|
||
|
password, enc_flags,
|
||
|
my_db_strerror(Status))) ;
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (info->re_source) {
|
||
|
Status = dbp->set_re_source(dbp, info->re_source) ;
|
||
|
Trace(("set_re_source [%s] returned %s\n",
|
||
|
info->re_source, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->db_cachesize) {
|
||
|
Status = dbp->set_cachesize(dbp, 0, info->db_cachesize, 0) ;
|
||
|
Trace(("set_cachesize [%d] returned %s\n",
|
||
|
info->db_cachesize, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->db_lorder) {
|
||
|
Status = dbp->set_lorder(dbp, info->db_lorder) ;
|
||
|
Trace(("set_lorder [%d] returned %s\n",
|
||
|
info->db_lorder, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->db_pagesize) {
|
||
|
Status = dbp->set_pagesize(dbp, info->db_pagesize) ;
|
||
|
Trace(("set_pagesize [%d] returned %s\n",
|
||
|
info->db_pagesize, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->h_ffactor) {
|
||
|
Status = dbp->set_h_ffactor(dbp, info->h_ffactor) ;
|
||
|
Trace(("set_h_ffactor [%d] returned %s\n",
|
||
|
info->h_ffactor, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->h_nelem) {
|
||
|
Status = dbp->set_h_nelem(dbp, info->h_nelem) ;
|
||
|
Trace(("set_h_nelem [%d] returned %s\n",
|
||
|
info->h_nelem, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->bt_minkey) {
|
||
|
Status = dbp->set_bt_minkey(dbp, info->bt_minkey) ;
|
||
|
Trace(("set_bt_minkey [%d] returned %s\n",
|
||
|
info->bt_minkey, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->bt_compare) {
|
||
|
Status = dbp->set_bt_compare(dbp, info->bt_compare) ;
|
||
|
Trace(("set_bt_compare [%p] returned %s\n",
|
||
|
info->bt_compare, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->h_hash) {
|
||
|
Status = dbp->set_h_hash(dbp, info->h_hash) ;
|
||
|
Trace(("set_h_hash [%d] returned %s\n",
|
||
|
info->h_hash, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
|
||
|
if (info->dup_compare) {
|
||
|
Status = dbp->set_dup_compare(dbp, info->dup_compare) ;
|
||
|
Trace(("set_dup_compare [%d] returned %s\n",
|
||
|
info->dup_compare, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->bt_prefix) {
|
||
|
Status = dbp->set_bt_prefix(dbp, info->bt_prefix) ;
|
||
|
Trace(("set_bt_prefix [%d] returned %s\n",
|
||
|
info->bt_prefix, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->re_len) {
|
||
|
Status = dbp->set_re_len(dbp, info->re_len) ;
|
||
|
Trace(("set_re_len [%d] returned %s\n",
|
||
|
info->re_len, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->re_delim) {
|
||
|
Status = dbp->set_re_delim(dbp, info->re_delim) ;
|
||
|
Trace(("set_re_delim [%d] returned %s\n",
|
||
|
info->re_delim, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->re_pad) {
|
||
|
Status = dbp->set_re_pad(dbp, info->re_pad) ;
|
||
|
Trace(("set_re_pad [%d] returned %s\n",
|
||
|
info->re_pad, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->flags) {
|
||
|
Status = dbp->set_flags(dbp, info->flags) ;
|
||
|
Trace(("set_flags [%d] returned %s\n",
|
||
|
info->flags, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
if (info->q_extentsize) {
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
Status = dbp->set_q_extentsize(dbp, info->q_extentsize) ;
|
||
|
Trace(("set_q_extentsize [%d] returned %s\n",
|
||
|
info->q_extentsize, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
#else
|
||
|
softCrash("-ExtentSize needs at least Berkeley DB 3.2.x") ;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
if (info->heapsize_bytes || info->heapsize_gbytes) {
|
||
|
#ifdef AT_LEAST_DB_5_2
|
||
|
Status = dbp->set_heapsize(dbp, info->heapsize_gbytes,
|
||
|
info->heapsize_bytes,0) ;
|
||
|
Trace(("set_heapsize [%d,%d] returned %s\n",
|
||
|
info->heapsize_gbytes, info->heapsize_bytes, my_db_strerror(Status)));
|
||
|
if (Status)
|
||
|
return RETVAL ;
|
||
|
#else
|
||
|
softCrash("-HeapSize/HeapSizeGb needs at least Berkeley DB 5.2.x") ;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
/* In-memory database need DB_CREATE from 4.4 */
|
||
|
if (! file)
|
||
|
flags |= DB_CREATE;
|
||
|
|
||
|
Trace(("db_open'ing\n"));
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
if ((Status = (dbp->open)(dbp, txnid, file, subname, type, flags, mode)) == 0) {
|
||
|
#else
|
||
|
if ((Status = (dbp->open)(dbp, file, subname, type, flags, mode)) == 0) {
|
||
|
#endif /* AT_LEAST_DB_4_1 */
|
||
|
#else /* DB_VERSION_MAJOR == 2 */
|
||
|
if ((Status = db_open(file, type, flags, mode, env, info, &dbp)) == 0) {
|
||
|
CurrentDB = db ;
|
||
|
#endif /* DB_VERSION_MAJOR == 2 */
|
||
|
|
||
|
|
||
|
Trace(("db_opened ok\n"));
|
||
|
RETVAL = db ;
|
||
|
RETVAL->dbp = dbp ;
|
||
|
RETVAL->txn = txnid ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
RETVAL->type = dbp->type ;
|
||
|
#else /* DB_VERSION_MAJOR > 2 */
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
dbp->get_type(dbp, &RETVAL->type) ;
|
||
|
#else /* DB 3.0 -> 3.2 */
|
||
|
RETVAL->type = dbp->get_type(dbp) ;
|
||
|
#endif
|
||
|
#endif /* DB_VERSION_MAJOR > 2 */
|
||
|
RETVAL->primary_recno_or_queue = FALSE;
|
||
|
RETVAL->recno_or_queue = (RETVAL->type == DB_RECNO ||
|
||
|
RETVAL->type == DB_QUEUE) ;
|
||
|
RETVAL->filename = my_strdup(file) ;
|
||
|
RETVAL->Status = Status ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
hash_store_iv("BerkeleyDB::Term::Db", (char *)RETVAL, 1) ;
|
||
|
Trace((" storing %p %p in BerkeleyDB::Term::Db\n", RETVAL, dbp)) ;
|
||
|
if (dbenv) {
|
||
|
RETVAL->cds_enabled = dbenv->cds_enabled ;
|
||
|
RETVAL->parent_env = dbenv ;
|
||
|
dbenv->Status = Status ;
|
||
|
++ dbenv->open_dbs ;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
(dbp->close)(dbp, 0) ;
|
||
|
#endif
|
||
|
destroyDB(db) ;
|
||
|
Trace(("db open returned %s\n", my_db_strerror(Status))) ;
|
||
|
}
|
||
|
|
||
|
Trace(("End of _db_open\n"));
|
||
|
return RETVAL ;
|
||
|
}
|
||
|
|
||
|
|
||
|
#include "constants.h"
|
||
|
|
||
|
MODULE = BerkeleyDB PACKAGE = BerkeleyDB PREFIX = env_
|
||
|
|
||
|
INCLUDE: constants.xs
|
||
|
|
||
|
#define env_db_version(maj, min, patch) db_version(&maj, &min, &patch)
|
||
|
char *
|
||
|
env_db_version(maj, min, patch)
|
||
|
int maj
|
||
|
int min
|
||
|
int patch
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
maj
|
||
|
min
|
||
|
patch
|
||
|
|
||
|
int has_heap()
|
||
|
CODE:
|
||
|
#ifdef AT_LEAST_DB_5_2
|
||
|
RETVAL = __heap_exist() ;
|
||
|
#else
|
||
|
RETVAL = 0 ;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
|
||
|
int
|
||
|
db_value_set(value, which)
|
||
|
int value
|
||
|
int which
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
|
||
|
DualType
|
||
|
_db_remove(ref)
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
softCrash("BerkeleyDB::db_remove needs Berkeley DB 3.x or better") ;
|
||
|
#else
|
||
|
HV * hash ;
|
||
|
DB * dbp ;
|
||
|
SV * sv ;
|
||
|
const char * db = NULL ;
|
||
|
const char * subdb = NULL ;
|
||
|
BerkeleyDB__Env env = NULL ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
DB_ENV * dbenv = NULL ;
|
||
|
u_int32_t flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(db, "Filename", char *) ;
|
||
|
SetValue_pv(subdb, "Subname", char *) ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_ov(env, "Env", BerkeleyDB__Env) ;
|
||
|
if (txn) {
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
if (!env)
|
||
|
softCrash("transactional db_remove requires an environment");
|
||
|
RETVAL = env->Status = env->Env->dbremove(env->Env, txn->txn, db, subdb, flags);
|
||
|
#else
|
||
|
softCrash("transactional db_remove requires Berkeley DB 4.1 or better");
|
||
|
#endif
|
||
|
} else {
|
||
|
if (env)
|
||
|
dbenv = env->Env ;
|
||
|
RETVAL = db_create(&dbp, dbenv, 0) ;
|
||
|
if (RETVAL == 0) {
|
||
|
RETVAL = dbp->remove(dbp, db, subdb, flags) ;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
_db_verify(ref)
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("BerkeleyDB::db_verify needs Berkeley DB 3.1.x or better") ;
|
||
|
#else
|
||
|
HV * hash ;
|
||
|
DB * dbp ;
|
||
|
SV * sv ;
|
||
|
const char * db = NULL ;
|
||
|
const char * subdb = NULL ;
|
||
|
const char * outfile = NULL ;
|
||
|
FILE * ofh = NULL;
|
||
|
BerkeleyDB__Env env = NULL ;
|
||
|
DB_ENV * dbenv = NULL ;
|
||
|
u_int32_t flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(db, "Filename", char *) ;
|
||
|
SetValue_pv(subdb, "Subname", char *) ;
|
||
|
SetValue_pv(outfile, "Outfile", char *) ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_ov(env, "Env", BerkeleyDB__Env) ;
|
||
|
RETVAL = 0;
|
||
|
if (outfile){
|
||
|
ofh = fopen(outfile, "w");
|
||
|
if (! ofh)
|
||
|
RETVAL = errno;
|
||
|
}
|
||
|
if (! RETVAL) {
|
||
|
if (env)
|
||
|
dbenv = env->Env ;
|
||
|
RETVAL = db_create(&dbp, dbenv, 0) ;
|
||
|
if (RETVAL == 0) {
|
||
|
RETVAL = dbp->verify(dbp, db, subdb, ofh, flags) ;
|
||
|
}
|
||
|
if (outfile)
|
||
|
fclose(ofh);
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
_db_rename(ref)
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("BerkeleyDB::db_rename needs Berkeley DB 3.1.x or better") ;
|
||
|
#else
|
||
|
HV * hash ;
|
||
|
DB * dbp ;
|
||
|
SV * sv ;
|
||
|
const char * db = NULL ;
|
||
|
const char * subdb = NULL ;
|
||
|
const char * newname = NULL ;
|
||
|
BerkeleyDB__Env env = NULL ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
DB_ENV * dbenv = NULL ;
|
||
|
u_int32_t flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(db, "Filename", char *) ;
|
||
|
SetValue_pv(subdb, "Subname", char *) ;
|
||
|
SetValue_pv(newname, "Newname", char *) ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_ov(env, "Env", BerkeleyDB__Env) ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
if (txn) {
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
if (!env)
|
||
|
softCrash("transactional db_rename requires an environment");
|
||
|
RETVAL = env->Status = env->Env->dbrename(env->Env, txn->txn, db, subdb, newname, flags);
|
||
|
#else
|
||
|
softCrash("transactional db_rename requires Berkeley DB 4.1 or better");
|
||
|
#endif
|
||
|
} else {
|
||
|
if (env)
|
||
|
dbenv = env->Env ;
|
||
|
RETVAL = db_create(&dbp, dbenv, 0) ;
|
||
|
if (RETVAL == 0) {
|
||
|
RETVAL = (dbp->rename)(dbp, db, subdb, newname, flags) ;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
MODULE = BerkeleyDB::Env PACKAGE = BerkeleyDB::Env PREFIX = env_
|
||
|
|
||
|
BerkeleyDB::Env::Raw
|
||
|
create(flags=0)
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_1
|
||
|
softCrash("$env->create needs Berkeley DB 4.1 or better") ;
|
||
|
#else
|
||
|
DB_ENV * env ;
|
||
|
int status;
|
||
|
RETVAL = NULL;
|
||
|
Trace(("in BerkeleyDB::Env::create flags=%d\n", flags)) ;
|
||
|
status = db_env_create(&env, flags) ;
|
||
|
Trace(("db_env_create returned %s\n", my_db_strerror(status))) ;
|
||
|
if (status == 0) {
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_ENV_type) ;
|
||
|
RETVAL->Env = env ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
RETVAL->opened = FALSE;
|
||
|
env->set_alloc(env, safemalloc, MyRealloc, safefree) ;
|
||
|
env->set_errcall(env, db_errcall_cb) ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
open(env, db_home=NULL, flags=0, mode=0777)
|
||
|
BerkeleyDB::Env env
|
||
|
char * db_home
|
||
|
u_int32_t flags
|
||
|
int mode
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_1
|
||
|
softCrash("$env->create needs Berkeley DB 4.1 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Env->open(env->Env, db_home, flags, mode);
|
||
|
env->opened = TRUE;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
bool
|
||
|
cds_enabled(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = env->cds_enabled ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
int
|
||
|
set_encrypt(env, passwd, flags)
|
||
|
BerkeleyDB::Env env
|
||
|
const char * passwd
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_1
|
||
|
softCrash("$env->set_encrypt needs Berkeley DB 4.1 or better") ;
|
||
|
#else
|
||
|
dieIfEnvOpened(env, "set_encrypt");
|
||
|
RETVAL = env->Env->set_encrypt(env->Env, passwd, flags);
|
||
|
env->opened = TRUE;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
BerkeleyDB::Env::Raw
|
||
|
_db_appinit(self, ref, errfile=NULL)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
SV * errfile
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
char * home = NULL ;
|
||
|
char * server = NULL ;
|
||
|
char ** config = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int setflags = 0 ;
|
||
|
int cachesize = 0 ;
|
||
|
int lk_detect = 0 ;
|
||
|
int tx_max = 0 ;
|
||
|
int log_config = 0 ;
|
||
|
int max_lockers = 0 ;
|
||
|
int max_locks = 0 ;
|
||
|
int max_objects = 0 ;
|
||
|
long shm_key = 0 ;
|
||
|
char* data_dir = 0;
|
||
|
char* log_dir = 0;
|
||
|
char* temp_dir = 0;
|
||
|
SV * msgfile = NULL ;
|
||
|
int thread_count = 0 ;
|
||
|
SV * errprefix = NULL;
|
||
|
DB_ENV * env ;
|
||
|
int status ;
|
||
|
|
||
|
Trace(("in _db_appinit [%s] %d\n", self, ref)) ;
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(home, "Home", char *) ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
SetValue_pv(config, "Config", char **) ;
|
||
|
SetValue_sv(errprefix, "ErrPrefix") ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(setflags, "SetFlags") ;
|
||
|
SetValue_pv(server, "Server", char *) ;
|
||
|
SetValue_iv(cachesize, "Cachesize") ;
|
||
|
SetValue_iv(lk_detect, "LockDetect") ;
|
||
|
SetValue_iv(tx_max, "TxMax") ;
|
||
|
SetValue_iv(log_config,"LogConfig") ;
|
||
|
SetValue_iv(max_lockers,"MaxLockers") ;
|
||
|
SetValue_iv(max_locks, "MaxLocks") ;
|
||
|
SetValue_iv(max_objects,"MaxObjects") ;
|
||
|
SetValue_iv(shm_key, "SharedMemKey") ;
|
||
|
SetValue_iv(thread_count, "ThreadCount") ;
|
||
|
SetValue_pv(data_dir, "DB_DATA_DIR", char*) ;
|
||
|
SetValue_pv(temp_dir, "DB_TEMP_DIR", char*) ;
|
||
|
SetValue_pv(log_dir, "DB_LOG_DIR", char*) ;
|
||
|
SetValue_sv(msgfile, "MsgFile") ;
|
||
|
#ifndef AT_LEAST_DB_3_2
|
||
|
if (setflags)
|
||
|
softCrash("-SetFlags needs Berkeley DB 3.x or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_3 */
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
if (shm_key)
|
||
|
softCrash("-SharedMemKey needs Berkeley DB 3.1 or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_3_1 */
|
||
|
#if ! defined(AT_LEAST_DB_3_1) || defined(AT_LEAST_DB_5_1)
|
||
|
if (server)
|
||
|
softCrash("-Server only supported Berkeley DB 3.1 to 5.1") ;
|
||
|
#endif /* ! AT_LEAST_DB_3_1 */
|
||
|
#ifndef AT_LEAST_DB_3_2
|
||
|
if (max_lockers)
|
||
|
softCrash("-MaxLockers needs Berkeley DB 3.2 or better") ;
|
||
|
if (max_locks)
|
||
|
softCrash("-MaxLocks needs Berkeley DB 3.2 or better") ;
|
||
|
if (max_objects)
|
||
|
softCrash("-MaxObjects needs Berkeley DB 3.2 or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_3_2 */
|
||
|
#ifndef AT_LEAST_DB_4_1
|
||
|
if (enc_passwd)
|
||
|
softCrash("-Encrypt needs Berkeley DB 4.x or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_4_1 */
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
if (msgfile)
|
||
|
softCrash("-MsgFile needs Berkeley DB 4.3.x or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_4_3 */
|
||
|
#ifdef _WIN32
|
||
|
if (thread_count)
|
||
|
softCrash("-ThreadCount not supported on Windows") ;
|
||
|
#endif /* ! _WIN32 */
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
if (thread_count)
|
||
|
softCrash("-ThreadCount needs Berkeley DB 4.4 or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_4_4 */
|
||
|
#ifndef AT_LEAST_DB_4_7
|
||
|
if (log_config)
|
||
|
softCrash("-LogConfig needs Berkeley DB 4.7 or better") ;
|
||
|
#endif /* ! AT_LEAST_DB_4_7 */
|
||
|
Trace(("_db_appinit(config=[%d], home=[%s],errprefix=[%s],flags=[%d]\n",
|
||
|
config, home, errprefix, flags)) ;
|
||
|
#ifdef TRACE
|
||
|
if (config) {
|
||
|
int i ;
|
||
|
for (i = 0 ; i < 10 ; ++ i) {
|
||
|
if (config[i] == NULL) {
|
||
|
printf(" End\n") ;
|
||
|
break ;
|
||
|
}
|
||
|
printf(" config = [%s]\n", config[i]) ;
|
||
|
}
|
||
|
}
|
||
|
#endif /* TRACE */
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_ENV_type) ;
|
||
|
if (flags & DB_INIT_TXN)
|
||
|
RETVAL->txn_enabled = TRUE ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
ZMALLOC(RETVAL->Env, DB_ENV) ;
|
||
|
env = RETVAL->Env ;
|
||
|
{
|
||
|
/* Take a copy of the error prefix */
|
||
|
if (errprefix) {
|
||
|
Trace(("copying errprefix\n" )) ;
|
||
|
RETVAL->ErrPrefix = newSVsv(errprefix) ;
|
||
|
SvPOK_only(RETVAL->ErrPrefix) ;
|
||
|
}
|
||
|
if (RETVAL->ErrPrefix)
|
||
|
RETVAL->Env->db_errpfx = SvPVX(RETVAL->ErrPrefix) ;
|
||
|
|
||
|
if (SvGMAGICAL(errfile))
|
||
|
mg_get(errfile);
|
||
|
if (SvOK(errfile)) {
|
||
|
FILE * ef = GetFILEptr(errfile) ;
|
||
|
if (! ef)
|
||
|
croak("Cannot open file ErrFile", Strerror(errno));
|
||
|
RETVAL->ErrHandle = newSVsv(errfile) ;
|
||
|
env->db_errfile = ef;
|
||
|
}
|
||
|
SetValue_iv(env->db_verbose, "Verbose") ;
|
||
|
env->db_errcall = db_errcall_cb ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
RETVAL->opened = TRUE;
|
||
|
RETVAL->cds_enabled = ((flags & DB_INIT_CDB) != 0 ? TRUE : FALSE) ;
|
||
|
status = db_appinit(home, config, env, flags) ;
|
||
|
printf(" status = %d errno %d \n", status, errno) ;
|
||
|
Trace((" status = %d env %d Env %d\n", status, RETVAL, env)) ;
|
||
|
if (status == 0)
|
||
|
hash_store_iv("BerkeleyDB::Term::Env", (char *)RETVAL, 1) ;
|
||
|
else {
|
||
|
|
||
|
if (RETVAL->ErrHandle)
|
||
|
SvREFCNT_dec(RETVAL->ErrHandle) ;
|
||
|
if (RETVAL->ErrPrefix)
|
||
|
SvREFCNT_dec(RETVAL->ErrPrefix) ;
|
||
|
Safefree(RETVAL->Env) ;
|
||
|
Safefree(RETVAL) ;
|
||
|
RETVAL = NULL ;
|
||
|
}
|
||
|
}
|
||
|
#else /* DB_VERSION_MAJOR > 2 */
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
# define DB_CLIENT 0
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_5_1
|
||
|
# define DB_CLIENT 0
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4_2
|
||
|
# define DB_CLIENT DB_RPCCLIENT
|
||
|
# endif
|
||
|
#endif
|
||
|
status = db_env_create(&RETVAL->Env, server ? DB_CLIENT : 0) ;
|
||
|
Trace(("db_env_create flags = %d returned %s\n", flags,
|
||
|
my_db_strerror(status))) ;
|
||
|
env = RETVAL->Env ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
env->set_alloc(env, safemalloc, MyRealloc, safefree) ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_3_1
|
||
|
if (status == 0 && shm_key) {
|
||
|
status = env->set_shm_key(env, shm_key) ;
|
||
|
Trace(("set_shm_key [%d] returned %s\n", shm_key,
|
||
|
my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && data_dir) {
|
||
|
status = env->set_data_dir(env, data_dir) ;
|
||
|
Trace(("set_data_dir [%s] returned %s\n", data_dir,
|
||
|
my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && temp_dir) {
|
||
|
status = env->set_tmp_dir(env, temp_dir) ;
|
||
|
Trace(("set_tmp_dir [%s] returned %s\n", temp_dir,
|
||
|
my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && log_dir) {
|
||
|
status = env->set_lg_dir(env, log_dir) ;
|
||
|
Trace(("set_lg_dir [%s] returned %s\n", log_dir,
|
||
|
my_db_strerror(status)));
|
||
|
}
|
||
|
#endif
|
||
|
if (status == 0 && cachesize) {
|
||
|
status = env->set_cachesize(env, 0, cachesize, 0) ;
|
||
|
Trace(("set_cachesize [%d] returned %s\n",
|
||
|
cachesize, my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && lk_detect) {
|
||
|
status = env->set_lk_detect(env, lk_detect) ;
|
||
|
Trace(("set_lk_detect [%d] returned %s\n",
|
||
|
lk_detect, my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && tx_max) {
|
||
|
status = env->set_tx_max(env, tx_max) ;
|
||
|
Trace(("set_tx_max [%d] returned %s\n",
|
||
|
tx_max, my_db_strerror(status)));
|
||
|
}
|
||
|
#ifdef AT_LEAST_DB_4_7
|
||
|
if (status == 0 && log_config) {
|
||
|
status = env->log_set_config(env, log_config, 1) ;
|
||
|
Trace(("log_set_config [%d] returned %s\n",
|
||
|
log_config, my_db_strerror(status)));
|
||
|
}
|
||
|
#endif /* AT_LEAST_DB_4_7 */
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
if (status == 0 && max_lockers) {
|
||
|
status = env->set_lk_max_lockers(env, max_lockers) ;
|
||
|
Trace(("set_lk_max_lockers [%d] returned %s\n",
|
||
|
max_lockers, my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && max_locks) {
|
||
|
status = env->set_lk_max_locks(env, max_locks) ;
|
||
|
Trace(("set_lk_max_locks [%d] returned %s\n",
|
||
|
max_locks, my_db_strerror(status)));
|
||
|
}
|
||
|
|
||
|
if (status == 0 && max_objects) {
|
||
|
status = env->set_lk_max_objects(env, max_objects) ;
|
||
|
Trace(("set_lk_max_objects [%d] returned %s\n",
|
||
|
max_objects, my_db_strerror(status)));
|
||
|
}
|
||
|
#endif /* AT_LEAST_DB_3_2 */
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
/* set encryption */
|
||
|
if (enc_passwd && status == 0)
|
||
|
{
|
||
|
status = env->set_encrypt(env, enc_passwd, enc_flags);
|
||
|
Trace(("ENV->set_encrypt passwd = %s, flags %d returned %s\n",
|
||
|
enc_passwd, enc_flags,
|
||
|
my_db_strerror(status))) ;
|
||
|
}
|
||
|
#endif
|
||
|
#if ! defined(AT_LEAST_DB_5_1)
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
/* set the server */
|
||
|
if (server && status == 0)
|
||
|
{
|
||
|
status = env->set_rpc_server(env, NULL, server, 0, 0, 0);
|
||
|
Trace(("ENV->set_rpc_server server = %s returned %s\n", server,
|
||
|
my_db_strerror(status))) ;
|
||
|
}
|
||
|
#else
|
||
|
# if defined(AT_LEAST_DB_3_1) && ! defined(AT_LEAST_DB_4)
|
||
|
/* set the server */
|
||
|
if (server && status == 0)
|
||
|
{
|
||
|
status = env->set_server(env, server, 0, 0, 0);
|
||
|
Trace(("ENV->set_server server = %s returned %s\n", server,
|
||
|
my_db_strerror(status))) ;
|
||
|
}
|
||
|
# endif
|
||
|
#endif
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
if (setflags && status == 0)
|
||
|
{
|
||
|
status = env->set_flags(env, setflags, 1);
|
||
|
Trace(("ENV->set_flags value = %d returned %s\n", setflags,
|
||
|
my_db_strerror(status))) ;
|
||
|
}
|
||
|
#endif
|
||
|
#if defined(AT_LEAST_DB_4_4) && ! defined(_WIN32)
|
||
|
if (thread_count && status == 0)
|
||
|
{
|
||
|
status = env->set_thread_count(env, thread_count);
|
||
|
Trace(("ENV->set_thread_count value = %d returned %s\n", thread_count,
|
||
|
my_db_strerror(status))) ;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (status == 0)
|
||
|
{
|
||
|
int mode = 0 ;
|
||
|
/* Take a copy of the error prefix */
|
||
|
if (errprefix) {
|
||
|
Trace(("copying errprefix\n" )) ;
|
||
|
RETVAL->ErrPrefix = newSVsv(errprefix) ;
|
||
|
SvPOK_only(RETVAL->ErrPrefix) ;
|
||
|
}
|
||
|
if (RETVAL->ErrPrefix)
|
||
|
env->set_errpfx(env, SvPVX(RETVAL->ErrPrefix)) ;
|
||
|
|
||
|
if (SvGMAGICAL(errfile))
|
||
|
mg_get(errfile);
|
||
|
if (SvOK(errfile)) {
|
||
|
FILE * ef = GetFILEptr(errfile);
|
||
|
if (! ef)
|
||
|
croak("Cannot open file ErrFile", Strerror(errno));
|
||
|
RETVAL->ErrHandle = newSVsv(errfile) ;
|
||
|
env->set_errfile(env, ef) ;
|
||
|
|
||
|
}
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
if (msgfile) {
|
||
|
if (SvGMAGICAL(msgfile))
|
||
|
mg_get(msgfile);
|
||
|
if (SvOK(msgfile)) {
|
||
|
FILE * ef = GetFILEptr(msgfile);
|
||
|
if (! ef)
|
||
|
croak("Cannot open file MsgFile", Strerror(errno));
|
||
|
RETVAL->MsgHandle = newSVsv(msgfile) ;
|
||
|
env->set_msgfile(env, ef) ;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
env->set_errcall(env, db_errcall_cb) ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
RETVAL->cds_enabled = ((flags & DB_INIT_CDB) != 0 ? TRUE : FALSE) ;
|
||
|
#ifdef IS_DB_3_0_x
|
||
|
status = (env->open)(env, home, config, flags, mode) ;
|
||
|
#else /* > 3.0 */
|
||
|
status = (env->open)(env, home, flags, mode) ;
|
||
|
#endif
|
||
|
Trace(("ENV->open(env=%s,home=%s,flags=%d,mode=%d)\n",env,home,flags,mode)) ;
|
||
|
Trace(("ENV->open returned %s\n", my_db_strerror(status))) ;
|
||
|
}
|
||
|
|
||
|
if (status == 0)
|
||
|
hash_store_iv("BerkeleyDB::Term::Env", (char *)RETVAL, 1) ;
|
||
|
else {
|
||
|
(env->close)(env, 0) ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
if (RETVAL->MsgHandle)
|
||
|
SvREFCNT_dec(RETVAL->MsgHandle) ;
|
||
|
#endif
|
||
|
if (RETVAL->ErrHandle)
|
||
|
SvREFCNT_dec(RETVAL->ErrHandle) ;
|
||
|
if (RETVAL->ErrPrefix)
|
||
|
SvREFCNT_dec(RETVAL->ErrPrefix) ;
|
||
|
Safefree(RETVAL) ;
|
||
|
RETVAL = NULL ;
|
||
|
}
|
||
|
#endif /* DB_VERSION_MAJOR > 2 */
|
||
|
{
|
||
|
SV * sv_err = perl_get_sv(ERR_BUFF, FALSE);
|
||
|
sv_setpv(sv_err, db_strerror(status));
|
||
|
}
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DB_ENV*
|
||
|
DB_ENV(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
if (env->active)
|
||
|
RETVAL = env->Env ;
|
||
|
else
|
||
|
RETVAL = NULL;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
void
|
||
|
log_archive(env, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
PPCODE:
|
||
|
{
|
||
|
char ** list;
|
||
|
char ** file;
|
||
|
AV * av;
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("log_archive needs at least Berkeley DB 3.x.x");
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
env->Status = env->Env->log_archive(env->Env, &list, flags) ;
|
||
|
# else
|
||
|
# ifdef AT_LEAST_DB_3_3
|
||
|
env->Status = log_archive(env->Env, &list, flags) ;
|
||
|
# else
|
||
|
env->Status = log_archive(env->Env, &list, flags, safemalloc) ;
|
||
|
# endif
|
||
|
# endif
|
||
|
#ifdef DB_ARCH_REMOVE
|
||
|
if (env->Status == 0 && list != NULL && flags != DB_ARCH_REMOVE)
|
||
|
#else
|
||
|
if (env->Status == 0 && list != NULL )
|
||
|
#endif
|
||
|
{
|
||
|
for (file = list; *file != NULL; ++file)
|
||
|
{
|
||
|
XPUSHs(sv_2mortal(newSVpv(*file, 0))) ;
|
||
|
}
|
||
|
safefree(list);
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
DualType
|
||
|
log_set_config(env, flags=0, onoff=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
int onoff
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_7
|
||
|
softCrash("log_set_config needs at least Berkeley DB 4.7.x");
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->log_set_config(env->Env, flags, onoff) ;
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
log_get_config(env, flags, onoff)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
int onoff=NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_7
|
||
|
softCrash("log_get_config needs at least Berkeley DB 4.7.x");
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->log_get_config(env->Env, flags, &onoff) ;
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
onoff
|
||
|
|
||
|
|
||
|
BerkeleyDB::Txn::Raw
|
||
|
_txn_begin(env, pid=NULL, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Env env
|
||
|
BerkeleyDB::Txn pid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DB_TXN *txn ;
|
||
|
DB_TXN *p_id = NULL ;
|
||
|
Trace(("txn_begin pid %d, flags %d\n", pid, flags)) ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
if (env->Env->tx_info == NULL)
|
||
|
softCrash("Transaction Manager not enabled") ;
|
||
|
#endif
|
||
|
if (!env->txn_enabled)
|
||
|
softCrash("Transaction Manager not enabled") ;
|
||
|
if (pid)
|
||
|
p_id = pid->txn ;
|
||
|
env->TxnMgrStatus =
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
txn_begin(env->Env->tx_info, p_id, &txn) ;
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
env->Env->txn_begin(env->Env, p_id, &txn, flags) ;
|
||
|
# else
|
||
|
txn_begin(env->Env, p_id, &txn, flags) ;
|
||
|
# endif
|
||
|
#endif
|
||
|
if (env->TxnMgrStatus == 0) {
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_Txn_type) ;
|
||
|
RETVAL->txn = txn ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
Trace(("_txn_begin created txn [%p] in [%p]\n", txn, RETVAL));
|
||
|
hash_store_iv("BerkeleyDB::Term::Txn", (char *)RETVAL, 1) ;
|
||
|
}
|
||
|
else
|
||
|
RETVAL = NULL ;
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
# define env_txn_checkpoint(e,k,m,f) txn_checkpoint(e->Env->tx_info, k, m)
|
||
|
#else /* DB 3.0 or better */
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
# define env_txn_checkpoint(e,k,m,f) e->Env->txn_checkpoint(e->Env, k, m, f)
|
||
|
# else
|
||
|
# ifdef AT_LEAST_DB_3_1
|
||
|
# define env_txn_checkpoint(e,k,m,f) txn_checkpoint(e->Env, k, m, 0)
|
||
|
# else
|
||
|
# define env_txn_checkpoint(e,k,m,f) txn_checkpoint(e->Env, k, m)
|
||
|
# endif
|
||
|
# endif
|
||
|
#endif
|
||
|
DualType
|
||
|
env_txn_checkpoint(env, kbyte, min, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
long kbyte
|
||
|
long min
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
|
||
|
HV *
|
||
|
txn_stat(env)
|
||
|
BerkeleyDB::Env env
|
||
|
HV * RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DB_TXN_STAT * stat ;
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
if(env->Env->txn_stat(env->Env, &stat, 0) == 0) {
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_3_3
|
||
|
if(txn_stat(env->Env, &stat) == 0) {
|
||
|
# else
|
||
|
# if DB_VERSION_MAJOR == 2
|
||
|
if(txn_stat(env->Env->tx_info, &stat, safemalloc) == 0) {
|
||
|
# else
|
||
|
if(txn_stat(env->Env, &stat, safemalloc) == 0) {
|
||
|
# endif
|
||
|
# endif
|
||
|
#endif
|
||
|
RETVAL = (HV*)sv_2mortal((SV*)newHV()) ;
|
||
|
hv_store_iv(RETVAL, "st_time_ckp", stat->st_time_ckp) ;
|
||
|
hv_store_iv(RETVAL, "st_last_txnid", stat->st_last_txnid) ;
|
||
|
hv_store_iv(RETVAL, "st_maxtxns", stat->st_maxtxns) ;
|
||
|
hv_store_iv(RETVAL, "st_naborts", stat->st_naborts) ;
|
||
|
hv_store_iv(RETVAL, "st_nbegins", stat->st_nbegins) ;
|
||
|
hv_store_iv(RETVAL, "st_ncommits", stat->st_ncommits) ;
|
||
|
hv_store_iv(RETVAL, "st_nactive", stat->st_nactive) ;
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
hv_store_iv(RETVAL, "st_maxnactive", stat->st_maxnactive) ;
|
||
|
hv_store_iv(RETVAL, "st_regsize", stat->st_regsize) ;
|
||
|
hv_store_iv(RETVAL, "st_region_wait", stat->st_region_wait) ;
|
||
|
hv_store_iv(RETVAL, "st_region_nowait", stat->st_region_nowait) ;
|
||
|
#endif
|
||
|
safefree(stat) ;
|
||
|
}
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
#define EnDis(x) ((x) ? "Enabled" : "Disabled")
|
||
|
void
|
||
|
printEnv(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Environment(env->active) ;
|
||
|
CODE:
|
||
|
#if 0
|
||
|
printf("env [0x%X]\n", env) ;
|
||
|
printf(" ErrPrefix [%s]\n", env->ErrPrefix
|
||
|
? SvPVX(env->ErrPrefix) : 0) ;
|
||
|
printf(" DB_ENV\n") ;
|
||
|
printf(" db_lorder [%d]\n", env->Env.db_lorder) ;
|
||
|
printf(" db_home [%s]\n", env->Env.db_home) ;
|
||
|
printf(" db_data_dir [%s]\n", env->Env.db_data_dir) ;
|
||
|
printf(" db_log_dir [%s]\n", env->Env.db_log_dir) ;
|
||
|
printf(" db_tmp_dir [%s]\n", env->Env.db_tmp_dir) ;
|
||
|
printf(" lk_info [%s]\n", EnDis(env->Env.lk_info)) ;
|
||
|
printf(" lk_max [%d]\n", env->Env.lk_max) ;
|
||
|
printf(" lg_info [%s]\n", EnDis(env->Env.lg_info)) ;
|
||
|
printf(" lg_max [%d]\n", env->Env.lg_max) ;
|
||
|
printf(" mp_info [%s]\n", EnDis(env->Env.mp_info)) ;
|
||
|
printf(" mp_size [%d]\n", env->Env.mp_size) ;
|
||
|
printf(" tx_info [%s]\n", EnDis(env->Env.tx_info)) ;
|
||
|
printf(" tx_max [%d]\n", env->Env.tx_max) ;
|
||
|
printf(" flags [%d]\n", env->Env.flags) ;
|
||
|
printf("\n") ;
|
||
|
#endif
|
||
|
|
||
|
SV *
|
||
|
errPrefix(env, prefix)
|
||
|
BerkeleyDB::Env env
|
||
|
SV * prefix
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Environment(env->active) ;
|
||
|
CODE:
|
||
|
if (env->ErrPrefix) {
|
||
|
RETVAL = newSVsv(env->ErrPrefix) ;
|
||
|
SvPOK_only(RETVAL) ;
|
||
|
sv_setsv(env->ErrPrefix, prefix) ;
|
||
|
}
|
||
|
else {
|
||
|
RETVAL = NULL ;
|
||
|
env->ErrPrefix = newSVsv(prefix) ;
|
||
|
}
|
||
|
SvPOK_only(env->ErrPrefix) ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
env->Env->db_errpfx = SvPVX(env->ErrPrefix) ;
|
||
|
#else
|
||
|
env->Env->set_errpfx(env->Env, SvPVX(env->ErrPrefix)) ;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
status(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = env->Status ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
|
||
|
DualType
|
||
|
db_appexit(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
ALIAS: close =1
|
||
|
INIT:
|
||
|
ckActive_Environment(env->active) ;
|
||
|
CODE:
|
||
|
#ifdef STRICT_CLOSE
|
||
|
if (env->open_dbs)
|
||
|
softCrash("attempted to close an environment with %d open database(s)",
|
||
|
env->open_dbs) ;
|
||
|
#endif /* STRICT_CLOSE */
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
RETVAL = db_appexit(env->Env) ;
|
||
|
#else
|
||
|
RETVAL = (env->Env->close)(env->Env, 0) ;
|
||
|
#endif
|
||
|
env->active = FALSE ;
|
||
|
hash_delete("BerkeleyDB::Term::Env", (char *)env) ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
void
|
||
|
_DESTROY(env)
|
||
|
BerkeleyDB::Env env
|
||
|
int RETVAL = 0 ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
Trace(("In BerkeleyDB::Env::DESTROY\n"));
|
||
|
Trace((" env %ld Env %ld dirty %d\n", env, &env->Env, PL_dirty)) ;
|
||
|
if (env->active)
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
db_appexit(env->Env) ;
|
||
|
#else
|
||
|
(env->Env->close)(env->Env, 0) ;
|
||
|
#endif
|
||
|
if (env->ErrHandle)
|
||
|
SvREFCNT_dec(env->ErrHandle) ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
if (env->MsgHandle)
|
||
|
SvREFCNT_dec(env->MsgHandle) ;
|
||
|
#endif
|
||
|
if (env->ErrPrefix)
|
||
|
SvREFCNT_dec(env->ErrPrefix) ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
Safefree(env->Env) ;
|
||
|
#endif
|
||
|
Safefree(env) ;
|
||
|
hash_delete("BerkeleyDB::Term::Env", (char *)env) ;
|
||
|
Trace(("End of BerkeleyDB::Env::DESTROY %d\n", RETVAL)) ;
|
||
|
|
||
|
BerkeleyDB::TxnMgr::Raw
|
||
|
_TxnMgr(env)
|
||
|
BerkeleyDB::Env env
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Environment(env->active) ;
|
||
|
if (!env->txn_enabled)
|
||
|
softCrash("Transaction Manager not enabled") ;
|
||
|
CODE:
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_TxnMgr_type) ;
|
||
|
RETVAL->env = env ;
|
||
|
/* hash_store_iv("BerkeleyDB::Term::TxnMgr", (char *)txn, 1) ; */
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
get_shm_key(env, id)
|
||
|
BerkeleyDB::Env env
|
||
|
long id = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_2
|
||
|
softCrash("$env->get_shm_key needs Berkeley DB 4.2 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Env->get_shm_key(env->Env, &id);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
id
|
||
|
|
||
|
|
||
|
int
|
||
|
set_lg_dir(env, dir)
|
||
|
BerkeleyDB::Env env
|
||
|
char * dir
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("$env->set_lg_dir needs Berkeley DB 3.1 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_lg_dir(env->Env, dir);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_lg_bsize(env, bsize)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t bsize
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("$env->set_lg_bsize needs Berkeley DB 3.0.55 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_lg_bsize(env->Env, bsize);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_lg_max(env, lg_max)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t lg_max
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("$env->set_lg_max needs Berkeley DB 3.0.55 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_lg_max(env->Env, lg_max);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_data_dir(env, dir)
|
||
|
BerkeleyDB::Env env
|
||
|
char * dir
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("$env->set_data_dir needs Berkeley DB 3.1 or better") ;
|
||
|
#else
|
||
|
dieIfEnvOpened(env, "set_data_dir");
|
||
|
RETVAL = env->Status = env->Env->set_data_dir(env->Env, dir);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_tmp_dir(env, dir)
|
||
|
BerkeleyDB::Env env
|
||
|
char * dir
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("$env->set_tmp_dir needs Berkeley DB 3.1 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_tmp_dir(env->Env, dir);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_mutexlocks(env, do_lock)
|
||
|
BerkeleyDB::Env env
|
||
|
int do_lock
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("$env->set_setmutexlocks needs Berkeley DB 3.0 or better") ;
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
RETVAL = env->Status = env->Env->set_flags(env->Env, DB_NOLOCKING, !do_lock);
|
||
|
# else
|
||
|
# if defined(AT_LEAST_DB_3_2_6) || defined(IS_DB_3_0_x)
|
||
|
RETVAL = env->Status = env->Env->set_mutexlocks(env->Env, do_lock);
|
||
|
# else /* DB 3.1 or 3.2.3 */
|
||
|
RETVAL = env->Status = db_env_set_mutexlocks(do_lock);
|
||
|
# endif
|
||
|
# endif
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_verbose(env, which, onoff)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t which
|
||
|
int onoff
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("$env->set_verbose needs Berkeley DB 3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_verbose(env->Env, which, onoff);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_flags(env, flags, onoff)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
int onoff
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_2
|
||
|
softCrash("$env->set_flags needs Berkeley DB 3.2.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_flags(env->Env, flags, onoff);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
lsn_reset(env, file, flags)
|
||
|
BerkeleyDB::Env env
|
||
|
char* file
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$env->lsn_reset needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->lsn_reset(env->Env, file, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
lock_detect(env, atype=DB_LOCK_DEFAULT, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t atype
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_2_2
|
||
|
softCrash("$env->lock_detect needs Berkeley DB 2.2.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->lock_detect(env->Env,flags,atype,NULL);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
int
|
||
|
set_timeout(env, timeout, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
db_timeout_t timeout
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4
|
||
|
softCrash("$env->set_timeout needs Berkeley DB 4.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_timeout(env->Env, timeout, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
get_timeout(env, timeout, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
db_timeout_t timeout = NO_INIT
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_2
|
||
|
softCrash("$env->set_timeout needs Berkeley DB 4.2.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->get_timeout(env->Env, &timeout, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
timeout
|
||
|
|
||
|
int
|
||
|
stat_print(env, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$env->stat_print needs Berkeley DB 4.3 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->stat_print(env->Env, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
lock_stat_print(env, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$env->lock_stat_print needs Berkeley DB 4.3 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->lock_stat_print(env->Env, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
mutex_stat_print(env, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
softCrash("$env->mutex_stat_print needs Berkeley DB 4.4 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->mutex_stat_print(env->Env, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
int
|
||
|
txn_stat_print(env, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$env->mutex_stat_print needs Berkeley DB 4.3 or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->txn_stat_print(env->Env, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
failchk(env, flags=0)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#if ! defined(AT_LEAST_DB_4_4) || defined(_WIN32)
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
softCrash("$env->failchk needs Berkeley DB 4.4 or better") ;
|
||
|
#endif
|
||
|
#ifdef _WIN32
|
||
|
softCrash("$env->failchk not supported on Windows") ;
|
||
|
#endif
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->failchk(env->Env, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_isalive(env)
|
||
|
BerkeleyDB::Env env
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#if ! defined(AT_LEAST_DB_4_4) || defined(_WIN32)
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
softCrash("$env->set_isalive needs Berkeley DB 4.4 or better") ;
|
||
|
#endif
|
||
|
#ifdef _WIN32
|
||
|
softCrash("$env->set_isalive not supported on Windows") ;
|
||
|
#endif
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->set_isalive(env->Env, db_isalive_cb);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Term PACKAGE = BerkeleyDB::Term
|
||
|
|
||
|
void
|
||
|
close_everything()
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
|
||
|
#define safeCroak(string) softCrash(string)
|
||
|
void
|
||
|
safeCroak(string)
|
||
|
char * string
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
|
||
|
MODULE = BerkeleyDB::Hash PACKAGE = BerkeleyDB::Hash PREFIX = hash_
|
||
|
|
||
|
BerkeleyDB::Hash::Raw
|
||
|
_db_open_hash(self, ref)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
Trace(("_db_open_hash start\n")) ;
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Filename", char *) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.h_ffactor, "Ffactor") ;
|
||
|
SetValue_iv(info.h_nelem, "Nelem") ;
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
if ((sv = readHash(hash, "Hash")) && sv != &PL_sv_undef) {
|
||
|
info.h_hash = hash_cb ;
|
||
|
db->hash = newSVsv(sv) ;
|
||
|
}
|
||
|
/* DB_DUPSORT was introduced in DB 2.5.9 */
|
||
|
if ((sv = readHash(hash, "DupCompare")) && sv != &PL_sv_undef) {
|
||
|
#ifdef DB_DUPSORT
|
||
|
info.dup_compare = dup_compare ;
|
||
|
db->dup_compare = newSVsv(sv) ;
|
||
|
info.flags |= DB_DUP|DB_DUPSORT ;
|
||
|
#else
|
||
|
croak("DupCompare needs Berkeley DB 2.5.9 or later") ;
|
||
|
#endif
|
||
|
}
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_HASH, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
Trace(("_db_open_hash end\n")) ;
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
HV *
|
||
|
db_stat(db, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Common db
|
||
|
HV * RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
softCrash("$db->db_stat for a Hash needs Berkeley DB 3.x or better") ;
|
||
|
#else
|
||
|
DB_HASH_STAT * stat ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, db->txn, &stat, flags) ;
|
||
|
#else
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, flags) ;
|
||
|
#else
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, safemalloc, flags) ;
|
||
|
#endif
|
||
|
#endif
|
||
|
if (db->Status) {
|
||
|
XSRETURN_UNDEF;
|
||
|
} else {
|
||
|
RETVAL = (HV*)sv_2mortal((SV*)newHV()) ;
|
||
|
hv_store_iv(RETVAL, "hash_magic", stat->hash_magic) ;
|
||
|
hv_store_iv(RETVAL, "hash_version", stat->hash_version);
|
||
|
hv_store_iv(RETVAL, "hash_pagesize", stat->hash_pagesize);
|
||
|
#ifdef AT_LEAST_DB_3_1
|
||
|
hv_store_iv(RETVAL, "hash_nkeys", stat->hash_nkeys);
|
||
|
hv_store_iv(RETVAL, "hash_ndata", stat->hash_ndata);
|
||
|
#else
|
||
|
hv_store_iv(RETVAL, "hash_nrecs", stat->hash_nrecs);
|
||
|
#endif
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
hv_store_iv(RETVAL, "hash_nelem", stat->hash_nelem);
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "hash_ffactor", stat->hash_ffactor);
|
||
|
hv_store_iv(RETVAL, "hash_buckets", stat->hash_buckets);
|
||
|
hv_store_iv(RETVAL, "hash_free", stat->hash_free);
|
||
|
hv_store_iv(RETVAL, "hash_bfree", stat->hash_bfree);
|
||
|
hv_store_iv(RETVAL, "hash_bigpages", stat->hash_bigpages);
|
||
|
hv_store_iv(RETVAL, "hash_big_bfree", stat->hash_big_bfree);
|
||
|
hv_store_iv(RETVAL, "hash_overflows", stat->hash_overflows);
|
||
|
hv_store_iv(RETVAL, "hash_ovfl_free", stat->hash_ovfl_free);
|
||
|
hv_store_iv(RETVAL, "hash_dup", stat->hash_dup);
|
||
|
hv_store_iv(RETVAL, "hash_dup_free", stat->hash_dup_free);
|
||
|
#if DB_VERSION_MAJOR >= 3
|
||
|
hv_store_iv(RETVAL, "hash_metaflags", stat->hash_metaflags);
|
||
|
#endif
|
||
|
safefree(stat) ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Unknown PACKAGE = BerkeleyDB::Unknown PREFIX = hash_
|
||
|
|
||
|
void
|
||
|
_db_open_unknown(ref)
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
PPCODE:
|
||
|
{
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB RETVAL ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
#ifdef AT_LEAST_DB_5_2
|
||
|
static char * Names[] = {"", "Btree", "Hash", "Recno", "Queue", "Unknown", "Heap"} ;
|
||
|
#else
|
||
|
static char * Names[] = {"", "Btree", "Hash", "Recno", "Queue", "Unknown", "Heap"} ;
|
||
|
#endif
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Filename", char *) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.h_ffactor, "Ffactor") ;
|
||
|
SetValue_iv(info.h_nelem, "Nelem") ;
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_UNKNOWN, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
XPUSHs(sv_2mortal(newSViv(PTR2IV(RETVAL))));
|
||
|
if (RETVAL)
|
||
|
XPUSHs(sv_2mortal(newSVpv(Names[RETVAL->type], 0))) ;
|
||
|
else
|
||
|
XPUSHs(sv_2mortal(newSViv((IV)NULL)));
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Btree PACKAGE = BerkeleyDB::Btree PREFIX = btree_
|
||
|
|
||
|
BerkeleyDB::Btree::Raw
|
||
|
_db_open_btree(self, ref)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
Trace(("In _db_open_btree\n"));
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Filename", char*) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.bt_minkey, "Minkey") ;
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
if ((sv = readHash(hash, "Compare")) && sv != &PL_sv_undef) {
|
||
|
Trace((" Parsed Compare callback\n"));
|
||
|
info.bt_compare = btree_compare ;
|
||
|
db->compare = newSVsv(sv) ;
|
||
|
}
|
||
|
/* DB_DUPSORT was introduced in DB 2.5.9 */
|
||
|
if ((sv = readHash(hash, "DupCompare")) && sv != &PL_sv_undef) {
|
||
|
#ifdef DB_DUPSORT
|
||
|
Trace((" Parsed DupCompare callback\n"));
|
||
|
info.dup_compare = dup_compare ;
|
||
|
db->dup_compare = newSVsv(sv) ;
|
||
|
info.flags |= DB_DUP|DB_DUPSORT ;
|
||
|
#else
|
||
|
softCrash("DupCompare needs Berkeley DB 2.5.9 or later") ;
|
||
|
#endif
|
||
|
}
|
||
|
if ((sv = readHash(hash, "Prefix")) && sv != &PL_sv_undef) {
|
||
|
Trace((" Parsed Prefix callback\n"));
|
||
|
info.bt_prefix = btree_prefix ;
|
||
|
db->prefix = newSVsv(sv) ;
|
||
|
}
|
||
|
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_BTREE, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
HV *
|
||
|
db_stat(db, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Common db
|
||
|
HV * RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
DB_BTREE_STAT * stat ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, db->txn, &stat, flags) ;
|
||
|
#else
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, flags) ;
|
||
|
#else
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, safemalloc, flags) ;
|
||
|
#endif
|
||
|
#endif
|
||
|
if (db->Status) {
|
||
|
XSRETURN_UNDEF;
|
||
|
} else {
|
||
|
RETVAL = (HV*)sv_2mortal((SV*)newHV()) ;
|
||
|
hv_store_iv(RETVAL, "bt_magic", stat->bt_magic);
|
||
|
hv_store_iv(RETVAL, "bt_version", stat->bt_version);
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
hv_store_iv(RETVAL, "bt_metaflags", stat->bt_metaflags) ;
|
||
|
hv_store_iv(RETVAL, "bt_flags", stat->bt_metaflags) ;
|
||
|
#else
|
||
|
hv_store_iv(RETVAL, "bt_flags", stat->bt_flags) ;
|
||
|
#endif
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
hv_store_iv(RETVAL, "bt_maxkey", stat->bt_maxkey) ;
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "bt_minkey", stat->bt_minkey);
|
||
|
hv_store_iv(RETVAL, "bt_re_len", stat->bt_re_len);
|
||
|
hv_store_iv(RETVAL, "bt_re_pad", stat->bt_re_pad);
|
||
|
hv_store_iv(RETVAL, "bt_pagesize", stat->bt_pagesize);
|
||
|
hv_store_iv(RETVAL, "bt_levels", stat->bt_levels);
|
||
|
#ifdef AT_LEAST_DB_3_1
|
||
|
hv_store_iv(RETVAL, "bt_nkeys", stat->bt_nkeys);
|
||
|
hv_store_iv(RETVAL, "bt_ndata", stat->bt_ndata);
|
||
|
#else
|
||
|
hv_store_iv(RETVAL, "bt_nrecs", stat->bt_nrecs);
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "bt_int_pg", stat->bt_int_pg);
|
||
|
hv_store_iv(RETVAL, "bt_leaf_pg", stat->bt_leaf_pg);
|
||
|
hv_store_iv(RETVAL, "bt_dup_pg", stat->bt_dup_pg);
|
||
|
hv_store_iv(RETVAL, "bt_over_pg", stat->bt_over_pg);
|
||
|
hv_store_iv(RETVAL, "bt_free", stat->bt_free);
|
||
|
#if DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 5
|
||
|
hv_store_iv(RETVAL, "bt_freed", stat->bt_freed);
|
||
|
hv_store_iv(RETVAL, "bt_pfxsaved", stat->bt_pfxsaved);
|
||
|
hv_store_iv(RETVAL, "bt_split", stat->bt_split);
|
||
|
hv_store_iv(RETVAL, "bt_rootsplit", stat->bt_rootsplit);
|
||
|
hv_store_iv(RETVAL, "bt_fastsplit", stat->bt_fastsplit);
|
||
|
hv_store_iv(RETVAL, "bt_added", stat->bt_added);
|
||
|
hv_store_iv(RETVAL, "bt_deleted", stat->bt_deleted);
|
||
|
hv_store_iv(RETVAL, "bt_get", stat->bt_get);
|
||
|
hv_store_iv(RETVAL, "bt_cache_hit", stat->bt_cache_hit);
|
||
|
hv_store_iv(RETVAL, "bt_cache_miss", stat->bt_cache_miss);
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "bt_int_pgfree", stat->bt_int_pgfree);
|
||
|
hv_store_iv(RETVAL, "bt_leaf_pgfree", stat->bt_leaf_pgfree);
|
||
|
hv_store_iv(RETVAL, "bt_dup_pgfree", stat->bt_dup_pgfree);
|
||
|
hv_store_iv(RETVAL, "bt_over_pgfree", stat->bt_over_pgfree);
|
||
|
safefree(stat) ;
|
||
|
}
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
MODULE = BerkeleyDB::Heap PACKAGE = BerkeleyDB::Heap PREFIX = heap_
|
||
|
|
||
|
BerkeleyDB::Heap::Raw
|
||
|
_db_open_heap(self, ref)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_5_2
|
||
|
softCrash("BerkeleyDB::Heap needs Berkeley DB 5.2.x or better");
|
||
|
#else
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
Trace(("In _db_open_btree\n"));
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Filename", char*) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
SetValue_iv(info.heapsize_bytes, "HeapSize") ;
|
||
|
SetValue_iv(info.heapsize_gbytes, "HeapSizeGb") ;
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_HEAP, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Recno PACKAGE = BerkeleyDB::Recno PREFIX = recno_
|
||
|
|
||
|
BerkeleyDB::Recno::Raw
|
||
|
_db_open_recno(self, ref)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Fname", char*) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.bt_minkey, "Minkey") ;
|
||
|
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
SetValue_pv(info.re_source, "Source", char*) ;
|
||
|
if ((sv = readHash(hash, "Len")) && sv != &PL_sv_undef) {
|
||
|
info.re_len = SvIV(sv) ; ;
|
||
|
flagSet_DB2(info.flags, DB_FIXEDLEN) ;
|
||
|
}
|
||
|
if ((sv = readHash(hash, "Delim")) && sv != &PL_sv_undef) {
|
||
|
info.re_delim = SvPOK(sv) ? *SvPV(sv,PL_na) : SvIV(sv) ; ;
|
||
|
flagSet_DB2(info.flags, DB_DELIMITER) ;
|
||
|
}
|
||
|
if ((sv = readHash(hash, "Pad")) && sv != &PL_sv_undef) {
|
||
|
info.re_pad = (u_int32_t)SvPOK(sv) ? *SvPV(sv,PL_na) : SvIV(sv) ; ;
|
||
|
flagSet_DB2(info.flags, DB_PAD) ;
|
||
|
}
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
SetValue_iv(db->array_base, "ArrayBase") ;
|
||
|
db->array_base = (db->array_base == 0 ? 1 : 0) ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_RECNO, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Queue PACKAGE = BerkeleyDB::Queue PREFIX = recno_
|
||
|
|
||
|
BerkeleyDB::Queue::Raw
|
||
|
_db_open_queue(self, ref)
|
||
|
char * self
|
||
|
SV * ref
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("BerkeleyDB::Queue needs Berkeley DB 3.0.x or better");
|
||
|
#else
|
||
|
HV * hash ;
|
||
|
SV * sv ;
|
||
|
DB_INFO info ;
|
||
|
BerkeleyDB__Env dbenv = NULL;
|
||
|
SV * ref_dbenv = NULL;
|
||
|
const char * file = NULL ;
|
||
|
const char * subname = NULL ;
|
||
|
int flags = 0 ;
|
||
|
int mode = 0 ;
|
||
|
BerkeleyDB db ;
|
||
|
BerkeleyDB__Txn txn = NULL ;
|
||
|
char * enc_passwd = NULL ;
|
||
|
int enc_flags = 0 ;
|
||
|
|
||
|
hash = (HV*) SvRV(ref) ;
|
||
|
SetValue_pv(file, "Fname", char*) ;
|
||
|
SetValue_pv(subname, "Subname", char *) ;
|
||
|
SetValue_ov(dbenv, "Env", BerkeleyDB__Env) ;
|
||
|
ref_dbenv = sv ;
|
||
|
SetValue_ov(txn, "Txn", BerkeleyDB__Txn) ;
|
||
|
SetValue_iv(flags, "Flags") ;
|
||
|
SetValue_iv(mode, "Mode") ;
|
||
|
SetValue_pv(enc_passwd,"Enc_Passwd", char *) ;
|
||
|
SetValue_iv(enc_flags, "Enc_Flags") ;
|
||
|
|
||
|
Zero(&info, 1, DB_INFO) ;
|
||
|
SetValue_iv(info.db_cachesize, "Cachesize") ;
|
||
|
SetValue_iv(info.db_lorder, "Lorder") ;
|
||
|
SetValue_iv(info.db_pagesize, "Pagesize") ;
|
||
|
SetValue_iv(info.bt_minkey, "Minkey") ;
|
||
|
SetValue_iv(info.q_extentsize, "ExtentSize") ;
|
||
|
|
||
|
|
||
|
SetValue_iv(info.flags, "Property") ;
|
||
|
if ((sv = readHash(hash, "Len")) && sv != &PL_sv_undef) {
|
||
|
info.re_len = SvIV(sv) ; ;
|
||
|
flagSet_DB2(info.flags, DB_FIXEDLEN) ;
|
||
|
}
|
||
|
if ((sv = readHash(hash, "Pad")) && sv != &PL_sv_undef) {
|
||
|
info.re_pad = (u_int32_t)SvPOK(sv) ? *SvPV(sv,PL_na) : SvIV(sv) ; ;
|
||
|
flagSet_DB2(info.flags, DB_PAD) ;
|
||
|
}
|
||
|
ZMALLOC(db, BerkeleyDB_type) ;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
SetValue_iv(db->array_base, "ArrayBase") ;
|
||
|
db->array_base = (db->array_base == 0 ? 1 : 0) ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
|
||
|
RETVAL = my_db_open(db, ref, ref_dbenv, dbenv, txn, file, subname,
|
||
|
DB_QUEUE, flags, mode, &info, enc_passwd, enc_flags, hash) ;
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
HV *
|
||
|
db_stat(db, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Common db
|
||
|
HV * RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
softCrash("$db->db_stat for a Queue needs Berkeley DB 3.x or better") ;
|
||
|
#else /* Berkeley DB 3, or better */
|
||
|
DB_QUEUE_STAT * stat ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, db->txn, &stat, flags) ;
|
||
|
#else
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, flags) ;
|
||
|
#else
|
||
|
db->Status = ((db->dbp)->stat)(db->dbp, &stat, safemalloc, flags) ;
|
||
|
#endif
|
||
|
#endif
|
||
|
if (db->Status) {
|
||
|
XSRETURN_UNDEF;
|
||
|
} else {
|
||
|
RETVAL = (HV*)sv_2mortal((SV*)newHV()) ;
|
||
|
hv_store_iv(RETVAL, "qs_magic", stat->qs_magic) ;
|
||
|
hv_store_iv(RETVAL, "qs_version", stat->qs_version);
|
||
|
#ifdef AT_LEAST_DB_3_1
|
||
|
hv_store_iv(RETVAL, "qs_nkeys", stat->qs_nkeys);
|
||
|
hv_store_iv(RETVAL, "qs_ndata", stat->qs_ndata);
|
||
|
#else
|
||
|
hv_store_iv(RETVAL, "qs_nrecs", stat->qs_nrecs);
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "qs_pages", stat->qs_pages);
|
||
|
hv_store_iv(RETVAL, "qs_pagesize", stat->qs_pagesize);
|
||
|
hv_store_iv(RETVAL, "qs_pgfree", stat->qs_pgfree);
|
||
|
hv_store_iv(RETVAL, "qs_re_len", stat->qs_re_len);
|
||
|
hv_store_iv(RETVAL, "qs_re_pad", stat->qs_re_pad);
|
||
|
#ifdef AT_LEAST_DB_3_2
|
||
|
#else
|
||
|
hv_store_iv(RETVAL, "qs_start", stat->qs_start);
|
||
|
#endif
|
||
|
hv_store_iv(RETVAL, "qs_first_recno", stat->qs_first_recno);
|
||
|
hv_store_iv(RETVAL, "qs_cur_recno", stat->qs_cur_recno);
|
||
|
#if DB_VERSION_MAJOR >= 3
|
||
|
hv_store_iv(RETVAL, "qs_metaflags", stat->qs_metaflags);
|
||
|
#endif
|
||
|
safefree(stat) ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
MODULE = BerkeleyDB::Common PACKAGE = BerkeleyDB::Common PREFIX = dab_
|
||
|
|
||
|
|
||
|
DualType
|
||
|
db_close(db,flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
CODE:
|
||
|
Trace(("BerkeleyDB::Common::db_close %d\n", db));
|
||
|
#ifdef STRICT_CLOSE
|
||
|
if (db->txn)
|
||
|
softCrash("attempted to close a database while a transaction was still open") ;
|
||
|
if (db->open_cursors)
|
||
|
softCrash("attempted to close a database with %d open cursor(s)",
|
||
|
db->open_cursors) ;
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
if (db->open_sequences)
|
||
|
softCrash("attempted to close a database with %d open sequence(s)",
|
||
|
db->open_sequences) ;
|
||
|
#endif /* AT_LEAST_DB_4_3 */
|
||
|
#endif /* STRICT_CLOSE */
|
||
|
RETVAL = db->Status = ((db->dbp)->close)(db->dbp, flags) ;
|
||
|
if (db->parent_env && db->parent_env->open_dbs)
|
||
|
-- db->parent_env->open_dbs ;
|
||
|
db->active = FALSE ;
|
||
|
hash_delete("BerkeleyDB::Term::Db", (char *)db) ;
|
||
|
-- db->open_cursors ;
|
||
|
Trace(("end of BerkeleyDB::Common::db_close\n"));
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
void
|
||
|
dab__DESTROY(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
saveCurrentDB(db) ;
|
||
|
Trace(("In BerkeleyDB::Common::_DESTROY db %d dirty=%d\n", db, PL_dirty)) ;
|
||
|
destroyDB(db) ;
|
||
|
Trace(("End of BerkeleyDB::Common::DESTROY \n")) ;
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 6
|
||
|
#define db_cursor(db, txn, cur,flags) ((db->dbp)->cursor)(db->dbp, txn, cur)
|
||
|
#else
|
||
|
#define db_cursor(db, txn, cur,flags) ((db->dbp)->cursor)(db->dbp, txn, cur,flags)
|
||
|
#endif
|
||
|
BerkeleyDB::Cursor::Raw
|
||
|
_db_cursor(db, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Common db
|
||
|
BerkeleyDB::Cursor RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
ALIAS: __db_write_cursor = 1
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
DBC * cursor ;
|
||
|
saveCurrentDB(db) ;
|
||
|
if (ix == 1 && db->cds_enabled) {
|
||
|
#ifdef AT_LEAST_DB_3
|
||
|
flags |= DB_WRITECURSOR;
|
||
|
#else
|
||
|
flags |= DB_RMW;
|
||
|
#endif
|
||
|
}
|
||
|
if ((db->Status = db_cursor(db, db->txn, &cursor, flags)) == 0){
|
||
|
ZMALLOC(RETVAL, BerkeleyDB__Cursor_type) ;
|
||
|
db->open_cursors ++ ;
|
||
|
RETVAL->parent_db = db ;
|
||
|
RETVAL->cursor = cursor ;
|
||
|
RETVAL->dbp = db->dbp ;
|
||
|
RETVAL->txn = db->txn ;
|
||
|
RETVAL->type = db->type ;
|
||
|
RETVAL->recno_or_queue = db->recno_or_queue ;
|
||
|
RETVAL->cds_enabled = db->cds_enabled ;
|
||
|
RETVAL->filename = my_strdup(db->filename) ;
|
||
|
RETVAL->compare = db->compare ;
|
||
|
RETVAL->dup_compare = db->dup_compare ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
RETVAL->associated = db->associated ;
|
||
|
RETVAL->secondary_db = db->secondary_db;
|
||
|
RETVAL->primary_recno_or_queue = db->primary_recno_or_queue ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
RETVAL->associated_foreign = db->associated_foreign ;
|
||
|
#endif
|
||
|
RETVAL->prefix = db->prefix ;
|
||
|
RETVAL->hash = db->hash ;
|
||
|
RETVAL->partial = db->partial ;
|
||
|
RETVAL->doff = db->doff ;
|
||
|
RETVAL->dlen = db->dlen ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
RETVAL->array_base = db->array_base ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
#ifdef DBM_FILTERING
|
||
|
RETVAL->filtering = FALSE ;
|
||
|
RETVAL->filter_fetch_key = db->filter_fetch_key ;
|
||
|
RETVAL->filter_store_key = db->filter_store_key ;
|
||
|
RETVAL->filter_fetch_value = db->filter_fetch_value ;
|
||
|
RETVAL->filter_store_value = db->filter_store_value ;
|
||
|
#endif
|
||
|
/* RETVAL->info ; */
|
||
|
hash_store_iv("BerkeleyDB::Term::Cursor", (char *)RETVAL, 1) ;
|
||
|
}
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
BerkeleyDB::Cursor::Raw
|
||
|
_db_join(db, cursors, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Common db
|
||
|
AV * cursors
|
||
|
BerkeleyDB::Cursor RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
#if DB_VERSION_MAJOR == 2 && (DB_VERSION_MINOR < 5 || (DB_VERSION_MINOR == 5 && DB_VERSION_PATCH < 2))
|
||
|
softCrash("join needs Berkeley DB 2.5.2 or later") ;
|
||
|
#else /* Berkeley DB >= 2.5.2 */
|
||
|
DBC * join_cursor ;
|
||
|
DBC ** cursor_list ;
|
||
|
I32 count = av_len(cursors) + 1 ;
|
||
|
int i ;
|
||
|
saveCurrentDB(db) ;
|
||
|
if (count < 1 )
|
||
|
softCrash("db_join: No cursors in parameter list") ;
|
||
|
cursor_list = (DBC **)safemalloc(sizeof(DBC*) * (count + 1));
|
||
|
for (i = 0 ; i < count ; ++i) {
|
||
|
SV * obj = (SV*) * av_fetch(cursors, i, FALSE) ;
|
||
|
IV tmp = SvIV(getInnerObject(obj)) ;
|
||
|
BerkeleyDB__Cursor cur = INT2PTR(BerkeleyDB__Cursor, tmp);
|
||
|
if (cur->dbp == db->dbp)
|
||
|
softCrash("attempted to do a self-join");
|
||
|
cursor_list[i] = cur->cursor ;
|
||
|
}
|
||
|
cursor_list[i] = NULL ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
if ((db->Status = ((db->dbp)->join)(db->dbp, cursor_list, flags, &join_cursor)) == 0){
|
||
|
#else
|
||
|
if ((db->Status = ((db->dbp)->join)(db->dbp, cursor_list, &join_cursor, flags)) == 0){
|
||
|
#endif
|
||
|
ZMALLOC(RETVAL, BerkeleyDB__Cursor_type) ;
|
||
|
db->open_cursors ++ ;
|
||
|
RETVAL->parent_db = db ;
|
||
|
RETVAL->cursor = join_cursor ;
|
||
|
RETVAL->dbp = db->dbp ;
|
||
|
RETVAL->type = db->type ;
|
||
|
RETVAL->filename = my_strdup(db->filename) ;
|
||
|
RETVAL->compare = db->compare ;
|
||
|
RETVAL->dup_compare = db->dup_compare ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
RETVAL->associated = db->associated ;
|
||
|
RETVAL->secondary_db = db->secondary_db;
|
||
|
RETVAL->primary_recno_or_queue = db->primary_recno_or_queue ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
RETVAL->associated_foreign = db->associated_foreign ;
|
||
|
#endif
|
||
|
RETVAL->prefix = db->prefix ;
|
||
|
RETVAL->hash = db->hash ;
|
||
|
RETVAL->partial = db->partial ;
|
||
|
RETVAL->doff = db->doff ;
|
||
|
RETVAL->dlen = db->dlen ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
RETVAL->array_base = db->array_base ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
#ifdef DBM_FILTERING
|
||
|
RETVAL->filtering = FALSE ;
|
||
|
RETVAL->filter_fetch_key = db->filter_fetch_key ;
|
||
|
RETVAL->filter_store_key = db->filter_store_key ;
|
||
|
RETVAL->filter_fetch_value = db->filter_fetch_value ;
|
||
|
RETVAL->filter_store_value = db->filter_store_value ;
|
||
|
#endif
|
||
|
/* RETVAL->info ; */
|
||
|
hash_store_iv("BerkeleyDB::Term::Cursor", (char *)RETVAL, 1) ;
|
||
|
}
|
||
|
safefree(cursor_list) ;
|
||
|
#endif /* Berkeley DB >= 2.5.2 */
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
ArrayOffset(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
RETVAL = db->array_base ? 0 : 1 ;
|
||
|
#else
|
||
|
RETVAL = 0 ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
bool
|
||
|
cds_enabled(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
RETVAL = db->cds_enabled ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
int
|
||
|
stat_print(db, flags=0)
|
||
|
BerkeleyDB::Common db
|
||
|
u_int32_t flags
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$db->stat_print needs Berkeley DB 4.3 or better") ;
|
||
|
#else
|
||
|
RETVAL = db->dbp->stat_print(db->dbp, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
int
|
||
|
type(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
RETVAL = db->type ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
byteswapped(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#if DB_VERSION_MAJOR == 2 && DB_VERSION_MINOR < 5
|
||
|
softCrash("byteswapped needs Berkeley DB 2.5 or later") ;
|
||
|
#else
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
RETVAL = db->dbp->byteswapped ;
|
||
|
#else
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
db->dbp->get_byteswapped(db->dbp, &RETVAL) ;
|
||
|
#else
|
||
|
RETVAL = db->dbp->get_byteswapped(db->dbp) ;
|
||
|
#endif
|
||
|
#endif
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
status(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = db->Status ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
#ifdef DBM_FILTERING
|
||
|
|
||
|
#define setFilter(ftype) \
|
||
|
{ \
|
||
|
if (db->ftype) \
|
||
|
RETVAL = sv_mortalcopy(db->ftype) ; \
|
||
|
ST(0) = RETVAL ; \
|
||
|
if (db->ftype && (code == &PL_sv_undef)) { \
|
||
|
SvREFCNT_dec(db->ftype) ; \
|
||
|
db->ftype = NULL ; \
|
||
|
} \
|
||
|
else if (code) { \
|
||
|
if (db->ftype) \
|
||
|
sv_setsv(db->ftype, code) ; \
|
||
|
else \
|
||
|
db->ftype = newSVsv(code) ; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
|
||
|
SV *
|
||
|
filter_fetch_key(db, code)
|
||
|
BerkeleyDB::Common db
|
||
|
SV * code
|
||
|
SV * RETVAL = &PL_sv_undef ;
|
||
|
CODE:
|
||
|
DBM_setFilter(db->filter_fetch_key, code) ;
|
||
|
|
||
|
SV *
|
||
|
filter_store_key(db, code)
|
||
|
BerkeleyDB::Common db
|
||
|
SV * code
|
||
|
SV * RETVAL = &PL_sv_undef ;
|
||
|
CODE:
|
||
|
DBM_setFilter(db->filter_store_key, code) ;
|
||
|
|
||
|
SV *
|
||
|
filter_fetch_value(db, code)
|
||
|
BerkeleyDB::Common db
|
||
|
SV * code
|
||
|
SV * RETVAL = &PL_sv_undef ;
|
||
|
CODE:
|
||
|
DBM_setFilter(db->filter_fetch_value, code) ;
|
||
|
|
||
|
SV *
|
||
|
filter_store_value(db, code)
|
||
|
BerkeleyDB::Common db
|
||
|
SV * code
|
||
|
SV * RETVAL = &PL_sv_undef ;
|
||
|
CODE:
|
||
|
DBM_setFilter(db->filter_store_value, code) ;
|
||
|
|
||
|
#endif /* DBM_FILTERING */
|
||
|
|
||
|
void
|
||
|
partial_set(db, offset, length)
|
||
|
BerkeleyDB::Common db
|
||
|
u_int32_t offset
|
||
|
u_int32_t length
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
PPCODE:
|
||
|
if (GIMME == G_ARRAY) {
|
||
|
XPUSHs(sv_2mortal(newSViv(db->partial == DB_DBT_PARTIAL))) ;
|
||
|
XPUSHs(sv_2mortal(newSViv(db->doff))) ;
|
||
|
XPUSHs(sv_2mortal(newSViv(db->dlen))) ;
|
||
|
}
|
||
|
db->partial = DB_DBT_PARTIAL ;
|
||
|
db->doff = offset ;
|
||
|
db->dlen = length ;
|
||
|
|
||
|
|
||
|
void
|
||
|
partial_clear(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
PPCODE:
|
||
|
if (GIMME == G_ARRAY) {
|
||
|
XPUSHs(sv_2mortal(newSViv(db->partial == DB_DBT_PARTIAL))) ;
|
||
|
XPUSHs(sv_2mortal(newSViv(db->doff))) ;
|
||
|
XPUSHs(sv_2mortal(newSViv(db->dlen))) ;
|
||
|
}
|
||
|
db->partial =
|
||
|
db->doff =
|
||
|
db->dlen = 0 ;
|
||
|
|
||
|
|
||
|
#define db_del(db, key, flags) \
|
||
|
(db->Status = ((db->dbp)->del)(db->dbp, db->txn, &key, flags))
|
||
|
DualType
|
||
|
db_del(db, key, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY key
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
Trace(("db_del db[%p] in [%p] txn[%p] key[%.*s] flags[%d]\n", db->dbp, db, db->txn, key.size, key.data, flags)) ;
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
|
||
|
|
||
|
#ifdef AT_LEAST_DB_3
|
||
|
# ifdef AT_LEAST_DB_3_2
|
||
|
# define writeToKey() (flagSet(DB_CONSUME)||flagSet(DB_CONSUME_WAIT)||flagSet(DB_GET_BOTH)||flagSet(DB_SET_RECNO))
|
||
|
# else
|
||
|
# define writeToKey() (flagSet(DB_CONSUME)||flagSet(DB_GET_BOTH)||flagSet(DB_SET_RECNO))
|
||
|
# endif
|
||
|
#else
|
||
|
#define writeToKey() (flagSet(DB_GET_BOTH)||flagSet(DB_SET_RECNO))
|
||
|
#endif
|
||
|
#define db_get(db, key, data, flags) \
|
||
|
(db->Status = ((db->dbp)->get)(db->dbp, db->txn, &key, &data, flags))
|
||
|
DualType
|
||
|
db_get(db, key, data, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY_B key
|
||
|
DBT_OPT data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
SetPartial(data,db) ;
|
||
|
Trace(("db_get db[%p] in [%p] txn[%p] key [%.*s] flags[%d]\n", db->dbp, db, db->txn, key.size, key.data, flags)) ;
|
||
|
RETVAL = db_get(db, key, data, flags);
|
||
|
Trace((" RETVAL %d\n", RETVAL));
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key if (writeToKey()) OutputKey(ST(1), key) ;
|
||
|
data
|
||
|
|
||
|
#define db_exists(db, key, flags) \
|
||
|
(db->Status = ((db->dbp)->exists)(db->dbp, db->txn, &key, flags))
|
||
|
DualType
|
||
|
db_exists(db, key, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY_B key
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_6
|
||
|
softCrash("db_exists needs at least Berkeley DB 4.6");
|
||
|
#else
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
Trace(("db_exists db[%p] in [%p] txn[%p] key [%.*s] flags[%d]\n", db->dbp, db, db->txn, key.size, key.data, flags)) ;
|
||
|
RETVAL = db_exists(db, key, flags);
|
||
|
Trace((" RETVAL %d\n", RETVAL));
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
#define db_pget(db, key, pkey, data, flags) \
|
||
|
(db->Status = ((db->dbp)->pget)(db->dbp, db->txn, &key, &pkey, &data, flags))
|
||
|
DualType
|
||
|
db_pget(db, key, pkey, data, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY_B key
|
||
|
DBTKEY_Bpr pkey
|
||
|
DBT_OPT data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_3
|
||
|
softCrash("db_pget needs at least Berkeley DB 3.3");
|
||
|
#else
|
||
|
Trace(("db_pget db [%p] in [%p] txn [%p] flags [%d]\n", db->dbp, db, db->txn, flags)) ;
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
SetPartial(data,db) ;
|
||
|
RETVAL = db_pget(db, key, pkey, data, flags);
|
||
|
Trace((" RETVAL %d\n", RETVAL));
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key if (writeToKey()) OutputKey(ST(1), key) ;
|
||
|
pkey
|
||
|
data
|
||
|
|
||
|
#define db_put(db,key,data,flag) \
|
||
|
(db->Status = (db->dbp->put)(db->dbp,db->txn,&key,&data,flag))
|
||
|
DualType
|
||
|
db_put(db, key, data, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY key
|
||
|
DBT data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
/* SetPartial(data,db) ; */
|
||
|
Trace(("db_put db[%p] in [%p] txn[%p] key[%.*s] data [%.*s] flags[%d]\n", db->dbp, db, db->txn, key.size, key.data, data.size, data.data, flags)) ;
|
||
|
RETVAL = db_put(db, key, data, flags);
|
||
|
Trace((" RETVAL %d\n", RETVAL));
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key if (flagSet(DB_APPEND)) OutputKey(ST(1), key) ;
|
||
|
|
||
|
#define db_key_range(db, key, range, flags) \
|
||
|
(db->Status = ((db->dbp)->key_range)(db->dbp, db->txn, &key, &range, flags))
|
||
|
DualType
|
||
|
db_key_range(db, key, less, equal, greater, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY_B key
|
||
|
double less = 0.0 ;
|
||
|
double equal = 0.0 ;
|
||
|
double greater = 0.0 ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("key_range needs Berkeley DB 3.1.x or later") ;
|
||
|
#else
|
||
|
DB_KEY_RANGE range ;
|
||
|
range.less = range.equal = range.greater = 0.0 ;
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
RETVAL = db_key_range(db, key, range, flags);
|
||
|
if (RETVAL == 0) {
|
||
|
less = range.less ;
|
||
|
equal = range.equal;
|
||
|
greater = range.greater;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
less
|
||
|
equal
|
||
|
greater
|
||
|
|
||
|
|
||
|
#define db_fd(d, x) (db->Status = (db->dbp->fd)(db->dbp, &x))
|
||
|
int
|
||
|
db_fd(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
saveCurrentDB(db) ;
|
||
|
db_fd(db, RETVAL) ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
#define db_sync(db, fl) (db->Status = (db->dbp->sync)(db->dbp, fl))
|
||
|
DualType
|
||
|
db_sync(db, flags=0)
|
||
|
u_int flags
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
|
||
|
void
|
||
|
_Txn(db, txn=NULL)
|
||
|
BerkeleyDB::Common db
|
||
|
BerkeleyDB::Txn txn
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
if (txn) {
|
||
|
Trace(("_Txn[%p] in[%p] active [%d]\n", txn->txn, txn, txn->active));
|
||
|
ckActive_Transaction(txn->active) ;
|
||
|
db->txn = txn->txn ;
|
||
|
}
|
||
|
else {
|
||
|
Trace(("_Txn[undef] \n"));
|
||
|
db->txn = NULL ;
|
||
|
}
|
||
|
|
||
|
|
||
|
#define db_truncate(db, countp, flags) \
|
||
|
(db->Status = ((db->dbp)->truncate)(db->dbp, db->txn, &countp, flags))
|
||
|
DualType
|
||
|
truncate(db, countp, flags=0)
|
||
|
BerkeleyDB::Common db
|
||
|
u_int32_t countp = NO_INIT
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_3
|
||
|
softCrash("truncate needs Berkeley DB 3.3 or later") ;
|
||
|
#else
|
||
|
saveCurrentDB(db) ;
|
||
|
RETVAL = db_truncate(db, countp, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
countp
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4_1
|
||
|
# define db_associate(db, sec, cb, flags)\
|
||
|
(db->Status = ((db->dbp)->associate)(db->dbp, db->txn, sec->dbp, &cb, flags))
|
||
|
#else
|
||
|
# define db_associate(db, sec, cb, flags)\
|
||
|
(db->Status = ((db->dbp)->associate)(db->dbp, sec->dbp, &cb, flags))
|
||
|
#endif
|
||
|
DualType
|
||
|
associate(db, secondary, callback, flags=0)
|
||
|
BerkeleyDB::Common db
|
||
|
BerkeleyDB::Common secondary
|
||
|
SV* callback
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_3
|
||
|
softCrash("associate needs Berkeley DB 3.3 or later") ;
|
||
|
#else
|
||
|
saveCurrentDB(db) ;
|
||
|
/* db->associated = newSVsv(callback) ; */
|
||
|
secondary->associated = newSVsv(callback) ;
|
||
|
secondary->primary_recno_or_queue = db->recno_or_queue ;
|
||
|
/* secondary->dbp->app_private = secondary->associated ; */
|
||
|
secondary->secondary_db = TRUE;
|
||
|
if (secondary->recno_or_queue)
|
||
|
RETVAL = db_associate(db, secondary, associate_cb_recno, flags);
|
||
|
else
|
||
|
RETVAL = db_associate(db, secondary, associate_cb, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
#define db_associate_foreign(db, sec, cb, flags)\
|
||
|
(db->Status = ((db->dbp)->associate_foreign)(db->dbp, sec->dbp, cb, flags))
|
||
|
DualType
|
||
|
associate_foreign(db, secondary, callback, flags)
|
||
|
BerkeleyDB::Common db
|
||
|
BerkeleyDB::Common secondary
|
||
|
SV* callback
|
||
|
u_int32_t flags
|
||
|
foreign_cb_type callback_ptr = NULL;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_8
|
||
|
softCrash("associate_foreign needs Berkeley DB 4.8 or later") ;
|
||
|
#else
|
||
|
saveCurrentDB(db) ;
|
||
|
if (callback != &PL_sv_undef)
|
||
|
{
|
||
|
//softCrash("associate_foreign does not support callbacks yet") ;
|
||
|
secondary->associated_foreign = newSVsv(callback) ;
|
||
|
callback_ptr = ( secondary->recno_or_queue
|
||
|
? associate_foreign_cb_recno
|
||
|
: associate_foreign_cb);
|
||
|
}
|
||
|
secondary->primary_recno_or_queue = db->recno_or_queue ;
|
||
|
secondary->secondary_db = TRUE;
|
||
|
RETVAL = db_associate_foreign(db, secondary, callback_ptr, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
compact(db, start=NULL, stop=NULL, c_data=NULL, flags=0, end=NULL)
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
PREINIT:
|
||
|
DBTKEY end_key;
|
||
|
INPUT:
|
||
|
BerkeleyDB::Common db
|
||
|
SVnull* start
|
||
|
SVnull* stop
|
||
|
SVnull* c_data
|
||
|
u_int32_t flags
|
||
|
SVnull* end
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_4
|
||
|
softCrash("compact needs Berkeley DB 4.4 or later") ;
|
||
|
#else
|
||
|
DBTKEY start_key;
|
||
|
DBTKEY stop_key;
|
||
|
DBTKEY* start_p = NULL;
|
||
|
DBTKEY* stop_p = NULL;
|
||
|
DBTKEY* end_p = NULL;
|
||
|
DB_COMPACT cmpt;
|
||
|
DB_COMPACT* cmpt_p = NULL;
|
||
|
SV * sv;
|
||
|
HV* hash = NULL;
|
||
|
|
||
|
DBT_clear(start_key);
|
||
|
DBT_clear(stop_key);
|
||
|
DBT_clear(end_key);
|
||
|
Zero(&cmpt, 1, DB_COMPACT) ;
|
||
|
ckActive_Database(db->active) ;
|
||
|
saveCurrentDB(db) ;
|
||
|
if (start && SvOK(start)) {
|
||
|
start_p = &start_key;
|
||
|
DBM_ckFilter(start, filter_store_key, "filter_store_key");
|
||
|
GetKey(db, start, start_p);
|
||
|
}
|
||
|
if (stop && SvOK(stop)) {
|
||
|
stop_p = &stop_key;
|
||
|
DBM_ckFilter(stop, filter_store_key, "filter_store_key");
|
||
|
GetKey(db, stop, stop_p);
|
||
|
}
|
||
|
if (end) {
|
||
|
end_p = &end_key;
|
||
|
}
|
||
|
if (c_data && SvOK(c_data)) {
|
||
|
hash = (HV*) SvRV(c_data) ;
|
||
|
cmpt_p = & cmpt;
|
||
|
cmpt.compact_fillpercent = GetValue_iv(hash,"compact_fillpercent") ;
|
||
|
cmpt.compact_timeout = (db_timeout_t) GetValue_iv(hash, "compact_timeout");
|
||
|
}
|
||
|
RETVAL = (db->dbp)->compact(db->dbp, db->txn, start_p, stop_p, cmpt_p, flags, end_p);
|
||
|
if (RETVAL == 0 && hash) {
|
||
|
hv_store_iv(hash, "compact_deadlock", cmpt.compact_deadlock) ;
|
||
|
hv_store_iv(hash, "compact_levels", cmpt.compact_levels) ;
|
||
|
hv_store_iv(hash, "compact_pages_free", cmpt.compact_pages_free) ;
|
||
|
hv_store_iv(hash, "compact_pages_examine", cmpt.compact_pages_examine) ;
|
||
|
hv_store_iv(hash, "compact_pages_truncated", cmpt.compact_pages_truncated) ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
end if (RETVAL == 0 && end) OutputValue_B(ST(5), end_key) ;
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Cursor PACKAGE = BerkeleyDB::Cursor PREFIX = cu_
|
||
|
|
||
|
BerkeleyDB::Cursor::Raw
|
||
|
_c_dup(db, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
BerkeleyDB::Cursor RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Database(db->active) ;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_3
|
||
|
softCrash("c_dup needs at least Berkeley DB 3.0.x");
|
||
|
#else
|
||
|
DBC * newcursor ;
|
||
|
db->Status = ((db->cursor)->c_dup)(db->cursor, &newcursor, flags) ;
|
||
|
if (db->Status == 0){
|
||
|
ZMALLOC(RETVAL, BerkeleyDB__Cursor_type) ;
|
||
|
db->parent_db->open_cursors ++ ;
|
||
|
RETVAL->parent_db = db->parent_db ;
|
||
|
RETVAL->cursor = newcursor ;
|
||
|
RETVAL->dbp = db->dbp ;
|
||
|
RETVAL->type = db->type ;
|
||
|
RETVAL->recno_or_queue = db->recno_or_queue ;
|
||
|
RETVAL->primary_recno_or_queue = db->primary_recno_or_queue ;
|
||
|
RETVAL->cds_enabled = db->cds_enabled ;
|
||
|
RETVAL->filename = my_strdup(db->filename) ;
|
||
|
RETVAL->compare = db->compare ;
|
||
|
RETVAL->dup_compare = db->dup_compare ;
|
||
|
#ifdef AT_LEAST_DB_3_3
|
||
|
RETVAL->associated = db->associated ;
|
||
|
#endif
|
||
|
#ifdef AT_LEAST_DB_4_8
|
||
|
RETVAL->associated_foreign = db->associated_foreign ;
|
||
|
#endif
|
||
|
RETVAL->prefix = db->prefix ;
|
||
|
RETVAL->hash = db->hash ;
|
||
|
RETVAL->partial = db->partial ;
|
||
|
RETVAL->doff = db->doff ;
|
||
|
RETVAL->dlen = db->dlen ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
#ifdef ALLOW_RECNO_OFFSET
|
||
|
RETVAL->array_base = db->array_base ;
|
||
|
#endif /* ALLOW_RECNO_OFFSET */
|
||
|
#ifdef DBM_FILTERING
|
||
|
RETVAL->filtering = FALSE ;
|
||
|
RETVAL->filter_fetch_key = db->filter_fetch_key ;
|
||
|
RETVAL->filter_store_key = db->filter_store_key ;
|
||
|
RETVAL->filter_fetch_value = db->filter_fetch_value ;
|
||
|
RETVAL->filter_store_value = db->filter_store_value ;
|
||
|
#endif /* DBM_FILTERING */
|
||
|
/* RETVAL->info ; */
|
||
|
hash_store_iv("BerkeleyDB::Term::Cursor", (char *)RETVAL, 1) ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
_c_close(db)
|
||
|
BerkeleyDB::Cursor db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
hash_delete("BerkeleyDB::Term::Cursor", (char *)db) ;
|
||
|
CODE:
|
||
|
RETVAL = db->Status =
|
||
|
((db->cursor)->c_close)(db->cursor) ;
|
||
|
db->active = FALSE ;
|
||
|
if (db->parent_db->open_cursors)
|
||
|
-- db->parent_db->open_cursors ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
void
|
||
|
_DESTROY(db)
|
||
|
BerkeleyDB::Cursor db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
Trace(("In BerkeleyDB::Cursor::_DESTROY db %d dirty=%d active=%d\n", db, PL_dirty, db->active));
|
||
|
hash_delete("BerkeleyDB::Term::Cursor", (char *)db) ;
|
||
|
if (db->active)
|
||
|
((db->cursor)->c_close)(db->cursor) ;
|
||
|
if (db->parent_db->open_cursors)
|
||
|
-- db->parent_db->open_cursors ;
|
||
|
Safefree(db->filename) ;
|
||
|
Safefree(db) ;
|
||
|
Trace(("End of BerkeleyDB::Cursor::_DESTROY\n")) ;
|
||
|
|
||
|
DualType
|
||
|
status(db)
|
||
|
BerkeleyDB::Cursor db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = db->Status ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
#define cu_c_del(c,f) (c->Status = ((c->cursor)->c_del)(c->cursor,f))
|
||
|
DualType
|
||
|
cu_c_del(db, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
#define cu_c_get(c,k,d,f) (c->Status = (c->cursor->c_get)(c->cursor,&k,&d,f))
|
||
|
DualType
|
||
|
cu_c_get(db, key, data, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
DBTKEY_B key
|
||
|
DBT_B data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
Trace(("c_get db [%p] in [%p] flags [%d]\n", db->dbp, db, flags)) ;
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
/* DBT_clear(key); */
|
||
|
/* DBT_clear(data); */
|
||
|
SetPartial(data,db) ;
|
||
|
Trace(("c_get end\n")) ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key
|
||
|
data if (! flagSet(DB_JOIN_ITEM)) OutputValue_B(ST(2), data) ;
|
||
|
|
||
|
#define cu_c_pget(c,k,p,d,f) (c->Status = (c->secondary_db ? (c->cursor->c_pget)(c->cursor,&k,&p,&d,f) : EINVAL))
|
||
|
DualType
|
||
|
cu_c_pget(db, key, pkey, data, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
DBTKEY_B key
|
||
|
DBTKEY_Bpr pkey
|
||
|
DBT_B data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_3
|
||
|
softCrash("db_c_pget needs at least Berkeley DB 3.3");
|
||
|
#else
|
||
|
Trace(("c_pget db [%d] flags [%d]\n", db, flags)) ;
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
SetPartial(data,db) ;
|
||
|
RETVAL = cu_c_pget(db, key, pkey, data, flags);
|
||
|
Trace(("c_pget end\n")) ;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key if (writeToKey()) OutputKey(ST(1), key) ;
|
||
|
pkey
|
||
|
data
|
||
|
|
||
|
|
||
|
|
||
|
#define cu_c_put(c,k,d,f) (c->Status = (c->cursor->c_put)(c->cursor,&k,&d,f))
|
||
|
DualType
|
||
|
cu_c_put(db, key, data, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
DBTKEY key
|
||
|
DBT data
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
/* SetPartial(data,db) ; */
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
#define cu_c_count(c,p,f) (c->Status = (c->cursor->c_count)(c->cursor,&p,f))
|
||
|
DualType
|
||
|
cu_c_count(db, count, flags=0)
|
||
|
int flags
|
||
|
BerkeleyDB::Cursor db
|
||
|
u_int32_t count = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_3_1
|
||
|
softCrash("c_count needs at least Berkeley DB 3.1.x");
|
||
|
#else
|
||
|
Trace(("c_get count [%d] flags [%d]\n", db, flags)) ;
|
||
|
saveCurrentDB(db->parent_db);
|
||
|
ckActive_Cursor(db->active) ;
|
||
|
RETVAL = cu_c_count(db, count, flags) ;
|
||
|
Trace((" c_count got %d duplicates\n", count)) ;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
count
|
||
|
|
||
|
MODULE = BerkeleyDB::TxnMgr PACKAGE = BerkeleyDB::TxnMgr PREFIX = xx_
|
||
|
|
||
|
BerkeleyDB::Txn::Raw
|
||
|
_txn_begin(txnmgr, pid=NULL, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::TxnMgr txnmgr
|
||
|
BerkeleyDB::Txn pid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DB_TXN *txn ;
|
||
|
DB_TXN *p_id = NULL ;
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
if (txnmgr->env->Env->tx_info == NULL)
|
||
|
softCrash("Transaction Manager not enabled") ;
|
||
|
#endif
|
||
|
if (pid)
|
||
|
p_id = pid->txn ;
|
||
|
txnmgr->env->TxnMgrStatus =
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
txn_begin(txnmgr->env->Env->tx_info, p_id, &txn) ;
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
txnmgr->env->Env->txn_begin(txnmgr->env->Env, p_id, &txn, flags) ;
|
||
|
# else
|
||
|
txn_begin(txnmgr->env->Env, p_id, &txn, flags) ;
|
||
|
# endif
|
||
|
#endif
|
||
|
if (txnmgr->env->TxnMgrStatus == 0) {
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_Txn_type) ;
|
||
|
RETVAL->txn = txn ;
|
||
|
RETVAL->active = TRUE ;
|
||
|
Trace(("_txn_begin created txn [%d] in [%d]\n", txn, RETVAL));
|
||
|
hash_store_iv("BerkeleyDB::Term::Txn", (char *)RETVAL, 1) ;
|
||
|
}
|
||
|
else
|
||
|
RETVAL = NULL ;
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
DualType
|
||
|
status(mgr)
|
||
|
BerkeleyDB::TxnMgr mgr
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = mgr->env->TxnMgrStatus ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
void
|
||
|
_DESTROY(mgr)
|
||
|
BerkeleyDB::TxnMgr mgr
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
Trace(("In BerkeleyDB::TxnMgr::DESTROY dirty=%d\n", PL_dirty)) ;
|
||
|
Safefree(mgr) ;
|
||
|
Trace(("End of BerkeleyDB::TxnMgr::DESTROY\n")) ;
|
||
|
|
||
|
DualType
|
||
|
txn_close(txnp)
|
||
|
BerkeleyDB::TxnMgr txnp
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
|
||
|
#if DB_VERSION_MAJOR == 2
|
||
|
# define xx_txn_checkpoint(t,k,m,f) txn_checkpoint(t->env->Env->tx_info, k, m)
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_4
|
||
|
# define xx_txn_checkpoint(e,k,m,f) e->env->Env->txn_checkpoint(e->env->Env, k, m, f)
|
||
|
# else
|
||
|
# ifdef AT_LEAST_DB_3_1
|
||
|
# define xx_txn_checkpoint(t,k,m,f) txn_checkpoint(t->env->Env, k, m, 0)
|
||
|
# else
|
||
|
# define xx_txn_checkpoint(t,k,m,f) txn_checkpoint(t->env->Env, k, m)
|
||
|
# endif
|
||
|
# endif
|
||
|
#endif
|
||
|
DualType
|
||
|
xx_txn_checkpoint(txnp, kbyte, min, flags=0)
|
||
|
BerkeleyDB::TxnMgr txnp
|
||
|
long kbyte
|
||
|
long min
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
|
||
|
HV *
|
||
|
txn_stat(txnp)
|
||
|
BerkeleyDB::TxnMgr txnp
|
||
|
HV * RETVAL = NULL ;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DB_TXN_STAT * stat ;
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
if(txnp->env->Env->txn_stat(txnp->env->Env, &stat, 0) == 0) {
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_3_3
|
||
|
if(txn_stat(txnp->env->Env, &stat) == 0) {
|
||
|
# else
|
||
|
# if DB_VERSION_MAJOR == 2
|
||
|
if(txn_stat(txnp->env->Env->tx_info, &stat, safemalloc) == 0) {
|
||
|
# else
|
||
|
if(txn_stat(txnp->env->Env, &stat, safemalloc) == 0) {
|
||
|
# endif
|
||
|
# endif
|
||
|
#endif
|
||
|
RETVAL = (HV*)sv_2mortal((SV*)newHV()) ;
|
||
|
hv_store_iv(RETVAL, "st_time_ckp", stat->st_time_ckp) ;
|
||
|
hv_store_iv(RETVAL, "st_last_txnid", stat->st_last_txnid) ;
|
||
|
hv_store_iv(RETVAL, "st_maxtxns", stat->st_maxtxns) ;
|
||
|
hv_store_iv(RETVAL, "st_naborts", stat->st_naborts) ;
|
||
|
hv_store_iv(RETVAL, "st_nbegins", stat->st_nbegins) ;
|
||
|
hv_store_iv(RETVAL, "st_ncommits", stat->st_ncommits) ;
|
||
|
hv_store_iv(RETVAL, "st_nactive", stat->st_nactive) ;
|
||
|
#if DB_VERSION_MAJOR > 2
|
||
|
hv_store_iv(RETVAL, "st_maxnactive", stat->st_maxnactive) ;
|
||
|
hv_store_iv(RETVAL, "st_regsize", stat->st_regsize) ;
|
||
|
hv_store_iv(RETVAL, "st_region_wait", stat->st_region_wait) ;
|
||
|
hv_store_iv(RETVAL, "st_region_nowait", stat->st_region_nowait) ;
|
||
|
#endif
|
||
|
safefree(stat) ;
|
||
|
}
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
BerkeleyDB::TxnMgr
|
||
|
txn_open(dir, flags, mode, dbenv)
|
||
|
int flags
|
||
|
const char * dir
|
||
|
int mode
|
||
|
BerkeleyDB::Env dbenv
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Txn PACKAGE = BerkeleyDB::Txn PREFIX = xx_
|
||
|
|
||
|
DualType
|
||
|
status(tid)
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
RETVAL = tid->Status ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_timeout(txn, timeout, flags=0)
|
||
|
BerkeleyDB::Txn txn
|
||
|
db_timeout_t timeout
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Transaction(txn->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4
|
||
|
softCrash("$env->set_timeout needs Berkeley DB 4.x or better") ;
|
||
|
#else
|
||
|
RETVAL = txn->Status = txn->txn->set_timeout(txn->txn, timeout, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
set_tx_max(env, max)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t max
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_2_3
|
||
|
softCrash("$env->set_tx_max needs Berkeley DB 2_3.x or better") ;
|
||
|
#else
|
||
|
dieIfEnvOpened(env, "set_tx_max");
|
||
|
RETVAL = env->Status = env->Env->set_tx_max(env->Env, max);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
int
|
||
|
get_tx_max(env, max)
|
||
|
BerkeleyDB::Env env
|
||
|
u_int32_t max = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Database(env->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_2_3
|
||
|
softCrash("$env->get_tx_max needs Berkeley DB 2_3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = env->Status = env->Env->get_tx_max(env->Env, &max);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
max
|
||
|
|
||
|
void
|
||
|
_DESTROY(tid)
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
Trace(("In BerkeleyDB::Txn::_DESTROY txn [%d] active [%d] dirty=%d\n", tid->txn, tid->active, PL_dirty)) ;
|
||
|
if (tid->active)
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
tid->txn->abort(tid->txn) ;
|
||
|
#else
|
||
|
txn_abort(tid->txn) ;
|
||
|
#endif
|
||
|
hash_delete("BerkeleyDB::Term::Txn", (char *)tid) ;
|
||
|
Safefree(tid) ;
|
||
|
Trace(("End of BerkeleyDB::Txn::DESTROY\n")) ;
|
||
|
|
||
|
#define xx_txn_unlink(d,f,e) txn_unlink(d,f,&(e->Env))
|
||
|
DualType
|
||
|
xx_txn_unlink(dir, force, dbenv)
|
||
|
const char * dir
|
||
|
int force
|
||
|
BerkeleyDB::Env dbenv
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
# define xx_txn_prepare(t) (t->Status = t->txn->prepare(t->txn, 0))
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_3_3
|
||
|
# define xx_txn_prepare(t) (t->Status = txn_prepare(t->txn, 0))
|
||
|
# else
|
||
|
# define xx_txn_prepare(t) (t->Status = txn_prepare(t->txn))
|
||
|
# endif
|
||
|
#endif
|
||
|
DualType
|
||
|
xx_txn_prepare(tid)
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Transaction(tid->active) ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
# define _txn_commit(t,flags) (t->Status = t->txn->commit(t->txn, flags))
|
||
|
#else
|
||
|
# if DB_VERSION_MAJOR == 2
|
||
|
# define _txn_commit(t,flags) (t->Status = txn_commit(t->txn))
|
||
|
# else
|
||
|
# define _txn_commit(t, flags) (t->Status = txn_commit(t->txn, flags))
|
||
|
# endif
|
||
|
#endif
|
||
|
DualType
|
||
|
_txn_commit(tid, flags=0)
|
||
|
u_int32_t flags
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Transaction(tid->active) ;
|
||
|
hash_delete("BerkeleyDB::Term::Txn", (char *)tid) ;
|
||
|
tid->active = FALSE ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
# define _txn_abort(t) (t->Status = t->txn->abort(t->txn))
|
||
|
#else
|
||
|
# define _txn_abort(t) (t->Status = txn_abort(t->txn))
|
||
|
#endif
|
||
|
DualType
|
||
|
_txn_abort(tid)
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Transaction(tid->active) ;
|
||
|
hash_delete("BerkeleyDB::Term::Txn", (char *)tid) ;
|
||
|
tid->active = FALSE ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
# define _txn_discard(t,f) (t->Status = t->txn->discard(t->txn, f))
|
||
|
#else
|
||
|
# ifdef AT_LEAST_DB_3_3_4
|
||
|
# define _txn_discard(t,f) (t->Status = txn_discard(t->txn, f))
|
||
|
# else
|
||
|
# define _txn_discard(t,f) (int)softCrash("txn_discard needs Berkeley DB 3.3.4 or better") ;
|
||
|
# endif
|
||
|
#endif
|
||
|
DualType
|
||
|
_txn_discard(tid, flags=0)
|
||
|
BerkeleyDB::Txn tid
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Transaction(tid->active) ;
|
||
|
hash_delete("BerkeleyDB::Term::Txn", (char *)tid) ;
|
||
|
tid->active = FALSE ;
|
||
|
|
||
|
#ifdef AT_LEAST_DB_4
|
||
|
# define xx_txn_id(t) t->txn->id(t->txn)
|
||
|
#else
|
||
|
# define xx_txn_id(t) txn_id(t->txn)
|
||
|
#endif
|
||
|
u_int32_t
|
||
|
xx_txn_id(tid)
|
||
|
BerkeleyDB::Txn tid
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
|
||
|
MODULE = BerkeleyDB::_tiedHash PACKAGE = BerkeleyDB::_tiedHash
|
||
|
|
||
|
int
|
||
|
FIRSTKEY(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DBTKEY key ;
|
||
|
DBT value ;
|
||
|
DBC * cursor ;
|
||
|
|
||
|
/*
|
||
|
TODO!
|
||
|
set partial value to 0 - to eliminate the retrieval of
|
||
|
the value need to store any existing partial settings &
|
||
|
restore at the end.
|
||
|
|
||
|
*/
|
||
|
saveCurrentDB(db) ;
|
||
|
DBT_clear(key) ;
|
||
|
DBT_clear(value) ;
|
||
|
/* If necessary create a cursor for FIRSTKEY/NEXTKEY use */
|
||
|
if (!db->cursor &&
|
||
|
(db->Status = db_cursor(db, db->txn, &cursor, 0)) == 0 )
|
||
|
db->cursor = cursor ;
|
||
|
|
||
|
if (db->cursor)
|
||
|
RETVAL = (db->Status) =
|
||
|
((db->cursor)->c_get)(db->cursor, &key, &value, DB_FIRST);
|
||
|
else
|
||
|
RETVAL = db->Status ;
|
||
|
/* check for end of cursor */
|
||
|
if (RETVAL == DB_NOTFOUND) {
|
||
|
((db->cursor)->c_close)(db->cursor) ;
|
||
|
db->cursor = NULL ;
|
||
|
}
|
||
|
ST(0) = sv_newmortal();
|
||
|
OutputKey(ST(0), key)
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
int
|
||
|
NEXTKEY(db, key)
|
||
|
BerkeleyDB::Common db
|
||
|
DBTKEY key = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
DBT value ;
|
||
|
|
||
|
saveCurrentDB(db) ;
|
||
|
DBT_clear(key) ;
|
||
|
DBT_clear(value) ;
|
||
|
key.flags = 0 ;
|
||
|
RETVAL = (db->Status) =
|
||
|
((db->cursor)->c_get)(db->cursor, &key, &value, DB_NEXT);
|
||
|
|
||
|
/* check for end of cursor */
|
||
|
if (RETVAL == DB_NOTFOUND) {
|
||
|
((db->cursor)->c_close)(db->cursor) ;
|
||
|
db->cursor = NULL ;
|
||
|
}
|
||
|
ST(0) = sv_newmortal();
|
||
|
OutputKey(ST(0), key)
|
||
|
}
|
||
|
|
||
|
MODULE = BerkeleyDB::_tiedArray PACKAGE = BerkeleyDB::_tiedArray
|
||
|
|
||
|
I32
|
||
|
FETCHSIZE(db)
|
||
|
BerkeleyDB::Common db
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
saveCurrentDB(db) ;
|
||
|
RETVAL = GetArrayLength(db) ;
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Common PACKAGE = BerkeleyDB::Common
|
||
|
|
||
|
BerkeleyDB::Sequence
|
||
|
db_create_sequence(db, flags=0)
|
||
|
BerkeleyDB::Common db
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
{
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->create_sequence needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
DB_SEQUENCE * seq ;
|
||
|
saveCurrentDB(db);
|
||
|
RETVAL = NULL;
|
||
|
if (db_sequence_create(&seq, db->dbp, flags) == 0)
|
||
|
{
|
||
|
ZMALLOC(RETVAL, BerkeleyDB_Sequence_type);
|
||
|
RETVAL->db = db;
|
||
|
RETVAL->seq = seq;
|
||
|
RETVAL->active = TRUE;
|
||
|
++ db->open_sequences ;
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB::Sequence PACKAGE = BerkeleyDB::Sequence PREFIX = seq_
|
||
|
|
||
|
DualType
|
||
|
open(seq, key, flags=0)
|
||
|
BerkeleyDB::Sequence seq
|
||
|
DBTKEY_seq key
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->create_sequence needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->open(seq->seq, seq->db->txn, &key, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
close(seq,flags=0)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
u_int32_t flags;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->close needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = 0;
|
||
|
if (seq->active) {
|
||
|
-- seq->db->open_sequences;
|
||
|
RETVAL = (seq->seq->close)(seq->seq, flags);
|
||
|
}
|
||
|
seq->active = FALSE;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
remove(seq,flags=0)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
u_int32_t flags;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->remove needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = 0;
|
||
|
if (seq->active)
|
||
|
RETVAL = seq->seq->remove(seq->seq, seq->db->txn, flags);
|
||
|
seq->active = FALSE;
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
void
|
||
|
DESTROY(seq)
|
||
|
BerkeleyDB::Sequence seq
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
CODE:
|
||
|
#ifdef AT_LEAST_DB_4_3
|
||
|
if (seq->active)
|
||
|
(seq->seq->close)(seq->seq, 0);
|
||
|
Safefree(seq);
|
||
|
#endif
|
||
|
|
||
|
DualType
|
||
|
get(seq, element, delta=1, flags=0)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
IV delta;
|
||
|
db_seq_t element = NO_INIT
|
||
|
u_int32_t flags;
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->get needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->get(seq->seq, seq->db->txn, delta, &element, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
element
|
||
|
|
||
|
DualType
|
||
|
get_key(seq, key)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
DBTKEY_seq key = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->get_key needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
DBT_clear(key);
|
||
|
RETVAL = seq->seq->get_key(seq->seq, &key);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
key
|
||
|
|
||
|
DualType
|
||
|
initial_value(seq, low, high=0)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
int low
|
||
|
int high
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->initial_value needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->initial_value(seq->seq, (db_seq_t)(high << 32 + low));
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
set_cachesize(seq, size)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
int32_t size
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->set_cachesize needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->set_cachesize(seq->seq, size);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
get_cachesize(seq, size)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
int32_t size = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->get_cachesize needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->get_cachesize(seq->seq, &size);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
size
|
||
|
|
||
|
DualType
|
||
|
set_flags(seq, flags)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
u_int32_t flags
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->set_flags needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->set_flags(seq->seq, flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
|
||
|
DualType
|
||
|
get_flags(seq, flags)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
u_int32_t flags = NO_INIT
|
||
|
PREINIT:
|
||
|
dMY_CXT;
|
||
|
INIT:
|
||
|
ckActive_Sequence(seq->active) ;
|
||
|
CODE:
|
||
|
#ifndef AT_LEAST_DB_4_3
|
||
|
softCrash("$seq->get_flags needs Berkeley DB 4.3.x or better") ;
|
||
|
#else
|
||
|
RETVAL = seq->seq->get_flags(seq->seq, &flags);
|
||
|
#endif
|
||
|
OUTPUT:
|
||
|
RETVAL
|
||
|
flags
|
||
|
|
||
|
DualType
|
||
|
set_range(seq)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
DualType
|
||
|
stat(seq)
|
||
|
BerkeleyDB::Sequence seq;
|
||
|
NOT_IMPLEMENTED_YET
|
||
|
|
||
|
|
||
|
MODULE = BerkeleyDB PACKAGE = BerkeleyDB
|
||
|
|
||
|
BOOT:
|
||
|
{
|
||
|
#ifdef dTHX
|
||
|
dTHX;
|
||
|
#endif
|
||
|
SV * sv_err = perl_get_sv(ERR_BUFF, GV_ADD|GV_ADDMULTI) ;
|
||
|
SV * version_sv = perl_get_sv("BerkeleyDB::db_version", GV_ADD|GV_ADDMULTI) ;
|
||
|
SV * ver_sv = perl_get_sv("BerkeleyDB::db_ver", GV_ADD|GV_ADDMULTI) ;
|
||
|
int Major, Minor, Patch ;
|
||
|
MY_CXT_INIT;
|
||
|
(void)db_version(&Major, &Minor, &Patch) ;
|
||
|
/* Check that the versions of db.h and libdb.a are the same */
|
||
|
if (Major != DB_VERSION_MAJOR || Minor != DB_VERSION_MINOR
|
||
|
|| Patch != DB_VERSION_PATCH)
|
||
|
croak("\nBerkeleyDB needs compatible versions of libdb & db.h\n\tyou have db.h version %d.%d.%d and libdb version %d.%d.%d\n",
|
||
|
DB_VERSION_MAJOR, DB_VERSION_MINOR, DB_VERSION_PATCH,
|
||
|
Major, Minor, Patch) ;
|
||
|
|
||
|
if (Major < 2 || (Major == 2 && Minor < 6))
|
||
|
{
|
||
|
croak("BerkeleyDB needs Berkeley DB 2.6 or greater. This is %d.%d.%d\n",
|
||
|
Major, Minor, Patch) ;
|
||
|
}
|
||
|
sv_setpvf(version_sv, "%d.%d", Major, Minor) ;
|
||
|
sv_setpvf(ver_sv, "%d.%03d%03d", Major, Minor, Patch) ;
|
||
|
sv_setpv(sv_err, "");
|
||
|
|
||
|
DBT_clear(empty) ;
|
||
|
empty.data = &zero ;
|
||
|
empty.size = sizeof(db_recno_t) ;
|
||
|
empty.flags = 0 ;
|
||
|
|
||
|
}
|
||
|
|