mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 06:44:16 +01:00
155e78f014
BitKeeper/deleted/.del-ex_access.wpj~3df6ae8c99bf7c5f: Delete: bdb/build_vxworks/ex_access/ex_access.wpj BitKeeper/deleted/.del-ex_btrec.wpj~a7622f1c6f432dc6: Delete: bdb/build_vxworks/ex_btrec/ex_btrec.wpj BitKeeper/deleted/.del-ex_dbclient.wpj~7345440f3b204cdd: Delete: bdb/build_vxworks/ex_dbclient/ex_dbclient.wpj BitKeeper/deleted/.del-ex_env.wpj~fbe1ab10b04e8b74: Delete: bdb/build_vxworks/ex_env/ex_env.wpj BitKeeper/deleted/.del-ex_mpool.wpj~4479cfd5c45f327d: Delete: bdb/build_vxworks/ex_mpool/ex_mpool.wpj BitKeeper/deleted/.del-ex_tpcb.wpj~f78093006e14bf41: Delete: bdb/build_vxworks/ex_tpcb/ex_tpcb.wpj BitKeeper/deleted/.del-db_buildall.dsp~bd749ff6da11682: Delete: bdb/build_win32/db_buildall.dsp BitKeeper/deleted/.del-cxx_app.cpp~ad8df8e0791011ed: Delete: bdb/cxx/cxx_app.cpp BitKeeper/deleted/.del-cxx_log.cpp~a50ff3118fe06952: Delete: bdb/cxx/cxx_log.cpp BitKeeper/deleted/.del-cxx_table.cpp~ecd751e79b055556: Delete: bdb/cxx/cxx_table.cpp BitKeeper/deleted/.del-namemap.txt~796a3acd3885d8fd: Delete: bdb/cxx/namemap.txt BitKeeper/deleted/.del-Design.fileop~3ca4da68f1727373: Delete: bdb/db/Design.fileop BitKeeper/deleted/.del-db185_int.h~61bee3736e7959ef: Delete: bdb/db185/db185_int.h BitKeeper/deleted/.del-acconfig.h~411e8854d67ad8b5: Delete: bdb/dist/acconfig.h BitKeeper/deleted/.del-mutex.m4~a13383cde18a64e1: Delete: bdb/dist/aclocal/mutex.m4 BitKeeper/deleted/.del-options.m4~b9d0ca637213750a: Delete: bdb/dist/aclocal/options.m4 BitKeeper/deleted/.del-programs.m4~3ce7890b47732b30: Delete: bdb/dist/aclocal/programs.m4 BitKeeper/deleted/.del-tcl.m4~f944e2db93c3b6db: Delete: bdb/dist/aclocal/tcl.m4 BitKeeper/deleted/.del-types.m4~59cae158c9a32cff: Delete: bdb/dist/aclocal/types.m4 BitKeeper/deleted/.del-script~d38f6d3a4f159cb4: Delete: bdb/dist/build/script BitKeeper/deleted/.del-configure.in~ac795a92c8fe049c: Delete: bdb/dist/configure.in BitKeeper/deleted/.del-ltconfig~66bbd007d8024af: Delete: bdb/dist/ltconfig BitKeeper/deleted/.del-rec_ctemp~a28554362534f00a: Delete: bdb/dist/rec_ctemp BitKeeper/deleted/.del-s_tcl~2ffe4326459fcd9f: Delete: bdb/dist/s_tcl BitKeeper/deleted/.del-.IGNORE_ME~d8148b08fa7d5d15: Delete: bdb/dist/template/.IGNORE_ME BitKeeper/deleted/.del-btree.h~179f2aefec1753d: Delete: bdb/include/btree.h BitKeeper/deleted/.del-cxx_int.h~6b649c04766508f8: Delete: bdb/include/cxx_int.h BitKeeper/deleted/.del-db.src~6b433ae615b16a8d: Delete: bdb/include/db.src BitKeeper/deleted/.del-db_185.h~ad8b373d9391d35c: Delete: bdb/include/db_185.h BitKeeper/deleted/.del-db_am.h~a714912b6b75932f: Delete: bdb/include/db_am.h BitKeeper/deleted/.del-db_cxx.h~fcafadf45f5d19e9: Delete: bdb/include/db_cxx.h BitKeeper/deleted/.del-db_dispatch.h~6844f20f7eb46904: Delete: bdb/include/db_dispatch.h BitKeeper/deleted/.del-db_int.src~419a3f48b6a01da7: Delete: bdb/include/db_int.src BitKeeper/deleted/.del-db_join.h~76f9747a42c3399a: Delete: bdb/include/db_join.h BitKeeper/deleted/.del-db_page.h~e302ca3a4db3abdc: Delete: bdb/include/db_page.h BitKeeper/deleted/.del-db_server_int.h~e1d20b6ba3bca1ab: Delete: bdb/include/db_server_int.h BitKeeper/deleted/.del-db_shash.h~5fbf2d696fac90f3: Delete: bdb/include/db_shash.h BitKeeper/deleted/.del-db_swap.h~1e60887550864a59: Delete: bdb/include/db_swap.h BitKeeper/deleted/.del-db_upgrade.h~c644eee73701fc8d: Delete: bdb/include/db_upgrade.h BitKeeper/deleted/.del-db_verify.h~b8d6c297c61f342e: Delete: bdb/include/db_verify.h BitKeeper/deleted/.del-debug.h~dc2b4f2cf27ccebc: Delete: bdb/include/debug.h BitKeeper/deleted/.del-hash.h~2aaa548b28882dfb: Delete: bdb/include/hash.h BitKeeper/deleted/.del-lock.h~a761c1b7de57b77f: Delete: bdb/include/lock.h BitKeeper/deleted/.del-log.h~ff20184238e35e4d: Delete: bdb/include/log.h BitKeeper/deleted/.del-mp.h~7e317597622f3411: Delete: bdb/include/mp.h BitKeeper/deleted/.del-mutex.h~d3ae7a2977a68137: Delete: bdb/include/mutex.h BitKeeper/deleted/.del-os.h~91867cc8757cd0e3: Delete: bdb/include/os.h BitKeeper/deleted/.del-os_jump.h~e1b939fa5151d4be: Delete: bdb/include/os_jump.h BitKeeper/deleted/.del-qam.h~6fad0c1b5723d597: Delete: bdb/include/qam.h BitKeeper/deleted/.del-queue.h~4c72c0826c123d5: Delete: bdb/include/queue.h BitKeeper/deleted/.del-region.h~513fe04d977ca0fc: Delete: bdb/include/region.h BitKeeper/deleted/.del-shqueue.h~525fc3e6c2025c36: Delete: bdb/include/shqueue.h BitKeeper/deleted/.del-tcl_db.h~c536fd61a844f23f: Delete: bdb/include/tcl_db.h BitKeeper/deleted/.del-txn.h~c8d94b221ec147e4: Delete: bdb/include/txn.h BitKeeper/deleted/.del-xa.h~ecc466493aae9d9a: Delete: bdb/include/xa.h BitKeeper/deleted/.del-DbRecoveryInit.java~756b52601a0b9023: Delete: bdb/java/src/com/sleepycat/db/DbRecoveryInit.java BitKeeper/deleted/.del-DbTxnRecover.java~74607cba7ab89d6d: Delete: bdb/java/src/com/sleepycat/db/DbTxnRecover.java BitKeeper/deleted/.del-lock_conflict.c~fc5e0f14cf597a2b: Delete: bdb/lock/lock_conflict.c BitKeeper/deleted/.del-log.src~53ac9e7b5cb023f2: Delete: bdb/log/log.src BitKeeper/deleted/.del-log_findckp.c~24287f008916e81f: Delete: bdb/log/log_findckp.c BitKeeper/deleted/.del-log_rec.c~d51711f2cac09297: Delete: bdb/log/log_rec.c BitKeeper/deleted/.del-log_register.c~b40bb4efac75ca15: Delete: bdb/log/log_register.c BitKeeper/deleted/.del-Design~b3d0f179f2767b: Delete: bdb/mp/Design BitKeeper/deleted/.del-os_finit.c~95dbefc6fe79b26c: Delete: bdb/os/os_finit.c BitKeeper/deleted/.del-os_abs.c~df95d1e7db81924: Delete: bdb/os_vxworks/os_abs.c BitKeeper/deleted/.del-os_finit.c~803b484bdb9d0122: Delete: bdb/os_vxworks/os_finit.c BitKeeper/deleted/.del-os_map.c~3a6d7926398b76d3: Delete: bdb/os_vxworks/os_map.c BitKeeper/deleted/.del-os_finit.c~19a227c6d3c78ad: Delete: bdb/os_win32/os_finit.c BitKeeper/deleted/.del-log-corruption.patch~1cf2ecc7c6408d5d: Delete: bdb/patches/log-corruption.patch BitKeeper/deleted/.del-Btree.pm~af6d0c5eaed4a98e: Delete: bdb/perl.BerkeleyDB/BerkeleyDB/Btree.pm BitKeeper/deleted/.del-BerkeleyDB.pm~7244036d4482643: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pm BitKeeper/deleted/.del-BerkeleyDB.pod~e7b18fd6132448e3: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pod BitKeeper/deleted/.del-Hash.pm~10292a26c06a5c95: Delete: bdb/perl.BerkeleyDB/BerkeleyDB/Hash.pm BitKeeper/deleted/.del-BerkeleyDB.pod.P~79f76a1495eda203: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.pod.P BitKeeper/deleted/.del-BerkeleyDB.xs~80c99afbd98e392c: Delete: bdb/perl.BerkeleyDB/BerkeleyDB.xs BitKeeper/deleted/.del-Changes~729c1891efa60de9: Delete: bdb/perl.BerkeleyDB/Changes BitKeeper/deleted/.del-MANIFEST~63a1e34aecf157a0: Delete: bdb/perl.BerkeleyDB/MANIFEST BitKeeper/deleted/.del-Makefile.PL~c68797707d8df87a: Delete: bdb/perl.BerkeleyDB/Makefile.PL BitKeeper/deleted/.del-README~5f2f579b1a241407: Delete: bdb/perl.BerkeleyDB/README BitKeeper/deleted/.del-Todo~dca3c66c193adda9: Delete: bdb/perl.BerkeleyDB/Todo BitKeeper/deleted/.del-config.in~ae81681e450e0999: Delete: bdb/perl.BerkeleyDB/config.in BitKeeper/deleted/.del-dbinfo~28ad67d83be4f68e: Delete: bdb/perl.BerkeleyDB/dbinfo BitKeeper/deleted/.del-mkconsts~543ab60669c7a04e: Delete: bdb/perl.BerkeleyDB/mkconsts BitKeeper/deleted/.del-mkpod~182c0ca54e439afb: Delete: bdb/perl.BerkeleyDB/mkpod BitKeeper/deleted/.del-5.004~e008cb5a48805543: Delete: bdb/perl.BerkeleyDB/patches/5.004 BitKeeper/deleted/.del-irix_6_5.pl~61662bb08afcdec8: Delete: bdb/perl.BerkeleyDB/hints/irix_6_5.pl BitKeeper/deleted/.del-solaris.pl~6771e7182394e152: Delete: bdb/perl.BerkeleyDB/hints/solaris.pl BitKeeper/deleted/.del-typemap~783b8f5295b05f3d: Delete: bdb/perl.BerkeleyDB/typemap BitKeeper/deleted/.del-5.004_01~6081ce2fff7b0bc: Delete: bdb/perl.BerkeleyDB/patches/5.004_01 BitKeeper/deleted/.del-5.004_02~87214eac35ad9e6: Delete: bdb/perl.BerkeleyDB/patches/5.004_02 BitKeeper/deleted/.del-5.004_03~9a672becec7cb40f: Delete: bdb/perl.BerkeleyDB/patches/5.004_03 BitKeeper/deleted/.del-5.004_04~e326cb51af09d154: Delete: bdb/perl.BerkeleyDB/patches/5.004_04 BitKeeper/deleted/.del-5.004_05~7ab457a1e41a92fe: Delete: bdb/perl.BerkeleyDB/patches/5.004_05 BitKeeper/deleted/.del-5.005~f9e2d59b5964cd4b: Delete: bdb/perl.BerkeleyDB/patches/5.005 BitKeeper/deleted/.del-5.005_01~3eb9fb7b5842ea8e: Delete: bdb/perl.BerkeleyDB/patches/5.005_01 BitKeeper/deleted/.del-5.005_02~67477ce0bef717cb: Delete: bdb/perl.BerkeleyDB/patches/5.005_02 BitKeeper/deleted/.del-5.005_03~c4c29a1fb21e290a: Delete: bdb/perl.BerkeleyDB/patches/5.005_03 BitKeeper/deleted/.del-5.6.0~e1fb9897d124ee22: Delete: bdb/perl.BerkeleyDB/patches/5.6.0 BitKeeper/deleted/.del-btree.t~e4a1a3c675ddc406: Delete: bdb/perl.BerkeleyDB/t/btree.t BitKeeper/deleted/.del-db-3.0.t~d2c60991d84558f2: Delete: bdb/perl.BerkeleyDB/t/db-3.0.t BitKeeper/deleted/.del-db-3.1.t~6ee88cd13f55e018: Delete: bdb/perl.BerkeleyDB/t/db-3.1.t BitKeeper/deleted/.del-db-3.2.t~f73b6461f98fd1cf: Delete: bdb/perl.BerkeleyDB/t/db-3.2.t BitKeeper/deleted/.del-destroy.t~cc6a2ae1980a2ecd: Delete: bdb/perl.BerkeleyDB/t/destroy.t BitKeeper/deleted/.del-env.t~a8604a4499c4bd07: Delete: bdb/perl.BerkeleyDB/t/env.t BitKeeper/deleted/.del-examples.t~2571b77c3cc75574: Delete: bdb/perl.BerkeleyDB/t/examples.t BitKeeper/deleted/.del-examples.t.T~8228bdd75ac78b88: Delete: bdb/perl.BerkeleyDB/t/examples.t.T BitKeeper/deleted/.del-examples3.t.T~66a186897a87026d: Delete: bdb/perl.BerkeleyDB/t/examples3.t.T BitKeeper/deleted/.del-examples3.t~fe3822ba2f2d7f83: Delete: bdb/perl.BerkeleyDB/t/examples3.t BitKeeper/deleted/.del-filter.t~f87b045c1b708637: Delete: bdb/perl.BerkeleyDB/t/filter.t BitKeeper/deleted/.del-hash.t~616bfb4d644de3a3: Delete: bdb/perl.BerkeleyDB/t/hash.t BitKeeper/deleted/.del-join.t~29fc39f74a83ca22: Delete: bdb/perl.BerkeleyDB/t/join.t BitKeeper/deleted/.del-mldbm.t~31f5015341eea040: Delete: bdb/perl.BerkeleyDB/t/mldbm.t BitKeeper/deleted/.del-queue.t~8f338034ce44a641: Delete: bdb/perl.BerkeleyDB/t/queue.t BitKeeper/deleted/.del-recno.t~d4ddbd3743add63e: Delete: bdb/perl.BerkeleyDB/t/recno.t BitKeeper/deleted/.del-strict.t~6885cdd2ea71ca2d: Delete: bdb/perl.BerkeleyDB/t/strict.t BitKeeper/deleted/.del-subdb.t~aab62a5d5864c603: Delete: bdb/perl.BerkeleyDB/t/subdb.t BitKeeper/deleted/.del-txn.t~65033b8558ae1216: Delete: bdb/perl.BerkeleyDB/t/txn.t BitKeeper/deleted/.del-unknown.t~f3710458682665e1: Delete: bdb/perl.BerkeleyDB/t/unknown.t BitKeeper/deleted/.del-Changes~436f74a5c414c65b: Delete: bdb/perl.DB_File/Changes BitKeeper/deleted/.del-DB_File.pm~ae0951c6c7665a82: Delete: bdb/perl.DB_File/DB_File.pm BitKeeper/deleted/.del-DB_File.xs~89e49a0b5556f1d8: Delete: bdb/perl.DB_File/DB_File.xs BitKeeper/deleted/.del-DB_File_BS~290fad5dbbb87069: Delete: bdb/perl.DB_File/DB_File_BS BitKeeper/deleted/.del-MANIFEST~90ee581572bdd4ac: Delete: bdb/perl.DB_File/MANIFEST BitKeeper/deleted/.del-Makefile.PL~ac0567bb5a377e38: Delete: bdb/perl.DB_File/Makefile.PL BitKeeper/deleted/.del-README~77e924a5a9bae6b3: Delete: bdb/perl.DB_File/README BitKeeper/deleted/.del-config.in~ab4c2792b86a810b: Delete: bdb/perl.DB_File/config.in BitKeeper/deleted/.del-dbinfo~461c43b30fab2cb: Delete: bdb/perl.DB_File/dbinfo BitKeeper/deleted/.del-dynixptx.pl~50dcddfae25d17e9: Delete: bdb/perl.DB_File/hints/dynixptx.pl BitKeeper/deleted/.del-typemap~55cffb3288a9e587: Delete: bdb/perl.DB_File/typemap BitKeeper/deleted/.del-version.c~a4df0e646f8b3975: Delete: bdb/perl.DB_File/version.c BitKeeper/deleted/.del-5.004_01~d6830d0082702af7: Delete: bdb/perl.DB_File/patches/5.004_01 BitKeeper/deleted/.del-5.004_02~78b082dc80c91031: Delete: bdb/perl.DB_File/patches/5.004_02 BitKeeper/deleted/.del-5.004~4411ec2e3c9e008b: Delete: bdb/perl.DB_File/patches/5.004 BitKeeper/deleted/.del-sco.pl~1e795fe14fe4dcfe: Delete: bdb/perl.DB_File/hints/sco.pl BitKeeper/deleted/.del-5.004_03~33f274648b160d95: Delete: bdb/perl.DB_File/patches/5.004_03 BitKeeper/deleted/.del-5.004_04~8f3d1b3cf18bb20a: Delete: bdb/perl.DB_File/patches/5.004_04 BitKeeper/deleted/.del-5.004_05~9c0f02e7331e142: Delete: bdb/perl.DB_File/patches/5.004_05 BitKeeper/deleted/.del-5.005~c2108cb2e3c8d951: Delete: bdb/perl.DB_File/patches/5.005 BitKeeper/deleted/.del-5.005_01~3b45e9673afc4cfa: Delete: bdb/perl.DB_File/patches/5.005_01 BitKeeper/deleted/.del-5.005_02~9fe5766bb02a4522: Delete: bdb/perl.DB_File/patches/5.005_02 BitKeeper/deleted/.del-5.005_03~ffa1c38c19ae72ea: Delete: bdb/perl.DB_File/patches/5.005_03 BitKeeper/deleted/.del-5.6.0~373be3a5ce47be85: Delete: bdb/perl.DB_File/patches/5.6.0 BitKeeper/deleted/.del-db-btree.t~3231595a1c241eb3: Delete: bdb/perl.DB_File/t/db-btree.t BitKeeper/deleted/.del-db-hash.t~7c4ad0c795c7fad2: Delete: bdb/perl.DB_File/t/db-hash.t BitKeeper/deleted/.del-db-recno.t~6c2d3d80b9ba4a50: Delete: bdb/perl.DB_File/t/db-recno.t BitKeeper/deleted/.del-db_server.sed~cdb00ebcd48a64e2: Delete: bdb/rpc_server/db_server.sed BitKeeper/deleted/.del-db_server_proc.c~d46c8f409c3747f4: Delete: bdb/rpc_server/db_server_proc.c BitKeeper/deleted/.del-db_server_svc.sed~3f5e59f334fa4607: Delete: bdb/rpc_server/db_server_svc.sed BitKeeper/deleted/.del-db_server_util.c~a809f3a4629acda: Delete: bdb/rpc_server/db_server_util.c BitKeeper/deleted/.del-log.tcl~ff1b41f1355b97d7: Delete: bdb/test/log.tcl BitKeeper/deleted/.del-mpool.tcl~b0df4dc1b04db26c: Delete: bdb/test/mpool.tcl BitKeeper/deleted/.del-mutex.tcl~52fd5c73a150565: Delete: bdb/test/mutex.tcl BitKeeper/deleted/.del-txn.tcl~c4ff071550b5446e: Delete: bdb/test/txn.tcl BitKeeper/deleted/.del-README~e800a12a5392010a: Delete: bdb/test/upgrade/README BitKeeper/deleted/.del-pack-2.6.6.pl~89d5076d758d3e98: Delete: bdb/test/upgrade/generate-2.X/pack-2.6.6.pl BitKeeper/deleted/.del-test-2.6.patch~4a52dc83d447547b: Delete: bdb/test/upgrade/generate-2.X/test-2.6.patch
717 lines
16 KiB
C
717 lines
16 KiB
C
/*-
|
|
* See the file LICENSE for redistribution information.
|
|
*
|
|
* Copyright (c) 1999-2001
|
|
* Sleepycat Software. All rights reserved.
|
|
*/
|
|
|
|
#include "db_config.h"
|
|
|
|
#ifndef lint
|
|
static const char revid[] = "$Id: tcl_internal.c,v 11.54 2002/08/15 02:47:46 bostic Exp $";
|
|
#endif /* not lint */
|
|
|
|
#ifndef NO_SYSTEM_INCLUDES
|
|
#include <sys/types.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <tcl.h>
|
|
#endif
|
|
|
|
#include "db_int.h"
|
|
#include "dbinc/tcl_db.h"
|
|
#include "dbinc/db_page.h"
|
|
#include "dbinc/db_am.h"
|
|
#include "dbinc_auto/db_ext.h"
|
|
|
|
/*
|
|
*
|
|
* internal.c --
|
|
*
|
|
* This file contains internal functions we need to maintain
|
|
* state for our Tcl interface.
|
|
*
|
|
* NOTE: This all uses a linear linked list. If we end up with
|
|
* too many info structs such that this is a performance hit, it
|
|
* should be redone using hashes or a list per type. The assumption
|
|
* is that the user won't have more than a few dozen info structs
|
|
* in operation at any given point in time. Even a complicated
|
|
* application with a few environments, nested transactions, locking,
|
|
* and several databases open, using cursors should not have a
|
|
* negative performance impact, in terms of searching the list to
|
|
* get/manipulate the info structure.
|
|
*/
|
|
|
|
/*
|
|
* Prototypes for procedures defined later in this file:
|
|
*/
|
|
static void tcl_flag_callback __P((u_int32_t, const FN *, void *));
|
|
|
|
/*
|
|
* Private structure type used to pass both an interp and an object into
|
|
* a callback's single void *.
|
|
*/
|
|
struct __tcl_callback_bundle {
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *obj;
|
|
};
|
|
|
|
#define GLOB_CHAR(c) ((c) == '*' || (c) == '?')
|
|
|
|
/*
|
|
* PUBLIC: DBTCL_INFO *_NewInfo __P((Tcl_Interp *,
|
|
* PUBLIC: void *, char *, enum INFOTYPE));
|
|
*
|
|
* _NewInfo --
|
|
*
|
|
* This function will create a new info structure and fill it in
|
|
* with the name and pointer, id and type.
|
|
*/
|
|
DBTCL_INFO *
|
|
_NewInfo(interp, anyp, name, type)
|
|
Tcl_Interp *interp;
|
|
void *anyp;
|
|
char *name;
|
|
enum INFOTYPE type;
|
|
{
|
|
DBTCL_INFO *p;
|
|
int i, ret;
|
|
|
|
if ((ret = __os_malloc(NULL, sizeof(DBTCL_INFO), &p)) != 0) {
|
|
Tcl_SetResult(interp, db_strerror(ret), TCL_STATIC);
|
|
return (NULL);
|
|
}
|
|
|
|
if ((ret = __os_strdup(NULL, name, &p->i_name)) != 0) {
|
|
Tcl_SetResult(interp, db_strerror(ret), TCL_STATIC);
|
|
__os_free(NULL, p);
|
|
return (NULL);
|
|
}
|
|
p->i_interp = interp;
|
|
p->i_anyp = anyp;
|
|
p->i_data = 0;
|
|
p->i_data2 = 0;
|
|
p->i_type = type;
|
|
p->i_parent = NULL;
|
|
p->i_err = NULL;
|
|
p->i_errpfx = NULL;
|
|
p->i_lockobj.data = NULL;
|
|
p->i_btcompare = NULL;
|
|
p->i_dupcompare = NULL;
|
|
p->i_hashproc = NULL;
|
|
p->i_second_call = NULL;
|
|
p->i_rep_eid = NULL;
|
|
p->i_rep_send = NULL;
|
|
for (i = 0; i < MAX_ID; i++)
|
|
p->i_otherid[i] = 0;
|
|
|
|
LIST_INSERT_HEAD(&__db_infohead, p, entries);
|
|
return (p);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: void *_NameToPtr __P((CONST char *));
|
|
*/
|
|
void *
|
|
_NameToPtr(name)
|
|
CONST char *name;
|
|
{
|
|
DBTCL_INFO *p;
|
|
|
|
for (p = LIST_FIRST(&__db_infohead); p != NULL;
|
|
p = LIST_NEXT(p, entries))
|
|
if (strcmp(name, p->i_name) == 0)
|
|
return (p->i_anyp);
|
|
return (NULL);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: DBTCL_INFO *_PtrToInfo __P((CONST void *));
|
|
*/
|
|
DBTCL_INFO *
|
|
_PtrToInfo(ptr)
|
|
CONST void *ptr;
|
|
{
|
|
DBTCL_INFO *p;
|
|
|
|
for (p = LIST_FIRST(&__db_infohead); p != NULL;
|
|
p = LIST_NEXT(p, entries))
|
|
if (p->i_anyp == ptr)
|
|
return (p);
|
|
return (NULL);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: DBTCL_INFO *_NameToInfo __P((CONST char *));
|
|
*/
|
|
DBTCL_INFO *
|
|
_NameToInfo(name)
|
|
CONST char *name;
|
|
{
|
|
DBTCL_INFO *p;
|
|
|
|
for (p = LIST_FIRST(&__db_infohead); p != NULL;
|
|
p = LIST_NEXT(p, entries))
|
|
if (strcmp(name, p->i_name) == 0)
|
|
return (p);
|
|
return (NULL);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: void _SetInfoData __P((DBTCL_INFO *, void *));
|
|
*/
|
|
void
|
|
_SetInfoData(p, data)
|
|
DBTCL_INFO *p;
|
|
void *data;
|
|
{
|
|
if (p == NULL)
|
|
return;
|
|
p->i_anyp = data;
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: void _DeleteInfo __P((DBTCL_INFO *));
|
|
*/
|
|
void
|
|
_DeleteInfo(p)
|
|
DBTCL_INFO *p;
|
|
{
|
|
if (p == NULL)
|
|
return;
|
|
LIST_REMOVE(p, entries);
|
|
if (p->i_lockobj.data != NULL)
|
|
__os_free(NULL, p->i_lockobj.data);
|
|
if (p->i_err != NULL) {
|
|
fclose(p->i_err);
|
|
p->i_err = NULL;
|
|
}
|
|
if (p->i_errpfx != NULL)
|
|
__os_free(NULL, p->i_errpfx);
|
|
if (p->i_btcompare != NULL)
|
|
Tcl_DecrRefCount(p->i_btcompare);
|
|
if (p->i_dupcompare != NULL)
|
|
Tcl_DecrRefCount(p->i_dupcompare);
|
|
if (p->i_hashproc != NULL)
|
|
Tcl_DecrRefCount(p->i_hashproc);
|
|
if (p->i_second_call != NULL)
|
|
Tcl_DecrRefCount(p->i_second_call);
|
|
if (p->i_rep_eid != NULL)
|
|
Tcl_DecrRefCount(p->i_rep_eid);
|
|
if (p->i_rep_send != NULL)
|
|
Tcl_DecrRefCount(p->i_rep_send);
|
|
__os_free(NULL, p->i_name);
|
|
__os_free(NULL, p);
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: int _SetListElem __P((Tcl_Interp *,
|
|
* PUBLIC: Tcl_Obj *, void *, int, void *, int));
|
|
*/
|
|
int
|
|
_SetListElem(interp, list, elem1, e1cnt, elem2, e2cnt)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *list;
|
|
void *elem1, *elem2;
|
|
int e1cnt, e2cnt;
|
|
{
|
|
Tcl_Obj *myobjv[2], *thislist;
|
|
int myobjc;
|
|
|
|
myobjc = 2;
|
|
myobjv[0] = Tcl_NewByteArrayObj((u_char *)elem1, e1cnt);
|
|
myobjv[1] = Tcl_NewByteArrayObj((u_char *)elem2, e2cnt);
|
|
thislist = Tcl_NewListObj(myobjc, myobjv);
|
|
if (thislist == NULL)
|
|
return (TCL_ERROR);
|
|
return (Tcl_ListObjAppendElement(interp, list, thislist));
|
|
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: int _SetListElemInt __P((Tcl_Interp *, Tcl_Obj *, void *, int));
|
|
*/
|
|
int
|
|
_SetListElemInt(interp, list, elem1, elem2)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *list;
|
|
void *elem1;
|
|
int elem2;
|
|
{
|
|
Tcl_Obj *myobjv[2], *thislist;
|
|
int myobjc;
|
|
|
|
myobjc = 2;
|
|
myobjv[0] = Tcl_NewByteArrayObj((u_char *)elem1, strlen((char *)elem1));
|
|
myobjv[1] = Tcl_NewIntObj(elem2);
|
|
thislist = Tcl_NewListObj(myobjc, myobjv);
|
|
if (thislist == NULL)
|
|
return (TCL_ERROR);
|
|
return (Tcl_ListObjAppendElement(interp, list, thislist));
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: int _SetListRecnoElem __P((Tcl_Interp *, Tcl_Obj *,
|
|
* PUBLIC: db_recno_t, u_char *, int));
|
|
*/
|
|
int
|
|
_SetListRecnoElem(interp, list, elem1, elem2, e2size)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *list;
|
|
db_recno_t elem1;
|
|
u_char *elem2;
|
|
int e2size;
|
|
{
|
|
Tcl_Obj *myobjv[2], *thislist;
|
|
int myobjc;
|
|
|
|
myobjc = 2;
|
|
myobjv[0] = Tcl_NewLongObj((long)elem1);
|
|
myobjv[1] = Tcl_NewByteArrayObj(elem2, e2size);
|
|
thislist = Tcl_NewListObj(myobjc, myobjv);
|
|
if (thislist == NULL)
|
|
return (TCL_ERROR);
|
|
return (Tcl_ListObjAppendElement(interp, list, thislist));
|
|
|
|
}
|
|
|
|
/*
|
|
* _Set3DBTList --
|
|
* This is really analogous to both _SetListElem and
|
|
* _SetListRecnoElem--it's used for three-DBT lists returned by
|
|
* DB->pget and DBC->pget(). We'd need a family of four functions
|
|
* to handle all the recno/non-recno cases, however, so we make
|
|
* this a little more aware of the internals and do the logic inside.
|
|
*
|
|
* XXX
|
|
* One of these days all these functions should probably be cleaned up
|
|
* to eliminate redundancy and bring them into the standard DB
|
|
* function namespace.
|
|
*
|
|
* PUBLIC: int _Set3DBTList __P((Tcl_Interp *, Tcl_Obj *, DBT *, int,
|
|
* PUBLIC: DBT *, int, DBT *));
|
|
*/
|
|
int
|
|
_Set3DBTList(interp, list, elem1, is1recno, elem2, is2recno, elem3)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *list;
|
|
DBT *elem1, *elem2, *elem3;
|
|
int is1recno, is2recno;
|
|
{
|
|
|
|
Tcl_Obj *myobjv[3], *thislist;
|
|
|
|
if (is1recno)
|
|
myobjv[0] = Tcl_NewLongObj((long)*(db_recno_t *)elem1->data);
|
|
else
|
|
myobjv[0] =
|
|
Tcl_NewByteArrayObj((u_char *)elem1->data, elem1->size);
|
|
|
|
if (is2recno)
|
|
myobjv[1] = Tcl_NewLongObj((long)*(db_recno_t *)elem2->data);
|
|
else
|
|
myobjv[1] =
|
|
Tcl_NewByteArrayObj((u_char *)elem2->data, elem2->size);
|
|
|
|
myobjv[2] = Tcl_NewByteArrayObj((u_char *)elem3->data, elem3->size);
|
|
|
|
thislist = Tcl_NewListObj(3, myobjv);
|
|
|
|
if (thislist == NULL)
|
|
return (TCL_ERROR);
|
|
return (Tcl_ListObjAppendElement(interp, list, thislist));
|
|
}
|
|
|
|
/*
|
|
* _SetMultiList -- build a list for return from multiple get.
|
|
*
|
|
* PUBLIC: int _SetMultiList __P((Tcl_Interp *,
|
|
* PUBLIC: Tcl_Obj *, DBT *, DBT*, int, int));
|
|
*/
|
|
int
|
|
_SetMultiList(interp, list, key, data, type, flag)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *list;
|
|
DBT *key, *data;
|
|
int type, flag;
|
|
{
|
|
db_recno_t recno;
|
|
u_int32_t dlen, klen;
|
|
int result;
|
|
void *pointer, *dp, *kp;
|
|
|
|
recno = 0;
|
|
dlen = 0;
|
|
kp = NULL;
|
|
|
|
DB_MULTIPLE_INIT(pointer, data);
|
|
result = TCL_OK;
|
|
|
|
if (type == DB_RECNO || type == DB_QUEUE)
|
|
recno = *(db_recno_t *) key->data;
|
|
else
|
|
kp = key->data;
|
|
klen = key->size;
|
|
do {
|
|
if (flag & DB_MULTIPLE_KEY) {
|
|
if (type == DB_RECNO || type == DB_QUEUE)
|
|
DB_MULTIPLE_RECNO_NEXT(pointer,
|
|
data, recno, dp, dlen);
|
|
else
|
|
DB_MULTIPLE_KEY_NEXT(pointer,
|
|
data, kp, klen, dp, dlen);
|
|
} else
|
|
DB_MULTIPLE_NEXT(pointer, data, dp, dlen);
|
|
|
|
if (pointer == NULL)
|
|
break;
|
|
|
|
if (type == DB_RECNO || type == DB_QUEUE) {
|
|
result =
|
|
_SetListRecnoElem(interp, list, recno, dp, dlen);
|
|
recno++;
|
|
} else
|
|
result = _SetListElem(interp, list, kp, klen, dp, dlen);
|
|
} while (result == TCL_OK);
|
|
|
|
return (result);
|
|
}
|
|
/*
|
|
* PUBLIC: int _GetGlobPrefix __P((char *, char **));
|
|
*/
|
|
int
|
|
_GetGlobPrefix(pattern, prefix)
|
|
char *pattern;
|
|
char **prefix;
|
|
{
|
|
int i, j;
|
|
char *p;
|
|
|
|
/*
|
|
* Duplicate it, we get enough space and most of the work is done.
|
|
*/
|
|
if (__os_strdup(NULL, pattern, prefix) != 0)
|
|
return (1);
|
|
|
|
p = *prefix;
|
|
for (i = 0, j = 0; p[i] && !GLOB_CHAR(p[i]); i++, j++)
|
|
/*
|
|
* Check for an escaped character and adjust
|
|
*/
|
|
if (p[i] == '\\' && p[i+1]) {
|
|
p[j] = p[i+1];
|
|
i++;
|
|
} else
|
|
p[j] = p[i];
|
|
p[j] = 0;
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: int _ReturnSetup __P((Tcl_Interp *, int, int, char *));
|
|
*/
|
|
int
|
|
_ReturnSetup(interp, ret, ok, errmsg)
|
|
Tcl_Interp *interp;
|
|
int ret, ok;
|
|
char *errmsg;
|
|
{
|
|
char *msg;
|
|
|
|
if (ret > 0)
|
|
return (_ErrorSetup(interp, ret, errmsg));
|
|
|
|
/*
|
|
* We either have success or a DB error. If a DB error, set up the
|
|
* string. We return an error if not one of the errors we catch.
|
|
* If anyone wants to reset the result to return anything different,
|
|
* then the calling function is responsible for doing so via
|
|
* Tcl_ResetResult or another Tcl_SetObjResult.
|
|
*/
|
|
if (ret == 0) {
|
|
Tcl_SetResult(interp, "0", TCL_STATIC);
|
|
return (TCL_OK);
|
|
}
|
|
|
|
msg = db_strerror(ret);
|
|
Tcl_AppendResult(interp, msg, NULL);
|
|
|
|
if (ok)
|
|
return (TCL_OK);
|
|
else {
|
|
Tcl_SetErrorCode(interp, "BerkeleyDB", msg, NULL);
|
|
return (TCL_ERROR);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: int _ErrorSetup __P((Tcl_Interp *, int, char *));
|
|
*/
|
|
int
|
|
_ErrorSetup(interp, ret, errmsg)
|
|
Tcl_Interp *interp;
|
|
int ret;
|
|
char *errmsg;
|
|
{
|
|
Tcl_SetErrno(ret);
|
|
Tcl_AppendResult(interp, errmsg, ":", Tcl_PosixError(interp), NULL);
|
|
return (TCL_ERROR);
|
|
}
|
|
|
|
/*
|
|
* PUBLIC: void _ErrorFunc __P((CONST char *, char *));
|
|
*/
|
|
void
|
|
_ErrorFunc(pfx, msg)
|
|
CONST char *pfx;
|
|
char *msg;
|
|
{
|
|
DBTCL_INFO *p;
|
|
Tcl_Interp *interp;
|
|
int size;
|
|
char *err;
|
|
|
|
p = _NameToInfo(pfx);
|
|
if (p == NULL)
|
|
return;
|
|
interp = p->i_interp;
|
|
|
|
size = strlen(pfx) + strlen(msg) + 4;
|
|
/*
|
|
* If we cannot allocate enough to put together the prefix
|
|
* and message then give them just the message.
|
|
*/
|
|
if (__os_malloc(NULL, size, &err) != 0) {
|
|
Tcl_AddErrorInfo(interp, msg);
|
|
Tcl_AppendResult(interp, msg, "\n", NULL);
|
|
return;
|
|
}
|
|
snprintf(err, size, "%s: %s", pfx, msg);
|
|
Tcl_AddErrorInfo(interp, err);
|
|
Tcl_AppendResult(interp, err, "\n", NULL);
|
|
__os_free(NULL, err);
|
|
return;
|
|
}
|
|
|
|
#define INVALID_LSNMSG "Invalid LSN with %d parts. Should have 2.\n"
|
|
|
|
/*
|
|
* PUBLIC: int _GetLsn __P((Tcl_Interp *, Tcl_Obj *, DB_LSN *));
|
|
*/
|
|
int
|
|
_GetLsn(interp, obj, lsn)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *obj;
|
|
DB_LSN *lsn;
|
|
{
|
|
Tcl_Obj **myobjv;
|
|
char msg[MSG_SIZE];
|
|
int myobjc, result;
|
|
u_int32_t tmp;
|
|
|
|
result = Tcl_ListObjGetElements(interp, obj, &myobjc, &myobjv);
|
|
if (result == TCL_ERROR)
|
|
return (result);
|
|
if (myobjc != 2) {
|
|
result = TCL_ERROR;
|
|
snprintf(msg, MSG_SIZE, INVALID_LSNMSG, myobjc);
|
|
Tcl_SetResult(interp, msg, TCL_VOLATILE);
|
|
return (result);
|
|
}
|
|
result = _GetUInt32(interp, myobjv[0], &tmp);
|
|
if (result == TCL_ERROR)
|
|
return (result);
|
|
lsn->file = tmp;
|
|
result = _GetUInt32(interp, myobjv[1], &tmp);
|
|
lsn->offset = tmp;
|
|
return (result);
|
|
}
|
|
|
|
/*
|
|
* _GetUInt32 --
|
|
* Get a u_int32_t from a Tcl object. Tcl_GetIntFromObj does the
|
|
* right thing most of the time, but on machines where a long is 8 bytes
|
|
* and an int is 4 bytes, it errors on integers between the maximum
|
|
* int32_t and the maximum u_int32_t. This is correct, but we generally
|
|
* want a u_int32_t in the end anyway, so we use Tcl_GetLongFromObj and do
|
|
* the bounds checking ourselves.
|
|
*
|
|
* This code looks much like Tcl_GetIntFromObj, only with a different
|
|
* bounds check. It's essentially Tcl_GetUnsignedIntFromObj, which
|
|
* unfortunately doesn't exist.
|
|
*
|
|
* PUBLIC: int _GetUInt32 __P((Tcl_Interp *, Tcl_Obj *, u_int32_t *));
|
|
*/
|
|
int
|
|
_GetUInt32(interp, obj, resp)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *obj;
|
|
u_int32_t *resp;
|
|
{
|
|
int result;
|
|
long ltmp;
|
|
|
|
result = Tcl_GetLongFromObj(interp, obj, <mp);
|
|
if (result != TCL_OK)
|
|
return (result);
|
|
|
|
if ((unsigned long)ltmp != (u_int32_t)ltmp) {
|
|
if (interp != NULL) {
|
|
Tcl_ResetResult(interp);
|
|
Tcl_AppendToObj(Tcl_GetObjResult(interp),
|
|
"integer value too large for u_int32_t", -1);
|
|
}
|
|
return (TCL_ERROR);
|
|
}
|
|
|
|
*resp = (u_int32_t)ltmp;
|
|
return (TCL_OK);
|
|
}
|
|
|
|
/*
|
|
* tcl_flag_callback --
|
|
* Callback for db_pr.c functions that contain the FN struct mapping
|
|
* flag values to meaningful strings. This function appends a Tcl_Obj
|
|
* containing each pertinent flag string to the specified Tcl list.
|
|
*/
|
|
static void
|
|
tcl_flag_callback(flags, fn, vtcbp)
|
|
u_int32_t flags;
|
|
const FN *fn;
|
|
void *vtcbp;
|
|
{
|
|
const FN *fnp;
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *newobj, *listobj;
|
|
int result;
|
|
struct __tcl_callback_bundle *tcbp;
|
|
|
|
tcbp = (struct __tcl_callback_bundle *)vtcbp;
|
|
interp = tcbp->interp;
|
|
listobj = tcbp->obj;
|
|
|
|
for (fnp = fn; fnp->mask != 0; ++fnp)
|
|
if (LF_ISSET(fnp->mask)) {
|
|
newobj = Tcl_NewStringObj(fnp->name, strlen(fnp->name));
|
|
result =
|
|
Tcl_ListObjAppendElement(interp, listobj, newobj);
|
|
|
|
/*
|
|
* Tcl_ListObjAppendElement is defined to return TCL_OK
|
|
* unless listobj isn't actually a list (or convertible
|
|
* into one). If this is the case, we screwed up badly
|
|
* somehow.
|
|
*/
|
|
DB_ASSERT(result == TCL_OK);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* _GetFlagsList --
|
|
* Get a new Tcl object, containing a list of the string values
|
|
* associated with a particular set of flag values, given a function
|
|
* that can extract the right names for the right flags.
|
|
*
|
|
* PUBLIC: Tcl_Obj *_GetFlagsList __P((Tcl_Interp *, u_int32_t,
|
|
* PUBLIC: void (*)(u_int32_t, void *,
|
|
* PUBLIC: void (*)(u_int32_t, const FN *, void *))));
|
|
*/
|
|
Tcl_Obj *
|
|
_GetFlagsList(interp, flags, func)
|
|
Tcl_Interp *interp;
|
|
u_int32_t flags;
|
|
void (*func)
|
|
__P((u_int32_t, void *, void (*)(u_int32_t, const FN *, void *)));
|
|
{
|
|
Tcl_Obj *newlist;
|
|
struct __tcl_callback_bundle tcb;
|
|
|
|
newlist = Tcl_NewObj();
|
|
|
|
memset(&tcb, 0, sizeof(tcb));
|
|
tcb.interp = interp;
|
|
tcb.obj = newlist;
|
|
|
|
func(flags, &tcb, tcl_flag_callback);
|
|
|
|
return (newlist);
|
|
}
|
|
|
|
int __debug_stop, __debug_on, __debug_print, __debug_test;
|
|
|
|
/*
|
|
* PUBLIC: void _debug_check __P((void));
|
|
*/
|
|
void
|
|
_debug_check()
|
|
{
|
|
if (__debug_on == 0)
|
|
return;
|
|
|
|
if (__debug_print != 0) {
|
|
printf("\r%7d:", __debug_on);
|
|
fflush(stdout);
|
|
}
|
|
if (__debug_on++ == __debug_test || __debug_stop)
|
|
__db_loadme();
|
|
}
|
|
|
|
/*
|
|
* XXX
|
|
* Tcl 8.1+ Tcl_GetByteArrayFromObj/Tcl_GetIntFromObj bug.
|
|
*
|
|
* There is a bug in Tcl 8.1+ and byte arrays in that if it happens
|
|
* to use an object as both a byte array and something else like
|
|
* an int, and you've done a Tcl_GetByteArrayFromObj, then you
|
|
* do a Tcl_GetIntFromObj, your memory is deleted.
|
|
*
|
|
* Workaround is for all byte arrays we want to use, if it can be
|
|
* represented as an integer, we copy it so that we don't lose the
|
|
* memory.
|
|
*/
|
|
/*
|
|
* PUBLIC: int _CopyObjBytes __P((Tcl_Interp *, Tcl_Obj *obj, void **,
|
|
* PUBLIC: u_int32_t *, int *));
|
|
*/
|
|
int
|
|
_CopyObjBytes(interp, obj, newp, sizep, freep)
|
|
Tcl_Interp *interp;
|
|
Tcl_Obj *obj;
|
|
void **newp;
|
|
u_int32_t *sizep;
|
|
int *freep;
|
|
{
|
|
void *tmp, *new;
|
|
int i, len, ret;
|
|
|
|
/*
|
|
* If the object is not an int, then just return the byte
|
|
* array because it won't be transformed out from under us.
|
|
* If it is a number, we need to copy it.
|
|
*/
|
|
*freep = 0;
|
|
ret = Tcl_GetIntFromObj(interp, obj, &i);
|
|
tmp = Tcl_GetByteArrayFromObj(obj, &len);
|
|
*sizep = len;
|
|
if (ret == TCL_ERROR) {
|
|
Tcl_ResetResult(interp);
|
|
*newp = tmp;
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* If we get here, we have an integer that might be reused
|
|
* at some other point so we cannot count on GetByteArray
|
|
* keeping our pointer valid.
|
|
*/
|
|
if ((ret = __os_malloc(NULL, len, &new)) != 0)
|
|
return (ret);
|
|
memcpy(new, tmp, len);
|
|
*newp = new;
|
|
*freep = 1;
|
|
return (0);
|
|
}
|