mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 14:54:20 +01:00
920df11c62
git-svn-id: file:///svn/tokudb@2102 c7de825b-a66e-492c-adef-691d508d4ae1
181 lines
6.3 KiB
C++
181 lines
6.3 KiB
C++
#include <assert.h>
|
|
#include <db.h>
|
|
#include <errno.h>
|
|
#include <db_cxx.h>
|
|
|
|
#define do_maybe_error(errno)
|
|
|
|
Db::Db(DbEnv *env, u_int32_t flags)
|
|
: the_Env(env)
|
|
{
|
|
the_db = 0;
|
|
|
|
is_private_env = (the_Env == 0);
|
|
|
|
DB *tmp_db;
|
|
int ret = db_create(&tmp_db, the_Env->get_DB_ENV(), flags & ~(DB_CXX_NO_EXCEPTIONS));
|
|
if (ret!=0) {
|
|
the_Env->maybe_throw_error(ret);
|
|
// Otherwise cannot do much
|
|
return;
|
|
}
|
|
the_db = tmp_db;
|
|
tmp_db->api_internal = this;
|
|
if (is_private_env) {
|
|
the_Env = new DbEnv(tmp_db->dbenv, flags & DB_CXX_NO_EXCEPTIONS);
|
|
}
|
|
}
|
|
|
|
Db::~Db() {
|
|
if (the_db) {
|
|
close(0); // the user should have called close, but we do it here if not done.
|
|
assert(the_db==0);
|
|
}
|
|
if (is_private_env && the_Env) {
|
|
the_Env->close(0);
|
|
delete the_Env;
|
|
}
|
|
}
|
|
|
|
int Db::set_flags(u_int32_t flags) {
|
|
int ret = the_db->set_flags(the_db, flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::get_flags(u_int32_t *flags) {
|
|
int ret = the_db->get_flags(the_db, flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::close (u_int32_t flags) {
|
|
if (!the_db) {
|
|
return the_Env->maybe_throw_error(EINVAL);
|
|
}
|
|
the_db->api_internal = 0;
|
|
|
|
|
|
int ret = the_db->close(the_db, flags);
|
|
|
|
the_db = 0;
|
|
|
|
int no_exceptions = the_Env->do_no_exceptions; // Get this out before possibly deleting the env
|
|
|
|
if (is_private_env) {
|
|
// The env was closed by the_db->close, we have to tell the DbEnv that the DB_ENV is gone, and delete it.
|
|
the_Env->the_env = 0;
|
|
delete the_Env;
|
|
the_Env=0;
|
|
}
|
|
|
|
// Do we need to clean up "private environments"?
|
|
// What about cursors? They should be cleaned up already, but who did it?
|
|
|
|
// This maybe_throw must be the static one because the env is gone.
|
|
return DbEnv::maybe_throw_error(ret, NULL, no_exceptions);
|
|
}
|
|
|
|
int Db::open(DbTxn *txn, const char *filename, const char *subname, DBTYPE typ, u_int32_t flags, int mode) {
|
|
int ret = the_db->open(the_db, txn->get_DB_TXN(), filename, subname, typ, flags, mode);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::del(DbTxn *txn, Dbt *key, u_int32_t flags) {
|
|
int ret = the_db->del(the_db, txn->get_DB_TXN(), key->get_DBT(), flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::get(DbTxn *txn, Dbt *key, Dbt *data, u_int32_t flags) {
|
|
int ret = the_db->get(the_db, txn->get_DB_TXN(), key->get_DBT(), data->get_DBT(), flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::pget(DbTxn *txn, Dbt *key, Dbt *pkey, Dbt *data, u_int32_t flags) {
|
|
int ret = the_db->pget(the_db, txn->get_DB_TXN(), key->get_DBT(), pkey->get_DBT(), data->get_DBT(), flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::put(DbTxn *txn, Dbt *key, Dbt *data, u_int32_t flags) {
|
|
int ret = the_db->put(the_db, txn->get_DB_TXN(), key->get_DBT(), data->get_DBT(), flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::cursor(DbTxn *txn, Dbc **cursorp, u_int32_t flags) {
|
|
int ret = the_db->cursor(the_db, txn->get_DB_TXN(), (DBC**)cursorp, flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::set_pagesize(u_int32_t size) {
|
|
int ret = the_db->set_pagesize(the_db, size);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::remove(const char *file, const char *database, u_int32_t flags) {
|
|
int ret = the_db->remove(the_db, file, database, flags);
|
|
the_db = 0;
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
extern "C" int toku_bt_compare_callback_c(DB *db_c, const DBT *a, const DBT *b) {
|
|
Db *db_cxx=Db::get_Db(db_c);
|
|
return db_cxx->bt_compare_callback_cxx(db_cxx, Dbt::get_const_Dbt(a), Dbt::get_const_Dbt(b));
|
|
}
|
|
|
|
int Db::set_bt_compare(int (*bt_compare_callback)(Db *, const Dbt *, const Dbt *)) {
|
|
bt_compare_callback_cxx = bt_compare_callback;
|
|
int ret = the_db->set_bt_compare(the_db, toku_bt_compare_callback_c);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::set_bt_compare(bt_compare_fcn_type bt_compare_fcn) {
|
|
int ret = the_db->set_bt_compare(the_db, bt_compare_fcn);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::set_dup_compare(dup_compare_fcn_type dup_compare_fcn) {
|
|
int ret = the_db->set_dup_compare(the_db, dup_compare_fcn);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
int Db::fd(int *fdp) {
|
|
int ret = the_db->fd(the_db, fdp);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
extern "C" int toku_dup_compare_callback_c(DB *db_c, const DBT *a, const DBT *b) {
|
|
Db *db_cxx=Db::get_Db(db_c);
|
|
return db_cxx->dup_compare_callback_cxx(db_cxx, Dbt::get_const_Dbt(a), Dbt::get_const_Dbt(b));
|
|
}
|
|
|
|
int Db::set_dup_compare(int (*dup_compare_callback)(Db *, const Dbt *, const Dbt *)) {
|
|
dup_compare_callback_cxx = dup_compare_callback;
|
|
int ret = the_db->set_dup_compare(the_db, toku_dup_compare_callback_c);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
// Q: How to convert callback types in the most simple way?
|
|
// A: (Bradley) I see three ways to do this: The issue is that we have a C++ callback function, and we want to pass it to a C function.
|
|
// The fastest way is wrong: You cannot just pass the C++ function pointer since you cannot mix function pointers to C and C++.
|
|
// The "right" way is to declare an "extern C" function and do all the conversions. Create a Dbt from a DBT, and then call the C function. For returned data we would have do something too. But it turns out that DBT and Dbt pointers are interchangable so that leads to
|
|
// The "fast" way. Declare an "extern C" function, and then use Dbt::get_const_Dbt() to do the conversion quickly.
|
|
|
|
extern "C" int toku_associate_callback_c (DB*db_c, const DBT *k, const DBT *d, DBT *result) {
|
|
assert(db_c!=0);
|
|
Db *db_cxx=Db::get_Db(db_c);
|
|
assert(db_cxx);
|
|
return (*db_cxx->associate_callback_cxx)(db_cxx, Dbt::get_const_Dbt(k), Dbt::get_const_Dbt(d), Dbt::get_Dbt(result));
|
|
}
|
|
|
|
int Db::associate(DbTxn *txnid, Db *secondary, int (*callback)(Db *secondary, const Dbt *key, const Dbt *data, Dbt *result), u_int32_t flags) {
|
|
// secondary->set_associate_callback(callback);
|
|
secondary->associate_callback_cxx = callback;
|
|
int ret = the_db->associate(the_db, txnid->get_DB_TXN(), secondary->get_DB(), toku_associate_callback_c, flags);
|
|
return the_Env->maybe_throw_error(ret);
|
|
}
|
|
|
|
void Db::set_errpfx(const char *errpfx) {
|
|
the_Env->set_errpfx(errpfx);
|
|
}
|
|
|
|
void Db::set_error_stream(std::ostream *new_error_stream) {
|
|
the_Env->set_error_stream(new_error_stream);
|
|
}
|