summaryrefslogtreecommitdiff
path: root/plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h')
-rw-r--r--plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h3807
1 files changed, 0 insertions, 3807 deletions
diff --git a/plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h b/plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h
deleted file mode 100644
index 92a979e317..0000000000
--- a/plugins/Dbx_kyoto/src/kyotocabinet/kcplantdb.h
+++ /dev/null
@@ -1,3807 +0,0 @@
-/*************************************************************************************************
- * Plant database
- * Copyright (C) 2009-2012 FAL Labs
- * This file is part of Kyoto Cabinet.
- * This program is free software: you can redistribute it and/or modify it under the terms of
- * the GNU General Public License as published by the Free Software Foundation, either version
- * 3 of the License, or any later version.
- * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
- * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- * See the GNU General Public License for more details.
- * You should have received a copy of the GNU General Public License along with this program.
- * If not, see <http://www.gnu.org/licenses/>.
- *************************************************************************************************/
-
-
-#ifndef _KCPLANTDB_H // duplication check
-#define _KCPLANTDB_H
-
-#include <kccommon.h>
-#include <kcutil.h>
-#include <kcthread.h>
-#include <kcfile.h>
-#include <kccompress.h>
-#include <kccompare.h>
-#include <kcmap.h>
-#include <kcregex.h>
-#include <kcdb.h>
-
-#define KCPDBMETAKEY "@" ///< key of the record for meta data
-#define KCPDBTMPPATHEXT "tmpkct" ///< extension of the temporary file
-#define KCPDRECBUFSIZ 128 ///< size of the record buffer
-
-namespace kyotocabinet { // common namespace
-
-
-/**
- * Plant database.
- * @param BASEDB a class compatible with the file hash database class.
- * @param DBTYPE the database type number of the class.
- * @note This class template is a template for concrete classes to operate tree databases.
- * Template instance classes can be inherited but overwriting methods is forbidden. The class
- * TreeDB is the instance of the file tree database. The class ForestDB is the instance of the
- * directory tree database. Before every database operation, it is necessary to call the
- * BasicDB::open method in order to open a database file and connect the database object to it.
- * To avoid data missing or corruption, it is important to close every database file by the
- * BasicDB::close method when the database is no longer in use. It is forbidden for multible
- * database objects in a process to open the same database at the same time. It is forbidden to
- * share a database object with child processes.
- */
-template <class BASEDB, uint8_t DBTYPE>
-class PlantDB : public BasicDB {
- public:
- class Cursor;
- private:
- struct Record;
- struct RecordComparator;
- struct LeafNode;
- struct Link;
- struct LinkComparator;
- struct InnerNode;
- struct LeafSlot;
- struct InnerSlot;
- class ScopedVisitor;
- /** An alias of array of records. */
- typedef std::vector<Record*> RecordArray;
- /** An alias of array of records. */
- typedef std::vector<Link*> LinkArray;
- /** An alias of leaf node cache. */
- typedef LinkedHashMap<int64_t, LeafNode*> LeafCache;
- /** An alias of inner node cache. */
- typedef LinkedHashMap<int64_t, InnerNode*> InnerCache;
- /** An alias of list of cursors. */
- typedef std::list<Cursor*> CursorList;
- /** The number of cache slots. */
- static const int32_t SLOTNUM = 16;
- /** The default alignment power. */
- static const uint8_t DEFAPOW = 8;
- /** The default free block pool power. */
- static const uint8_t DEFFPOW = 10;
- /** The default bucket number. */
- static const int64_t DEFBNUM = 64LL << 10;
- /** The default page size. */
- static const int32_t DEFPSIZ = 8192;
- /** The default capacity size of the page cache. */
- static const int64_t DEFPCCAP = 64LL << 20;
- /** The size of the header. */
- static const int64_t HEADSIZ = 80;
- /** The offset of the numbers. */
- static const int64_t MOFFNUMS = 8;
- /** The prefix of leaf nodes. */
- static const char LNPREFIX = 'L';
- /** The prefix of inner nodes. */
- static const char INPREFIX = 'I';
- /** The average number of ways of each node. */
- static const size_t AVGWAY = 16;
- /** The ratio of the warm cache. */
- static const size_t WARMRATIO = 4;
- /** The ratio of flushing inner nodes. */
- static const size_t INFLRATIO = 32;
- /** The default number of items in each leaf node. */
- static const size_t DEFLINUM = 64;
- /** The default number of items in each inner node. */
- static const size_t DEFIINUM = 128;
- /** The base ID number for inner nodes. */
- static const int64_t INIDBASE = 1LL << 48;
- /** The minimum number of links in each inner node. */
- static const size_t INLINKMIN = 8;
- /** The maximum level of B+ tree. */
- static const int32_t LEVELMAX = 16;
- /** The number of cached nodes for auto transaction. */
- static const int32_t ATRANCNUM = 256;
- /** The threshold of busy loop and sleep for locking. */
- static const uint32_t LOCKBUSYLOOP = 8192;
- public:
- /**
- * Cursor to indicate a record.
- */
- class Cursor : public BasicDB::Cursor {
- friend class PlantDB;
- public:
- /**
- * Constructor.
- * @param db the container database object.
- */
- explicit Cursor(PlantDB* db) :
- db_(db), stack_(), kbuf_(NULL), ksiz_(0), lid_(0), back_(false) {
- _assert_(db);
- db_->curs_.push_back(this);
- }
- /**
- * Destructor.
- */
- virtual ~Cursor() {
- _assert_(true);
- if (!db_) return;
- if (kbuf_) clear_position();
- db_->curs_.remove(this);
- }
- /**
- * Accept a visitor to the current record.
- * @param visitor a visitor object.
- * @param writable true for writable operation, or false for read-only operation.
- * @param step true to move the cursor to the next record, or false for no move.
- * @return true on success, or false on failure.
- * @note The operation for each record is performed atomically and other threads accessing
- * the same record are blocked. To avoid deadlock, any explicit database operation must not
- * be performed in this function.
- */
- bool accept(Visitor* visitor, bool writable = true, bool step = false) {
- _assert_(visitor);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (writable && !(db_->writer_)) {
- db_->set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- if (!kbuf_) {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- return false;
- }
- bool err = false;
- bool hit = false;
-
-
- if (lid_ > 0 && !accept_spec(visitor, writable, step, &hit)) err = true;
-
-
- if (!err && !hit) {
- if (kbuf_) {
- bool retry = true;
- while (!err && retry) {
- if (!accept_atom(visitor, step, &retry)) err = true;
- }
- } else {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- err = true;
- }
- }
- return !err;
- }
- /**
- * Jump the cursor to the first record for forward scan.
- * @return true on success, or false on failure.
- */
- bool jump() {
- _assert_(true);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- back_ = false;
- if (kbuf_) clear_position();
- bool err = false;
- if (!set_position(db_->first_)) err = true;
- return !err;
- }
- /**
- * Jump the cursor to a record for forward scan.
- * @param kbuf the pointer to the key region.
- * @param ksiz the size of the key region.
- * @return true on success, or false on failure.
- */
- bool jump(const char* kbuf, size_t ksiz) {
- _assert_(kbuf && ksiz <= MEMMAXSIZ);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- back_ = false;
- if (kbuf_) clear_position();
- set_position(kbuf, ksiz, 0);
- bool err = false;
- if (!adjust_position()) {
- if (kbuf_) clear_position();
- err = true;
- }
- return !err;
- }
- /**
- * Jump the cursor to a record for forward scan.
- * @note Equal to the original Cursor::jump method except that the parameter is std::string.
- */
- bool jump(const std::string& key) {
- _assert_(true);
- return jump(key.c_str(), key.size());
- }
- /**
- * Jump the cursor to the last record for backward scan.
- * @return true on success, or false on failure.
- * @note This method is dedicated to tree databases. Some database types, especially hash
- * databases, may provide a dummy implementation.
- */
- bool jump_back() {
- _assert_(true);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- back_ = true;
- if (kbuf_) clear_position();
- bool err = false;
- if (!set_position_back(db_->last_)) err = true;
- return !err;
- }
- /**
- * Jump the cursor to a record for backward scan.
- * @param kbuf the pointer to the key region.
- * @param ksiz the size of the key region.
- * @return true on success, or false on failure.
- */
- bool jump_back(const char* kbuf, size_t ksiz) {
- _assert_(kbuf && ksiz <= MEMMAXSIZ);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- back_ = true;
- if (kbuf_) clear_position();
- set_position(kbuf, ksiz, 0);
- bool err = false;
- if (adjust_position()) {
- if (db_->reccomp_.comp->compare(kbuf, ksiz, kbuf_, ksiz_) < 0) {
- bool hit = false;
- if (lid_ > 0 && !back_position_spec(&hit)) err = true;
- if (!err && !hit) {
- if (kbuf_) {
- if (!back_position_atom()) err = true;
- } else {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- err = true;
- }
- }
- }
- } else {
- if (kbuf_) clear_position();
- if (!set_position_back(db_->last_)) err = true;
- }
- return !err;
- }
- /**
- * Jump the cursor to a record for backward scan.
- * @note Equal to the original Cursor::jump_back method except that the parameter is
- * std::string.
- */
- bool jump_back(const std::string& key) {
- _assert_(true);
- return jump_back(key.c_str(), key.size());
- }
- /**
- * Step the cursor to the next record.
- * @return true on success, or false on failure.
- */
- bool step() {
- _assert_(true);
- back_ = false;
- DB::Visitor visitor;
- if (!accept(&visitor, false, true)) return false;
- if (!kbuf_) {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- return false;
- }
- return true;
- }
- /**
- * Step the cursor to the previous record.
- * @return true on success, or false on failure.
- */
- bool step_back() {
- _assert_(true);
- if (db_->omode_ == 0) {
- db_->set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!kbuf_) {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- return false;
- }
- back_ = true;
- bool err = false;
- bool hit = false;
- if (lid_ > 0 && !back_position_spec(&hit)) err = true;
- if (!err && !hit) {
- if (kbuf_) {
- if (!back_position_atom()) err = true;
- } else {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- err = true;
- }
- }
- return !err;
- }
- /**
- * Get the database object.
- * @return the database object.
- */
- PlantDB* db() {
- _assert_(true);
- return db_;
- }
-
- /**
- * Clear the position.
- */
- void clear_position() {
- _assert_(true);
- if (kbuf_ != stack_) delete[] kbuf_;
- kbuf_ = NULL;
- lid_ = 0;
- }
- /**
- * Set the current position.
- * @param kbuf the pointer to the key region.
- * @param ksiz the size of the key region.
- * @param id the ID of the current node.
- */
- void set_position(const char* kbuf, size_t ksiz, int64_t id) {
- _assert_(kbuf);
- kbuf_ = ksiz > sizeof(stack_) ? new char[ksiz] : stack_;
- ksiz_ = ksiz;
- std::memcpy(kbuf_, kbuf, ksiz);
- lid_ = id;
- }
- /**
- * Set the current position with a record.
- * @param rec the current record.
- * @param id the ID of the current node.
- */
- void set_position(Record* rec, int64_t id) {
- _assert_(rec);
- char* dbuf = (char*)rec + sizeof(*rec);
- set_position(dbuf, rec->ksiz, id);
- }
- /**
- * Set the current position at the next node.
- * @param id the ID of the next node.
- * @return true on success, or false on failure.
- */
- bool set_position(int64_t id) {
- _assert_(true);
- while (id > 0) {
- LeafNode* node = db_->load_leaf_node(id, false);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "missing leaf node");
- db_->db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)id);
- return false;
- }
- RecordArray& recs = node->recs;
- if (!recs.empty()) {
- set_position(recs.front(), id);
- return true;
- } else {
- id = node->next;
- }
- }
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- return false;
- }
- /**
- * Set the current position at the previous node.
- * @param id the ID of the previous node.
- * @return true on success, or false on failure.
- */
- bool set_position_back(int64_t id) {
- _assert_(true);
- while (id > 0) {
- LeafNode* node = db_->load_leaf_node(id, false);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "missing leaf node");
- db_->db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)id);
- return false;
- }
- RecordArray& recs = node->recs;
- if (!recs.empty()) {
- set_position(recs.back(), id);
- return true;
- } else {
- id = node->prev;
- }
- }
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- return false;
- }
- /**
- * Accept a visitor to the current record speculatively.
- * @param visitor a visitor object.
- * @param writable true for writable operation, or false for read-only operation.
- * @param step true to move the cursor to the next record, or false for no move.
- * @param hitp the pointer to the variable for the hit flag.
- * @return true on success, or false on failure.
- */
- bool accept_spec(Visitor* visitor, bool writable, bool step, bool* hitp) {
- _assert_(visitor && hitp);
- bool err = false;
- bool hit = false;
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz_;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- LeafNode* node = db_->load_leaf_node(lid_, false);
- if (node) {
- char lstack[KCPDRECBUFSIZ];
- char* lbuf = NULL;
- size_t lsiz = 0;
- Link* link = NULL;
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- RecordArray& recs = node->recs;
- if (!recs.empty()) {
- Record* frec = recs.front();
- Record* lrec = recs.back();
- if (!db_->reccomp_(rec, frec) && !db_->reccomp_(lrec, rec)) {
- typename RecordArray::iterator ritend = recs.end();
- typename RecordArray::iterator rit = std::lower_bound(recs.begin(), ritend,
- rec, db_->reccomp_);
- if (rit != ritend) {
- hit = true;
- if (db_->reccomp_(rec, *rit)) {
- clear_position();
- set_position(*rit, node->id);
- if (rbuf != rstack) delete[] rbuf;
- rsiz = sizeof(Record) + ksiz_;
- rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- }
- rec = *rit;
- char* kbuf = (char*)rec + sizeof(*rec);
- size_t ksiz = rec->ksiz;
- size_t vsiz;
- const char* vbuf = visitor->visit_full(kbuf, ksiz, kbuf + ksiz,
- rec->vsiz, &vsiz);
- if (vbuf == Visitor::REMOVE) {
- rsiz = sizeof(*rec) + rec->ksiz + rec->vsiz;
- db_->count_ -= 1;
- db_->cusage_ -= rsiz;
- node->size -= rsiz;
- node->dirty = true;
- if (recs.size() <= 1) {
- lsiz = sizeof(Link) + ksiz;
- lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz;
- std::memcpy(lbuf + sizeof(*link), kbuf, ksiz);
- }
- xfree(rec);
- if (back_) {
- if (rit == recs.begin()) {
- step = true;
- } else {
- typename RecordArray::iterator ritprev = rit - 1;
- set_position(*ritprev, node->id);
- step = false;
- }
- } else {
- typename RecordArray::iterator ritnext = rit + 1;
- if (ritnext == ritend) {
- step = true;
- } else {
- clear_position();
- set_position(*ritnext, node->id);
- step = false;
- }
- }
- recs.erase(rit);
- } else if (vbuf != Visitor::NOP) {
- int64_t diff = (int64_t)vsiz - (int64_t)rec->vsiz;
- db_->cusage_ += diff;
- node->size += diff;
- node->dirty = true;
- if (vsiz > rec->vsiz) {
- *rit = (Record*)xrealloc(rec, sizeof(*rec) + rec->ksiz + vsiz);
- rec = *rit;
- kbuf = (char*)rec + sizeof(*rec);
- }
- std::memcpy(kbuf + rec->ksiz, vbuf, vsiz);
- rec->vsiz = vsiz;
- if (node->size > db_->psiz_ && recs.size() > 1) {
- lsiz = sizeof(Link) + ksiz;
- lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz;
- std::memcpy(lbuf + sizeof(*link), kbuf, ksiz);
- }
- }
- if (step) {
- if (back_) {
- if (rit != recs.begin()) {
- --rit;
- set_position(*rit, node->id);
- step = false;
- }
- } else {
- ++rit;
- if (rit != ritend) {
- clear_position();
- set_position(*rit, node->id);
- step = false;
- }
- }
- }
- }
- }
- }
- bool atran = db_->autotran_ && !db_->tran_ && node->dirty;
- bool async = db_->autosync_ && !db_->autotran_ && !db_->tran_ && node->dirty;
- if (hit && step) {
- clear_position();
- if (back_) {
- set_position_back(node->prev);
- } else {
- set_position(node->next);
- }
- }
- if (hit) {
- bool flush = db_->cusage_ > db_->pccap_;
- if (link || flush || async) {
- int64_t id = node->id;
- if (atran && !link && !db_->fix_auto_transaction_leaf(node)) err = true;
- if (link) {
- node = db_->search_tree(link, true, hist, &hnum);
- if (node) {
- if (!db_->reorganize_tree(node, hist, hnum)) err = true;
- if (atran && !db_->tran_ && !db_->fix_auto_transaction_tree()) err = true;
- } else {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- err = true;
- }
- } else if (flush) {
- int32_t idx = id % SLOTNUM;
- LeafSlot* lslot = db_->lslots_ + idx;
- if (!db_->flush_leaf_cache_part(lslot)) err = true;
- InnerSlot* islot = db_->islots_ + idx;
- if (islot->warm->count() > lslot->warm->count() + lslot->hot->count() + 1 &&
- !db_->flush_inner_cache_part(islot)) err = true;
- }
- if (async && !db_->fix_auto_synchronization()) err = true;
- } else {
- if (!db_->fix_auto_transaction_leaf(node)) err = true;
- }
- }
- if (lbuf != lstack) delete[] lbuf;
- }
- if (rbuf != rstack) delete[] rbuf;
- *hitp = hit;
- return !err;
- }
- /**
- * Accept a visitor to the current record atomically.
- * @param visitor a visitor object.
- * @param step true to move the cursor to the next record, or false for no move.
- * @param retryp the pointer to the variable for the retry flag.
- * @return true on success, or false on failure.
- */
- bool accept_atom(Visitor* visitor, bool step, bool *retryp) {
- _assert_(visitor && retryp);
- bool err = false;
- bool reorg = false;
- *retryp = false;
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + ksiz_;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz_;
- std::memcpy(lbuf + sizeof(*link), kbuf_, ksiz_);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- LeafNode* node = db_->search_tree(link, true, hist, &hnum);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- if (lbuf != lstack) delete[] lbuf;
- return false;
- }
- if (node->recs.empty()) {
- if (lbuf != lstack) delete[] lbuf;
- clear_position();
- if (!set_position(node->next)) return false;
- node = db_->load_leaf_node(lid_, false);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- return false;
- }
- lsiz = sizeof(Link) + ksiz_;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz_;
- std::memcpy(lbuf + sizeof(*link), kbuf_, ksiz_);
- node = db_->search_tree(link, true, hist, &hnum);
- if (node->id != lid_) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "invalid tree");
- if (lbuf != lstack) delete[] lbuf;
- return false;
- }
- }
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz_;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- RecordArray& recs = node->recs;
- typename RecordArray::iterator ritend = recs.end();
- typename RecordArray::iterator rit = std::lower_bound(recs.begin(), ritend,
- rec, db_->reccomp_);
- if (rit != ritend) {
- if (db_->reccomp_(rec, *rit)) {
- clear_position();
- set_position(*rit, node->id);
- if (rbuf != rstack) delete[] rbuf;
- rsiz = sizeof(Record) + ksiz_;
- rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- }
- rec = *rit;
- char* kbuf = (char*)rec + sizeof(*rec);
- size_t ksiz = rec->ksiz;
- size_t vsiz;
- const char* vbuf = visitor->visit_full(kbuf, ksiz, kbuf + ksiz,
- rec->vsiz, &vsiz);
- if (vbuf == Visitor::REMOVE) {
- rsiz = sizeof(*rec) + rec->ksiz + rec->vsiz;
- db_->count_ -= 1;
- db_->cusage_ -= rsiz;
- node->size -= rsiz;
- node->dirty = true;
- xfree(rec);
- step = false;
- clear_position();
- if (back_) {
- if (rit == recs.begin()) {
- set_position_back(node->prev);
- } else {
- typename RecordArray::iterator ritprev = rit - 1;
- set_position(*ritprev, node->id);
- }
- } else {
- typename RecordArray::iterator ritnext = rit + 1;
- if (ritnext == ritend) {
- set_position(node->next);
- } else {
- set_position(*ritnext, node->id);
- }
- }
- recs.erase(rit);
- if (recs.empty()) reorg = true;
- } else if (vbuf != Visitor::NOP) {
- int64_t diff = (int64_t)vsiz - (int64_t)rec->vsiz;
- db_->cusage_ += diff;
- node->size += diff;
- node->dirty = true;
- if (vsiz > rec->vsiz) {
- *rit = (Record*)xrealloc(rec, sizeof(*rec) + rec->ksiz + vsiz);
- rec = *rit;
- kbuf = (char*)rec + sizeof(*rec);
- }
- std::memcpy(kbuf + rec->ksiz, vbuf, vsiz);
- rec->vsiz = vsiz;
- if (node->size > db_->psiz_ && recs.size() > 1) reorg = true;
- }
- if (step) {
- clear_position();
- if (back_) {
- if (rit == recs.begin()) {
- set_position_back(node->prev);
- } else {
- --rit;
- set_position(*rit, node->id);
- }
- } else {
- ++rit;
- if (rit == ritend) {
- set_position(node->next);
- } else {
- set_position(*rit, node->id);
- }
- }
- }
- bool atran = db_->autotran_ && !db_->tran_ && node->dirty;
- bool async = db_->autosync_ && !db_->autotran_ && !db_->tran_ && node->dirty;
- if (atran && !reorg && !db_->fix_auto_transaction_leaf(node)) err = true;
- if (reorg) {
- if (!db_->reorganize_tree(node, hist, hnum)) err = true;
- if (atran && !db_->fix_auto_transaction_tree()) err = true;
- } else if (db_->cusage_ > db_->pccap_) {
- int32_t idx = node->id % SLOTNUM;
- LeafSlot* lslot = db_->lslots_ + idx;
- if (!db_->flush_leaf_cache_part(lslot)) err = true;
- InnerSlot* islot = db_->islots_ + idx;
- if (islot->warm->count() > lslot->warm->count() + lslot->hot->count() + 1 &&
- !db_->flush_inner_cache_part(islot)) err = true;
- }
- if (async && !db_->fix_auto_synchronization()) err = true;
- } else {
- int64_t lid = lid_;
- clear_position();
- if (back_) {
- if (set_position_back(node->prev)) {
- if (lid_ == lid) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "invalid leaf node");
- err = true;
- } else {
- *retryp = true;
- }
- } else {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- err = true;
- }
- } else {
- if (set_position(node->next)) {
- if (lid_ == lid) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "invalid leaf node");
- err = true;
- } else {
- *retryp = true;
- }
- } else {
- db_->set_error(_KCCODELINE_, Error::NOREC, "no record");
- err = true;
- }
- }
- }
- if (rbuf != rstack) delete[] rbuf;
- if (lbuf != lstack) delete[] lbuf;
- return !err;
- }
- /**
- * Adjust the position to an existing record.
- * @return true on success, or false on failure.
- */
- bool adjust_position() {
- _assert_(true);
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + ksiz_;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz_;
- std::memcpy(lbuf + sizeof(*link), kbuf_, ksiz_);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- LeafNode* node = db_->search_tree(link, true, hist, &hnum);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- if (lbuf != lstack) delete[] lbuf;
- return false;
- }
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz_;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- bool err = false;
- const RecordArray& recs = node->recs;
- typename RecordArray::const_iterator ritend = node->recs.end();
- typename RecordArray::const_iterator rit = std::lower_bound(recs.begin(), ritend,
- rec, db_->reccomp_);
- clear_position();
- if (rit == ritend) {
- if (!set_position(node->next)) err = true;
- } else {
- set_position(*rit, node->id);
- }
- if (rbuf != rstack) delete[] rbuf;
- if (lbuf != lstack) delete[] lbuf;
- return !err;
- }
- /**
- * Back the position to the previous record speculatively.
- * @param hitp the pointer to the variable for the hit flag.
- * @return true on success, or false on failure.
- */
- bool back_position_spec(bool* hitp) {
- _assert_(hitp);
- bool err = false;
- bool hit = false;
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz_;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- LeafNode* node = db_->load_leaf_node(lid_, false);
- if (node) {
- RecordArray& recs = node->recs;
- if (!recs.empty()) {
- Record* frec = recs.front();
- Record* lrec = recs.back();
- if (db_->reccomp_(rec, frec)) {
- hit = true;
- clear_position();
- if (!set_position_back(node->prev)) err = true;
- } else if (db_->reccomp_(lrec, rec)) {
- } else {
- hit = true;
- typename RecordArray::iterator ritbeg = recs.begin();
- typename RecordArray::iterator ritend = recs.end();
- typename RecordArray::iterator rit = std::lower_bound(recs.begin(), ritend,
- rec, db_->reccomp_);
- clear_position();
- if (rit == ritbeg) {
- if (!set_position_back(node->prev)) err = true;
- } else {
- --rit;
- set_position(*rit, node->id);
- }
- }
- }
- }
- if (rbuf != rstack) delete[] rbuf;
- *hitp = hit;
- return !err;
- }
- /**
- * Back the position to the previous record atomically.
- * @return true on success, or false on failure.
- */
- bool back_position_atom() {
- _assert_(true);
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + ksiz_;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz_;
- std::memcpy(lbuf + sizeof(*link), kbuf_, ksiz_);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- LeafNode* node = db_->search_tree(link, true, hist, &hnum);
- if (!node) {
- db_->set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- if (lbuf != lstack) delete[] lbuf;
- return false;
- }
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz_;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz_;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf_, ksiz_);
- bool err = false;
- const RecordArray& recs = node->recs;
- typename RecordArray::const_iterator ritbeg = node->recs.begin();
- typename RecordArray::const_iterator ritend = node->recs.end();
- typename RecordArray::const_iterator rit = std::lower_bound(recs.begin(), ritend,
- rec, db_->reccomp_);
- clear_position();
- if (rit == ritbeg) {
- if (!set_position_back(node->prev)) err = true;
- } else if (rit == ritend) {
- ritend--;
- set_position(*ritend, node->id);
- } else {
- --rit;
- set_position(*rit, node->id);
- }
- if (rbuf != rstack) delete[] rbuf;
- if (lbuf != lstack) delete[] lbuf;
- return !err;
- }
- /** Dummy constructor to forbid the use. */
- Cursor(const Cursor&);
- /** Dummy Operator to forbid the use. */
- Cursor& operator =(const Cursor&);
- /** The inner database. */
- PlantDB* db_;
- /** The stack buffer for the key. */
- char stack_[KCPDRECBUFSIZ];
- /** The pointer to the key region. */
- char* kbuf_;
- /** The size of the key region. */
- size_t ksiz_;
- /** The last visited leaf. */
- int64_t lid_;
- /** The backward flag. */
- bool back_;
- };
- /**
- * Tuning options.
- */
- enum Option {
- TSMALL = BASEDB::TSMALL, ///< use 32-bit addressing
- TLINEAR = BASEDB::TLINEAR, ///< use linear collision chaining
- TCOMPRESS = BASEDB::TCOMPRESS ///< compress each record
- };
- /**
- * Status flags.
- */
- enum Flag {
- FOPEN = BASEDB::FOPEN, ///< whether opened
- FFATAL = BASEDB::FFATAL ///< whether with fatal error
- };
- /**
- * Default constructor.
- */
- explicit PlantDB() :
- mtrigger_(NULL), omode_(0), writer_(false), autotran_(false), autosync_(false),
- db_(), curs_(), apow_(DEFAPOW), fpow_(DEFFPOW), opts_(0), bnum_(DEFBNUM),
- psiz_(DEFPSIZ), pccap_(DEFPCCAP),
- root_(0), first_(0), last_(0), lcnt_(0), icnt_(0), count_(0), cusage_(0),
- lslots_(), islots_(), reccomp_(), linkcomp_(),
- tran_(false), trclock_(0), trlcnt_(0), trcount_(0) {
- _assert_(true);
- }
- /**
- * Destructor.
- * @note If the database is not closed, it is closed implicitly.
- */
- virtual ~PlantDB() {
- _assert_(true);
- if (omode_ != 0) close();
- if (!curs_.empty()) {
- typename CursorList::const_iterator cit = curs_.begin();
- typename CursorList::const_iterator citend = curs_.end();
- while (cit != citend) {
- Cursor* cur = *cit;
- cur->db_ = NULL;
- ++cit;
- }
- }
- }
- /**
- * Accept a visitor to a record.
- * @param kbuf the pointer to the key region.
- * @param ksiz the size of the key region.
- * @param visitor a visitor object.
- * @param writable true for writable operation, or false for read-only operation.
- * @return true on success, or false on failure.
- * @note The operation for each record is performed atomically and other threads accessing the
- * same record are blocked. To avoid deadlock, any explicit database operation must not be
- * performed in this function.
- */
- bool accept(const char* kbuf, size_t ksiz, Visitor* visitor, bool writable = true) {
- _assert_(kbuf && ksiz <= MEMMAXSIZ && visitor);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (writable && !writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + ksiz;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz;
- std::memcpy(lbuf + sizeof(*link), kbuf, ksiz);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- LeafNode* node = search_tree(link, true, hist, &hnum);
- if (!node) {
- set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- if (lbuf != lstack) delete[] lbuf;
- return false;
- }
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf, ksiz);
-
- bool reorg = accept_impl(node, rec, visitor);
- bool atran = autotran_ && !tran_ && node->dirty;
- bool async = autosync_ && !autotran_ && !tran_ && node->dirty;
-
- bool flush = false;
- bool err = false;
- int64_t id = node->id;
- if (atran && !reorg && !fix_auto_transaction_leaf(node)) err = true;
- if (cusage_ > pccap_) {
- int32_t idx = id % SLOTNUM;
- LeafSlot* lslot = lslots_ + idx;
- if (!clean_leaf_cache_part(lslot)) err = true;
- flush = true;
- }
- if (reorg) {
- node = search_tree(link, false, hist, &hnum);
- if (node) {
- if (!reorganize_tree(node, hist, hnum)) err = true;
- if (atran && !tran_ && !fix_auto_transaction_tree()) err = true;
- }
- } else if (flush) {
- int32_t idx = id % SLOTNUM;
- LeafSlot* lslot = lslots_ + idx;
- if (!flush_leaf_cache_part(lslot)) err = true;
- InnerSlot* islot = islots_ + idx;
- if (islot->warm->count() > lslot->warm->count() + lslot->hot->count() + 1 &&
- !flush_inner_cache_part(islot)) err = true;
- }
- if (rbuf != rstack) delete[] rbuf;
- if (lbuf != lstack) delete[] lbuf;
- if (async) {
- if (!fix_auto_synchronization()) err = true;
- }
- return !err;
- }
- /**
- * Accept a visitor to multiple records at once.
- * @param keys specifies a string vector of the keys.
- * @param visitor a visitor object.
- * @param writable true for writable operation, or false for read-only operation.
- * @return true on success, or false on failure.
- * @note The operations for specified records are performed atomically and other threads
- * accessing the same records are blocked. To avoid deadlock, any explicit database operation
- * must not be performed in this function.
- */
- bool accept_bulk(const std::vector<std::string>& keys, Visitor* visitor,
- bool writable = true) {
- _assert_(visitor);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (writable && !writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- ScopedVisitor svis(visitor);
- if (keys.empty()) return true;
- bool err = false;
- std::vector<std::string>::const_iterator kit = keys.begin();
- std::vector<std::string>::const_iterator kitend = keys.end();
- while (!err && kit != kitend) {
- const char* kbuf = kit->data();
- size_t ksiz = kit->size();
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + ksiz;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = ksiz;
- std::memcpy(lbuf + sizeof(*link), kbuf, ksiz);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- LeafNode* node = search_tree(link, true, hist, &hnum);
- if (!node) {
- set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- if (lbuf != lstack) delete[] lbuf;
- err = true;
- break;
- }
- char rstack[KCPDRECBUFSIZ];
- size_t rsiz = sizeof(Record) + ksiz;
- char* rbuf = rsiz > sizeof(rstack) ? new char[rsiz] : rstack;
- Record* rec = (Record*)rbuf;
- rec->ksiz = ksiz;
- rec->vsiz = 0;
- std::memcpy(rbuf + sizeof(*rec), kbuf, ksiz);
- bool reorg = accept_impl(node, rec, visitor);
- bool atran = autotran_ && !tran_ && node->dirty;
- bool async = autosync_ && !autotran_ && !tran_ && node->dirty;
- if (atran && !reorg && !fix_auto_transaction_leaf(node)) err = true;
- if (reorg) {
- if (!reorganize_tree(node, hist, hnum)) err = true;
- if (atran && !fix_auto_transaction_tree()) err = true;
- } else if (cusage_ > pccap_) {
- int32_t idx = node->id % SLOTNUM;
- LeafSlot* lslot = lslots_ + idx;
- if (!flush_leaf_cache_part(lslot)) err = true;
- InnerSlot* islot = islots_ + idx;
- if (islot->warm->count() > lslot->warm->count() + lslot->hot->count() + 1 &&
- !flush_inner_cache_part(islot)) err = true;
- }
- if (rbuf != rstack) delete[] rbuf;
- if (lbuf != lstack) delete[] lbuf;
- if (async && !fix_auto_synchronization()) err = true;
- ++kit;
- }
- return !err;
- }
- /**
- * Iterate to accept a visitor for each record.
- * @param visitor a visitor object.
- * @param writable true for writable operation, or false for read-only operation.
- * @param checker a progress checker object. If it is NULL, no checking is performed.
- * @return true on success, or false on failure.
- * @note The whole iteration is performed atomically and other threads are blocked. To avoid
- * deadlock, any explicit database operation must not be performed in this function.
- */
- bool iterate(Visitor *visitor, bool writable = true, ProgressChecker* checker = NULL) {
- _assert_(visitor);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (writable && !writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- ScopedVisitor svis(visitor);
- int64_t allcnt = count_;
- if (checker && !checker->check("iterate", "beginning", 0, allcnt)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- bool err = false;
- bool atran = false;
- if (autotran_ && writable && !tran_) {
- if (begin_transaction_impl(autosync_)) {
- atran = true;
- } else {
- err = true;
- }
- }
- int64_t id = first_;
- int64_t flcnt = 0;
- int64_t curcnt = 0;
- while (!err && id > 0) {
- LeafNode* node = load_leaf_node(id, false);
- if (!node) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing leaf node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)id);
- return false;
- }
- id = node->next;
- const RecordArray& recs = node->recs;
- RecordArray keys;
- keys.reserve(recs.size());
- typename RecordArray::const_iterator rit = recs.begin();
- typename RecordArray::const_iterator ritend = recs.end();
- while (rit != ritend) {
- Record* rec = *rit;
- size_t rsiz = sizeof(*rec) + rec->ksiz;
- char* dbuf = (char*)rec + sizeof(*rec);
- Record* key = (Record*)xmalloc(rsiz);
- key->ksiz = rec->ksiz;
- key->vsiz = 0;
- char* kbuf = (char*)key + sizeof(*key);
- std::memcpy(kbuf, dbuf, rec->ksiz);
- keys.push_back(key);
- ++rit;
- }
- typename RecordArray::const_iterator kit = keys.begin();
- typename RecordArray::const_iterator kitend = keys.end();
- bool reorg = false;
- while (kit != kitend) {
- Record* rec = *kit;
- if (accept_impl(node, rec, visitor)) reorg = true;
- curcnt++;
- if (checker && !checker->check("iterate", "processing", curcnt, allcnt)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- err = true;
- break;
- }
- ++kit;
- }
- if (reorg) {
- Record* rec = keys.front();
- char* dbuf = (char*)rec + sizeof(*rec);
- char lstack[KCPDRECBUFSIZ];
- size_t lsiz = sizeof(Link) + rec->ksiz;
- char* lbuf = lsiz > sizeof(lstack) ? new char[lsiz] : lstack;
- Link* link = (Link*)lbuf;
- link->child = 0;
- link->ksiz = rec->ksiz;
- std::memcpy(lbuf + sizeof(*link), dbuf, rec->ksiz);
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- node = search_tree(link, false, hist, &hnum);
- if (node) {
- if (!reorganize_tree(node, hist, hnum)) err = true;
- } else {
- set_error(_KCCODELINE_, Error::BROKEN, "search failed");
- err = true;
- }
- if (lbuf != lstack) delete[] lbuf;
- }
- if (cusage_ > pccap_) {
- for (int32_t i = 0; i < SLOTNUM; i++) {
- LeafSlot* lslot = lslots_ + i;
- if (!flush_leaf_cache_part(lslot)) err = true;
- }
- InnerSlot* islot = islots_ + (flcnt++) % SLOTNUM;
- if (islot->warm->count() > 2 && !flush_inner_cache_part(islot)) err = true;
- }
- kit = keys.begin();
- while (kit != kitend) {
- xfree(*kit);
- ++kit;
- }
- }
- if (checker && !checker->check("iterate", "ending", -1, allcnt)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- err = true;
- }
- if (atran && !commit_transaction()) err = true;
- if (autosync_ && !autotran_ && writable && !fix_auto_synchronization()) err = true;
- trigger_meta(MetaTrigger::ITERATE, "iterate");
- return !err;
- }
- /**
- * Scan each record in parallel.
- * @param visitor a visitor object.
- * @param thnum the number of worker threads.
- * @param checker a progress checker object. If it is NULL, no checking is performed.
- * @return true on success, or false on failure.
- * @note This function is for reading records and not for updating ones. The return value of
- * the visitor is just ignored. To avoid deadlock, any explicit database operation must not
- * be performed in this function.
- */
- bool scan_parallel(Visitor *visitor, size_t thnum, ProgressChecker* checker = NULL) {
- _assert_(visitor && thnum <= MEMMAXSIZ);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (thnum < 1) thnum = 0;
- if (thnum > (size_t)INT8MAX) thnum = INT8MAX;
- bool err = false;
- if (writer_) {
- if (checker && !checker->check("scan_parallel", "cleaning the leaf node cache", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!clean_leaf_cache()) err = true;
- }
- ScopedVisitor svis(visitor);
- int64_t allcnt = count_;
- if (checker && !checker->check("scan_parallel", "beginning", 0, allcnt)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- class ProgressCheckerImpl : public ProgressChecker {
- public:
- explicit ProgressCheckerImpl() : ok_(1) {}
- void stop() {
- ok_.set(0);
- }
- private:
- bool check(const char* name, const char* message, int64_t curcnt, int64_t allcnt) {
- return ok_ > 0;
- }
- AtomicInt64 ok_;
- };
- ProgressCheckerImpl ichecker;
- class VisitorImpl : public Visitor {
- public:
- explicit VisitorImpl(PlantDB* db, Visitor* visitor,
- ProgressChecker* checker, int64_t allcnt,
- ProgressCheckerImpl* ichecker) :
- db_(db), visitor_(visitor), checker_(checker), allcnt_(allcnt),
- ichecker_(ichecker), error_() {}
- const Error& error() {
- return error_;
- }
- private:
- const char* visit_full(const char* kbuf, size_t ksiz,
- const char* vbuf, size_t vsiz, size_t* sp) {
- if (ksiz < 2 || ksiz >= NUMBUFSIZ || kbuf[0] != LNPREFIX) return NOP;
- uint64_t prev;
- size_t step = readvarnum(vbuf, vsiz, &prev);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- uint64_t next;
- step = readvarnum(vbuf, vsiz, &next);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- while (vsiz > 1) {
- uint64_t rksiz;
- step = readvarnum(vbuf, vsiz, &rksiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- uint64_t rvsiz;
- step = readvarnum(vbuf, vsiz, &rvsiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- if (vsiz < rksiz + rvsiz) break;
- size_t xvsiz;
- visitor_->visit_full(vbuf, rksiz, vbuf + rksiz, rvsiz, &xvsiz);
- vbuf += rksiz;
- vsiz -= rksiz;
- vbuf += rvsiz;
- vsiz -= rvsiz;
- if (checker_ && !checker_->check("scan_parallel", "processing", -1, allcnt_)) {
- db_->set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- error_ = db_->error();
- ichecker_->stop();
- break;
- }
- }
- return NOP;
- }
- PlantDB* db_;
- Visitor* visitor_;
- ProgressChecker* checker_;
- int64_t allcnt_;
- ProgressCheckerImpl* ichecker_;
- Error error_;
- };
- VisitorImpl ivisitor(this, visitor, checker, allcnt, &ichecker);
- if (!db_.scan_parallel(&ivisitor, thnum, &ichecker)) err = true;
- if (ivisitor.error() != Error::SUCCESS) {
- const Error& e = ivisitor.error();
- db_.set_error(_KCCODELINE_, e.code(), e.message());
- err = true;
- }
- if (checker && !checker->check("scan_parallel", "ending", -1, allcnt)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- err = true;
- }
- trigger_meta(MetaTrigger::ITERATE, "scan_parallel");
- return !err;
- }
- /**
- * Get the last happened error.
- * @return the last happened error.
- */
- Error error() const {
- _assert_(true);
- return db_.error();
- }
- /**
- * Set the error information.
- * @param file the file name of the program source code.
- * @param line the line number of the program source code.
- * @param func the function name of the program source code.
- * @param code an error code.
- * @param message a supplement message.
- */
- void set_error(const char* file, int32_t line, const char* func,
- Error::Code code, const char* message) {
- _assert_(file && line > 0 && func && message);
- db_.set_error(file, line, func, code, message);
- }
- /**
- * Open a database file.
- * @param path the path of a database file.
- * @param mode the connection mode. BasicDB::OWRITER as a writer, BasicDB::OREADER as a
- * reader. The following may be added to the writer mode by bitwise-or: BasicDB::OCREATE,
- * which means it creates a new database if the file does not exist, BasicDB::OTRUNCATE, which
- * means it creates a new database regardless if the file exists, BasicDB::OAUTOTRAN, which
- * means each updating operation is performed in implicit transaction, BasicDB::OAUTOSYNC,
- * which means each updating operation is followed by implicit synchronization with the file
- * system. The following may be added to both of the reader mode and the writer mode by
- * bitwise-or: BasicDB::ONOLOCK, which means it opens the database file without file locking,
- * BasicDB::OTRYLOCK, which means locking is performed without blocking, BasicDB::ONOREPAIR,
- * which means the database file is not repaired implicitly even if file destruction is
- * detected.
- * @return true on success, or false on failure.
- * @note Every opened database must be closed by the BasicDB::close method when it is no
- * longer in use. It is not allowed for two or more database objects in the same process to
- * keep their connections to the same database file at the same time.
- */
- bool open(const std::string& path, uint32_t mode = OWRITER | OCREATE) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- report(_KCCODELINE_, Logger::DEBUG, "opening the database (path=%s)", path.c_str());
- if (DBTYPE == TYPEGRASS) {
- mode &= ~OREADER;
- mode |= OWRITER | OCREATE;
- }
- writer_ = false;
- autotran_ = false;
- autosync_ = false;
- if (mode & OWRITER) {
- writer_ = true;
- if (mode & OAUTOTRAN) autotran_ = true;
- if (mode & OAUTOSYNC) autosync_ = true;
- }
- if (!db_.tune_type(DBTYPE)) return false;
- if (!db_.tune_alignment(apow_)) return false;
- if (!db_.tune_fbp(fpow_)) return false;
- if (!db_.tune_options(opts_)) return false;
- if (!db_.tune_buckets(bnum_)) return false;
- if (!db_.open(path, mode)) return false;
- if (db_.type() != DBTYPE) {
- set_error(_KCCODELINE_, Error::INVALID, "invalid database type");
- db_.close();
- return false;
- }
- if (db_.reorganized()) {
- if (!reorganize_file(mode)) return false;
- } else if (db_.recovered()) {
- if (!writer_) {
- if (!db_.close()) return false;
- uint32_t tmode = (mode & ~OREADER) | OWRITER;
- if (!db_.open(path, tmode)) return false;
- }
- if (!recalc_count()) return false;
- if (!writer_) {
- if (!db_.close()) return false;
- if (!db_.open(path, mode)) return false;
- }
- if (count_ == INT64MAX && !reorganize_file(mode)) return false;
- }
- if (writer_ && db_.count() < 1) {
- root_ = 0;
- first_ = 0;
- last_ = 0;
- count_ = 0;
- create_leaf_cache();
- create_inner_cache();
- lcnt_ = 0;
- create_leaf_node(0, 0);
- root_ = 1;
- first_ = 1;
- last_ = 1;
- lcnt_ = 1;
- icnt_ = 0;
- count_ = 0;
- if (!reccomp_.comp) reccomp_.comp = LEXICALCOMP;
- if (!dump_meta() || !flush_leaf_cache(true) || !load_meta()) {
- delete_inner_cache();
- delete_leaf_cache();
- db_.close();
- return false;
- }
- } else {
- if (!load_meta()) {
- db_.close();
- return false;
- }
- create_leaf_cache();
- create_inner_cache();
- }
- if (psiz_ < 1 || root_ < 1 || first_ < 1 || last_ < 1 ||
- lcnt_ < 1 || icnt_ < 0 || count_ < 0 || bnum_ < 1) {
- set_error(_KCCODELINE_, Error::BROKEN, "invalid meta data");
- db_.report(_KCCODELINE_, Logger::WARN, "psiz=%lld root=%lld first=%lld last=%lld"
- " lcnt=%lld icnt=%lld count=%lld bnum=%lld",
- (long long)psiz_, (long long)root_, (long long)first_, (long long)last_,
- (long long)lcnt_, (long long)icnt_, (long long)count_, (long long)bnum_);
- delete_inner_cache();
- delete_leaf_cache();
- db_.close();
- return false;
- }
- omode_ = mode;
- cusage_ = 0;
- tran_ = false;
- trclock_ = 0;
- trigger_meta(MetaTrigger::OPEN, "open");
- return true;
- }
- /**
- * Close the database file.
- * @return true on success, or false on failure.
- */
- bool close() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- const std::string& path = db_.path();
- report(_KCCODELINE_, Logger::DEBUG, "closing the database (path=%s)", path.c_str());
- bool err = false;
- disable_cursors();
- int64_t lsiz = calc_leaf_cache_size();
- int64_t isiz = calc_inner_cache_size();
- if (cusage_ != lsiz + isiz) {
- set_error(_KCCODELINE_, Error::BROKEN, "invalid cache usage");
- db_.report(_KCCODELINE_, Logger::WARN, "cusage=%lld lsiz=%lld isiz=%lld",
- (long long)cusage_, (long long)lsiz, (long long)isiz);
- err = true;
- }
- if (!flush_leaf_cache(true)) err = true;
- if (!flush_inner_cache(true)) err = true;
- lsiz = calc_leaf_cache_size();
- isiz = calc_inner_cache_size();
- int64_t lcnt = calc_leaf_cache_count();
- int64_t icnt = calc_inner_cache_count();
- if (cusage_ != 0 || lsiz != 0 || isiz != 0 || lcnt != 0 || icnt != 0) {
- set_error(_KCCODELINE_, Error::BROKEN, "remaining cache");
- db_.report(_KCCODELINE_, Logger::WARN, "cusage=%lld lsiz=%lld isiz=%lld"
- " lcnt=%lld icnt=%lld", (long long)cusage_, (long long)lsiz, (long long)isiz,
- (long long)lcnt, (long long)icnt);
- err = true;
- }
- delete_inner_cache();
- delete_leaf_cache();
- if (writer_ && !dump_meta()) err = true;
- if (!db_.close()) err = true;
- omode_ = 0;
- trigger_meta(MetaTrigger::CLOSE, "close");
- return !err;
- }
- /**
- * Synchronize updated contents with the file and the device.
- * @param hard true for physical synchronization with the device, or false for logical
- * synchronization with the file system.
- * @param proc a postprocessor object. If it is NULL, no postprocessing is performed.
- * @param checker a progress checker object. If it is NULL, no checking is performed.
- * @return true on success, or false on failure.
- * @note The operation of the postprocessor is performed atomically and other threads accessing
- * the same record are blocked. To avoid deadlock, any explicit database operation must not
- * be performed in this function.
- */
- bool synchronize(bool hard = false, FileProcessor* proc = NULL,
- ProgressChecker* checker = NULL) {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- bool err = false;
- if (writer_) {
- if (checker && !checker->check("synchronize", "cleaning the leaf node cache", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!clean_leaf_cache()) err = true;
- if (checker && !checker->check("synchronize", "cleaning the inner node cache", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!clean_inner_cache()) err = true;
- if (checker && !checker->check("synchronize", "flushing the leaf node cache", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!flush_leaf_cache(true)) err = true;
- if (checker && !checker->check("synchronize", "flushing the inner node cache", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!flush_inner_cache(true)) err = true;
- if (checker && !checker->check("synchronize", "dumping the meta data", -1, -1)) {
- set_error(_KCCODELINE_, Error::LOGIC, "checker failed");
- return false;
- }
- if (!dump_meta()) err = true;
- }
- class Wrapper : public FileProcessor {
- public:
- Wrapper(FileProcessor* proc, int64_t count) : proc_(proc), count_(count) {}
- private:
- bool process(const std::string& path, int64_t count, int64_t size) {
- if (proc_) return proc_->process(path, count_, size);
- return true;
- }
- FileProcessor* proc_;
- int64_t count_;
- } wrapper(proc, count_);
- if (!db_.synchronize(hard, &wrapper, checker)) err = true;
- trigger_meta(MetaTrigger::SYNCHRONIZE, "synchronize");
- return !err;
- }
- /**
- * Occupy database by locking and do something meanwhile.
- * @param writable true to use writer lock, or false to use reader lock.
- * @param proc a processor object. If it is NULL, no processing is performed.
- * @return true on success, or false on failure.
- * @note The operation of the processor is performed atomically and other threads accessing
- * the same record are blocked. To avoid deadlock, any explicit database operation must not
- * be performed in this function.
- */
- bool occupy(bool writable = true, FileProcessor* proc = NULL) {
- _assert_(true);
- bool err = false;
- if (proc && !proc->process(db_.path(), count_, db_.size())) {
- set_error(_KCCODELINE_, Error::LOGIC, "processing failed");
- err = true;
- }
- trigger_meta(MetaTrigger::OCCUPY, "occupy");
- return !err;
- }
- /**
- * Begin transaction.
- * @param hard true for physical synchronization with the device, or false for logical
- * synchronization with the file system.
- * @return true on success, or false on failure.
- */
- bool begin_transaction(bool hard = false) {
- _assert_(true);
- uint32_t wcnt = 0;
- while (true) {
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- if (!tran_) break;
- if (wcnt >= LOCKBUSYLOOP) {
- Thread::chill();
- } else {
- Thread::yield();
- wcnt++;
- }
- }
- if (!begin_transaction_impl(hard)) {
- return false;
- }
- tran_ = true;
- trigger_meta(MetaTrigger::BEGINTRAN, "begin_transaction");
- return true;
- }
- /**
- * Try to begin transaction.
- * @param hard true for physical synchronization with the device, or false for logical
- * synchronization with the file system.
- * @return true on success, or false on failure.
- */
- bool begin_transaction_try(bool hard = false) {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- if (tran_) {
- set_error(_KCCODELINE_, Error::LOGIC, "competition avoided");
- return false;
- }
- if (!begin_transaction_impl(hard)) {
- return false;
- }
- tran_ = true;
- trigger_meta(MetaTrigger::BEGINTRAN, "begin_transaction_try");
- return true;
- }
- /**
- * End transaction.
- * @param commit true to commit the transaction, or false to abort the transaction.
- * @return true on success, or false on failure.
- */
- bool end_transaction(bool commit = true) {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!tran_) {
- set_error(_KCCODELINE_, Error::INVALID, "not in transaction");
- return false;
- }
- bool err = false;
- if (commit) {
- if (!commit_transaction()) err = true;
- } else {
- if (!abort_transaction()) err = true;
- }
- tran_ = false;
- trigger_meta(commit ? MetaTrigger::COMMITTRAN : MetaTrigger::ABORTTRAN, "end_transaction");
- return !err;
- }
- /**
- * Remove all records.
- * @return true on success, or false on failure.
- */
- bool clear() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!writer_) {
- set_error(_KCCODELINE_, Error::NOPERM, "permission denied");
- return false;
- }
- disable_cursors();
- flush_leaf_cache(false);
- flush_inner_cache(false);
- bool err = false;
- if (!db_.clear()) err = true;
- lcnt_ = 0;
- create_leaf_node(0, 0);
- root_ = 1;
- first_ = 1;
- last_ = 1;
- lcnt_ = 1;
- icnt_ = 0;
- count_ = 0;
- if (!dump_meta()) err = true;
- if (!flush_leaf_cache(true)) err = true;
- cusage_ = 0;
- trigger_meta(MetaTrigger::CLEAR, "clear");
- return !err;
- }
- /**
- * Get the number of records.
- * @return the number of records, or -1 on failure.
- */
- int64_t count() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return -1;
- }
- return count_;
- }
- /**
- * Get the size of the database file.
- * @return the size of the database file in bytes, or -1 on failure.
- */
- int64_t size() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return -1;
- }
- return db_.size();
- }
- /**
- * Get the path of the database file.
- * @return the path of the database file, or an empty string on failure.
- */
- std::string path() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return "";
- }
- return db_.path();
- }
- /**
- * Get the miscellaneous status information.
- * @param strmap a string map to contain the result.
- * @return true on success, or false on failure.
- */
- bool status(std::map<std::string, std::string>* strmap) {
- _assert_(strmap);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- if (!db_.status(strmap)) return false;
- (*strmap)["type"] = strprintf("%u", (unsigned)DBTYPE);
- (*strmap)["psiz"] = strprintf("%d", psiz_);
- (*strmap)["pccap"] = strprintf("%lld", (long long)pccap_);
- const char* compname = "external";
- if (reccomp_.comp == LEXICALCOMP) {
- compname = "lexical";
- } else if (reccomp_.comp == DECIMALCOMP) {
- compname = "decimal";
- } else if (reccomp_.comp == LEXICALDESCCOMP) {
- compname = "lexicaldesc";
- } else if (reccomp_.comp == DECIMALDESCCOMP) {
- compname = "decimaldesc";
- }
- (*strmap)["rcomp"] = compname;
- (*strmap)["root"] = strprintf("%lld", (long long)root_);
- (*strmap)["first"] = strprintf("%lld", (long long)first_);
- (*strmap)["last"] = strprintf("%lld", (long long)last_);
- (*strmap)["lcnt"] = strprintf("%lld", (long long)lcnt_);
- (*strmap)["icnt"] = strprintf("%lld", (long long)icnt_);
- (*strmap)["count"] = strprintf("%lld", (long long)count_);
- (*strmap)["bnum"] = strprintf("%lld", (long long)bnum_);
- (*strmap)["pnum"] = strprintf("%lld", (long long)db_.count());
- (*strmap)["cusage"] = strprintf("%lld", (long long)cusage_);
- if (strmap->count("cusage_lcnt") > 0)
- (*strmap)["cusage_lcnt"] = strprintf("%lld", (long long)calc_leaf_cache_count());
- if (strmap->count("cusage_lsiz") > 0)
- (*strmap)["cusage_lsiz"] = strprintf("%lld", (long long)calc_leaf_cache_size());
- if (strmap->count("cusage_icnt") > 0)
- (*strmap)["cusage_icnt"] = strprintf("%lld", (long long)calc_inner_cache_count());
- if (strmap->count("cusage_isiz") > 0)
- (*strmap)["cusage_isiz"] = strprintf("%lld", (long long)calc_inner_cache_size());
- if (strmap->count("tree_level") > 0) {
- Link link;
- link.ksiz = 0;
- int64_t hist[LEVELMAX];
- int32_t hnum = 0;
- search_tree(&link, false, hist, &hnum);
- (*strmap)["tree_level"] = strprintf("%d", hnum + 1);
- }
- return true;
- }
- /**
- * Create a cursor object.
- * @return the return value is the created cursor object.
- * @note Because the object of the return value is allocated by the constructor, it should be
- * released with the delete operator when it is no longer in use.
- */
- Cursor* cursor() {
- _assert_(true);
- return new Cursor(this);
- }
- /**
- * Write a log message.
- * @param file the file name of the program source code.
- * @param line the line number of the program source code.
- * @param func the function name of the program source code.
- * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
- * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
- * @param message the supplement message.
- */
- void log(const char* file, int32_t line, const char* func, Logger::Kind kind,
- const char* message) {
- _assert_(file && line > 0 && func && message);
- db_.log(file, line, func, kind, message);
- }
- /**
- * Set the internal logger.
- * @param logger the logger object.
- * @param kinds kinds of logged messages by bitwise-or: Logger::DEBUG for debugging,
- * Logger::INFO for normal information, Logger::WARN for warning, and Logger::ERROR for fatal
- * error.
- * @return true on success, or false on failure.
- */
- bool tune_logger(Logger* logger, uint32_t kinds = Logger::WARN | Logger::ERROR) {
- _assert_(logger);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- return db_.tune_logger(logger, kinds);
- }
- /**
- * Set the internal meta operation trigger.
- * @param trigger the trigger object.
- * @return true on success, or false on failure.
- */
- bool tune_meta_trigger(MetaTrigger* trigger) {
- _assert_(trigger);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- mtrigger_ = trigger;
- return true;
- }
- /**
- * Set the power of the alignment of record size.
- * @param apow the power of the alignment of record size.
- * @return true on success, or false on failure.
- */
- bool tune_alignment(int8_t apow) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- apow_ = apow >= 0 ? apow : DEFAPOW;
- return true;
- }
- /**
- * Set the power of the capacity of the free block pool.
- * @param fpow the power of the capacity of the free block pool.
- * @return true on success, or false on failure.
- */
- bool tune_fbp(int8_t fpow) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- fpow_ = fpow >= 0 ? fpow : DEFFPOW;
- return true;
- }
- /**
- * Set the optional features.
- * @param opts the optional features by bitwise-or: BasicDB::TSMALL to use 32-bit addressing,
- * BasicDB::TLINEAR to use linear collision chaining, BasicDB::TCOMPRESS to compress each
- * record.
- * @return true on success, or false on failure.
- */
- bool tune_options(int8_t opts) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- opts_ = opts;
- return true;
- }
- /**
- * Set the number of buckets of the hash table.
- * @param bnum the number of buckets of the hash table.
- * @return true on success, or false on failure.
- */
- bool tune_buckets(int64_t bnum) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- bnum_ = bnum > 0 ? bnum : DEFBNUM;
- return true;
- }
- /**
- * Set the size of each page.
- * @param psiz the size of each page.
- * @return true on success, or false on failure.
- */
- bool tune_page(int32_t psiz) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- psiz_ = psiz > 0 ? psiz : DEFPSIZ;
- return true;
- }
- /**
- * Set the size of the internal memory-mapped region.
- * @param msiz the size of the internal memory-mapped region.
- * @return true on success, or false on failure.
- */
- bool tune_map(int64_t msiz) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- return db_.tune_map(msiz);
- }
- /**
- * Set the unit step number of auto defragmentation.
- * @param dfunit the unit step number of auto defragmentation.
- * @return true on success, or false on failure.
- */
- bool tune_defrag(int64_t dfunit) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- return db_.tune_defrag(dfunit);
- }
- /**
- * Set the capacity size of the page cache.
- * @param pccap the capacity size of the page cache.
- * @return true on success, or false on failure.
- */
- bool tune_page_cache(int64_t pccap) {
- _assert_(true);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- pccap_ = pccap > 0 ? pccap : DEFPCCAP;
- return true;
- }
- /**
- * Set the data compressor.
- * @param comp the data compressor object.
- * @return true on success, or false on failure.
- */
- bool tune_compressor(Compressor* comp) {
- _assert_(comp);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- return db_.tune_compressor(comp);
- }
- /**
- * Set the record comparator.
- * @param rcomp the record comparator object.
- * @return true on success, or false on failure.
- * @note Several built-in comparators are provided. LEXICALCOMP for the default lexical
- * comparator. DECIMALCOMP for the decimal comparator. LEXICALDESCCOMP for the lexical
- * descending comparator. DECIMALDESCCOMP for the lexical descending comparator.
- */
- bool tune_comparator(Comparator* rcomp) {
- _assert_(rcomp);
- if (omode_ != 0) {
- set_error(_KCCODELINE_, Error::INVALID, "already opened");
- return false;
- }
- reccomp_.comp = rcomp;
- return true;
- }
- /**
- * Get the opaque data.
- * @return the pointer to the opaque data region, whose size is 16 bytes.
- */
- char* opaque() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return NULL;
- }
- return db_.opaque();
- }
- /**
- * Synchronize the opaque data.
- * @return true on success, or false on failure.
- */
- bool synchronize_opaque() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- return db_.synchronize_opaque();
- }
- /**
- * Perform defragmentation of the file.
- * @param step the number of steps. If it is not more than 0, the whole region is defraged.
- * @return true on success, or false on failure.
- */
- bool defrag(int64_t step = 0) {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return false;
- }
- bool err = false;
- if (step < 1 && writer_) {
- if (!clean_leaf_cache()) err = true;
- if (!clean_inner_cache()) err = true;
- }
- if (!db_.defrag(step)) err = true;
- return !err;
- }
- /**
- * Get the status flags.
- * @return the status flags, or 0 on failure.
- */
- uint8_t flags() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return 0;
- }
- return db_.flags();
- }
- /**
- * Get the record comparator.
- * @return the record comparator object.
- */
- Comparator* rcomp() {
- _assert_(true);
- if (omode_ == 0) {
- set_error(_KCCODELINE_, Error::INVALID, "not opened");
- return 0;
- }
- return reccomp_.comp;
- }
- protected:
- /**
- * Report a message for debugging.
- * @param file the file name of the program source code.
- * @param line the line number of the program source code.
- * @param func the function name of the program source code.
- * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
- * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
- * @param format the printf-like format string.
- * @param ... used according to the format string.
- */
- void report(const char* file, int32_t line, const char* func, Logger::Kind kind,
- const char* format, ...) {
- _assert_(file && line > 0 && func && format);
- va_list ap;
- va_start(ap, format);
- db_.report_valist(file, line, func, kind, format, ap);
- va_end(ap);
- }
- /**
- * Report a message for debugging with variable number of arguments.
- * @param file the file name of the program source code.
- * @param line the line number of the program source code.
- * @param func the function name of the program source code.
- * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
- * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
- * @param format the printf-like format string.
- * @param ap used according to the format string.
- */
- void report_valist(const char* file, int32_t line, const char* func, Logger::Kind kind,
- const char* format, va_list ap) {
- _assert_(file && line > 0 && func && format);
- db_.report_valist(file, line, func, kind, format, ap);
- }
- /**
- * Report the content of a binary buffer for debugging.
- * @param file the file name of the epicenter.
- * @param line the line number of the epicenter.
- * @param func the function name of the program source code.
- * @param kind the kind of the event. Logger::DEBUG for debugging, Logger::INFO for normal
- * information, Logger::WARN for warning, and Logger::ERROR for fatal error.
- * @param name the name of the information.
- * @param buf the binary buffer.
- * @param size the size of the binary buffer
- */
- void report_binary(const char* file, int32_t line, const char* func, Logger::Kind kind,
- const char* name, const char* buf, size_t size) {
- _assert_(file && line > 0 && func && name && buf && size <= MEMMAXSIZ);
- db_.report_binary(file, line, func, kind, name, buf, size);
- }
- /**
- * Trigger a meta database operation.
- * @param kind the kind of the event. MetaTrigger::OPEN for opening, MetaTrigger::CLOSE for
- * closing, MetaTrigger::CLEAR for clearing, MetaTrigger::ITERATE for iteration,
- * MetaTrigger::SYNCHRONIZE for synchronization, MetaTrigger::BEGINTRAN for beginning
- * transaction, MetaTrigger::COMMITTRAN for committing transaction, MetaTrigger::ABORTTRAN
- * for aborting transaction, and MetaTrigger::MISC for miscellaneous operations.
- * @param message the supplement message.
- */
- void trigger_meta(MetaTrigger::Kind kind, const char* message) {
- _assert_(message);
- if (mtrigger_) mtrigger_->trigger(kind, message);
- }
- private:
- /**
- * Record data.
- */
- struct Record {
- uint32_t ksiz; ///< size of the key
- uint32_t vsiz; ///< size of the value
- };
- /**
- * Comparator for records.
- */
- struct RecordComparator {
- Comparator* comp; ///< comparator
- /** constructor */
- explicit RecordComparator() : comp(NULL) {}
- /** comparing operator */
- bool operator ()(const Record* const& a, const Record* const& b) const {
- _assert_(true);
- char* akbuf = (char*)a + sizeof(*a);
- char* bkbuf = (char*)b + sizeof(*b);
- return comp->compare(akbuf, a->ksiz, bkbuf, b->ksiz) < 0;
- }
- };
- /**
- * Leaf node of B+ tree.
- */
- struct LeafNode {
- int64_t id; ///< page ID number
- RecordArray recs; ///< sorted array of records
- int64_t size; ///< total size of records
- int64_t prev; ///< previous leaf node
- int64_t next; ///< next leaf node
- bool hot; ///< whether in the hot cache
- bool dirty; ///< whether to be written back
- bool dead; ///< whether to be removed
- };
- /**
- * Link to a node.
- */
- struct Link {
- int64_t child; ///< child node
- int32_t ksiz; ///< size of the key
- };
- /**
- * Comparator for links.
- */
- struct LinkComparator {
- Comparator* comp; ///< comparator
- /** constructor */
- explicit LinkComparator() : comp(NULL) {
- _assert_(true);
- }
- /** comparing operator */
- bool operator ()(const Link* const& a, const Link* const& b) const {
- _assert_(true);
- char* akbuf = (char*)a + sizeof(*a);
- char* bkbuf = (char*)b + sizeof(*b);
- return comp->compare(akbuf, a->ksiz, bkbuf, b->ksiz) < 0;
- }
- };
- /**
- * Inner node of B+ tree.
- */
- struct InnerNode {
- int64_t id; ///< page ID numger
- int64_t heir; ///< child before the first link
- LinkArray links; ///< sorted array of links
- int64_t size; ///< total size of links
- bool dirty; ///< whether to be written back
- bool dead; ///< whether to be removed
- };
- /**
- * Slot cache of leaf nodes.
- */
- struct LeafSlot {
- Mutex lock; ///< lock
- LeafCache* hot; ///< hot cache
- LeafCache* warm; ///< warm cache
- };
- /**
- * Slot cache of inner nodes.
- */
- struct InnerSlot {
- Mutex lock; ///< lock
- InnerCache* warm; ///< warm cache
- };
- /**
- * Scoped visitor.
- */
- class ScopedVisitor {
- public:
- /** constructor */
- explicit ScopedVisitor(Visitor* visitor) : visitor_(visitor) {
- _assert_(visitor);
- visitor_->visit_before();
- }
- /** destructor */
- ~ScopedVisitor() {
- _assert_(true);
- visitor_->visit_after();
- }
- private:
- Visitor* visitor_; ///< visitor
- };
- /**
- * Open the leaf cache.
- */
- void create_leaf_cache() {
- _assert_(true);
- int64_t bnum = bnum_ / SLOTNUM + 1;
- if (bnum < INT8MAX) bnum = INT8MAX;
- bnum = nearbyprime(bnum);
- for (int32_t i = 0; i < SLOTNUM; i++) {
- lslots_[i].hot = new LeafCache(bnum);
- lslots_[i].warm = new LeafCache(bnum);
- }
- }
- /**
- * Close the leaf cache.
- */
- void delete_leaf_cache() {
- _assert_(true);
- for (int32_t i = SLOTNUM - 1; i >= 0; i--) {
- LeafSlot* slot = lslots_ + i;
- delete slot->warm;
- delete slot->hot;
- }
- }
- /**
- * Remove all leaf nodes from the leaf cache.
- * @param save whether to save dirty nodes.
- * @return true on success, or false on failure.
- */
- bool flush_leaf_cache(bool save) {
- _assert_(true);
- bool err = false;
- for (int32_t i = SLOTNUM - 1; i >= 0; i--) {
- LeafSlot* slot = lslots_ + i;
- typename LeafCache::Iterator it = slot->warm->begin();
- typename LeafCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- LeafNode* node = it.value();
- ++it;
- if (!flush_leaf_node(node, save)) err = true;
- }
- it = slot->hot->begin();
- itend = slot->hot->end();
- while (it != itend) {
- LeafNode* node = it.value();
- ++it;
- if (!flush_leaf_node(node, save)) err = true;
- }
- }
- return !err;
- }
- /**
- * Flush a part of the leaf cache.
- * @param slot a slot of leaf nodes.
- * @return true on success, or false on failure.
- */
- bool flush_leaf_cache_part(LeafSlot* slot) {
- _assert_(slot);
- bool err = false;
- if (slot->warm->count() > 0) {
- LeafNode* node = slot->warm->first_value();
- if (!flush_leaf_node(node, true)) err = true;
- } else if (slot->hot->count() > 0) {
- LeafNode* node = slot->hot->first_value();
- if (!flush_leaf_node(node, true)) err = true;
- }
- return !err;
- }
- /**
- * Clean all of the leaf cache.
- * @return true on success, or false on failure.
- */
- bool clean_leaf_cache() {
- _assert_(true);
- bool err = false;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- LeafSlot* slot = lslots_ + i;
- ScopedMutex lock(&slot->lock);
- typename LeafCache::Iterator it = slot->warm->begin();
- typename LeafCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- LeafNode* node = it.value();
- if (!save_leaf_node(node)) err = true;
- ++it;
- }
- it = slot->hot->begin();
- itend = slot->hot->end();
- while (it != itend) {
- LeafNode* node = it.value();
- if (!save_leaf_node(node)) err = true;
- ++it;
- }
- }
- return !err;
- }
- /**
- * Clean a part of the leaf cache.
- * @param slot a slot of leaf nodes.
- * @return true on success, or false on failure.
- */
- bool clean_leaf_cache_part(LeafSlot* slot) {
- _assert_(slot);
- bool err = false;
- ScopedMutex lock(&slot->lock);
- if (slot->warm->count() > 0) {
- LeafNode* node = slot->warm->first_value();
- if (!save_leaf_node(node)) err = true;
- } else if (slot->hot->count() > 0) {
- LeafNode* node = slot->hot->first_value();
- if (!save_leaf_node(node)) err = true;
- }
- return !err;
- }
- /**
- * Create a new leaf node.
- * @param prev the ID of the previous node.
- * @param next the ID of the next node.
- * @return the created leaf node.
- */
- LeafNode* create_leaf_node(int64_t prev, int64_t next) {
- _assert_(true);
- LeafNode* node = new LeafNode;
- node->id = ++lcnt_;
- node->size = sizeof(int32_t) * 2;
- node->recs.reserve(DEFLINUM);
- node->prev = prev;
- node->next = next;
- node->hot = false;
- node->dirty = true;
- node->dead = false;
- int32_t sidx = node->id % SLOTNUM;
- LeafSlot* slot = lslots_ + sidx;
- slot->warm->set(node->id, node, LeafCache::MLAST);
- cusage_ += node->size;
- return node;
- }
- /**
- * Remove a leaf node from the cache.
- * @param node the leaf node.
- * @param save whether to save dirty node.
- * @return true on success, or false on failure.
- */
- bool flush_leaf_node(LeafNode* node, bool save) {
- _assert_(node);
- bool err = false;
- if (save && !save_leaf_node(node)) err = true;
- typename RecordArray::const_iterator rit = node->recs.begin();
- typename RecordArray::const_iterator ritend = node->recs.end();
- while (rit != ritend) {
- Record* rec = *rit;
- xfree(rec);
- ++rit;
- }
- int32_t sidx = node->id % SLOTNUM;
- LeafSlot* slot = lslots_ + sidx;
- if (node->hot) {
- slot->hot->remove(node->id);
- } else {
- slot->warm->remove(node->id);
- }
- cusage_ -= node->size;
- delete node;
- return !err;
- }
- /**
- * Save a leaf node.
- * @param node the leaf node.
- * @return true on success, or false on failure.
- */
- bool save_leaf_node(LeafNode* node) {
- _assert_(node);
- if (!node->dirty) return true;
- bool err = false;
- char hbuf[NUMBUFSIZ];
- size_t hsiz = write_key(hbuf, LNPREFIX, node->id);
- if (node->dead) {
- if (!db_.remove(hbuf, hsiz) && db_.error().code() != Error::NOREC) err = true;
- } else {
- char* rbuf = new char[node->size];
- char* wp = rbuf;
- wp += writevarnum(wp, node->prev);
- wp += writevarnum(wp, node->next);
- typename RecordArray::const_iterator rit = node->recs.begin();
- typename RecordArray::const_iterator ritend = node->recs.end();
- while (rit != ritend) {
- Record* rec = *rit;
- wp += writevarnum(wp, rec->ksiz);
- wp += writevarnum(wp, rec->vsiz);
- char* dbuf = (char*)rec + sizeof(*rec);
- std::memcpy(wp, dbuf, rec->ksiz);
- wp += rec->ksiz;
- std::memcpy(wp, dbuf + rec->ksiz, rec->vsiz);
- wp += rec->vsiz;
- ++rit;
- }
- if (!db_.set(hbuf, hsiz, rbuf, wp - rbuf)) err = true;
- delete[] rbuf;
- }
- node->dirty = false;
- return !err;
- }
- /**
- * Load a leaf node.
- * @param id the ID number of the leaf node.
- * @param prom whether to promote the warm cache.
- * @return the loaded leaf node.
- */
- LeafNode* load_leaf_node(int64_t id, bool prom) {
- _assert_(id > 0);
- int32_t sidx = id % SLOTNUM;
- LeafSlot* slot = lslots_ + sidx;
- ScopedMutex lock(&slot->lock);
- LeafNode** np = slot->hot->get(id, LeafCache::MLAST);
- if (np) return *np;
- if (prom) {
- if (slot->hot->count() * WARMRATIO > slot->warm->count() + WARMRATIO) {
- slot->hot->first_value()->hot = false;
- slot->hot->migrate(slot->hot->first_key(), slot->warm, LeafCache::MLAST);
- }
- np = slot->warm->migrate(id, slot->hot, LeafCache::MLAST);
- if (np) {
- (*np)->hot = true;
- return *np;
- }
- } else {
- LeafNode** np = slot->warm->get(id, LeafCache::MLAST);
- if (np) return *np;
- }
- char hbuf[NUMBUFSIZ];
- size_t hsiz = write_key(hbuf, LNPREFIX, id);
- class VisitorImpl : public DB::Visitor {
- public:
- explicit VisitorImpl() : node_(NULL) {}
- LeafNode* pop() {
- return node_;
- }
- private:
- const char* visit_full(const char* kbuf, size_t ksiz,
- const char* vbuf, size_t vsiz, size_t* sp) {
- uint64_t prev;
- size_t step = readvarnum(vbuf, vsiz, &prev);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- uint64_t next;
- step = readvarnum(vbuf, vsiz, &next);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- LeafNode* node = new LeafNode;
- node->size = sizeof(int32_t) * 2;
- node->prev = prev;
- node->next = next;
- while (vsiz > 1) {
- uint64_t rksiz;
- step = readvarnum(vbuf, vsiz, &rksiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- uint64_t rvsiz;
- step = readvarnum(vbuf, vsiz, &rvsiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- if (vsiz < rksiz + rvsiz) break;
- size_t rsiz = sizeof(Record) + rksiz + rvsiz;
- Record* rec = (Record*)xmalloc(rsiz);
- rec->ksiz = rksiz;
- rec->vsiz = rvsiz;
- char* dbuf = (char*)rec + sizeof(*rec);
- std::memcpy(dbuf, vbuf, rksiz);
- vbuf += rksiz;
- vsiz -= rksiz;
- std::memcpy(dbuf + rksiz, vbuf, rvsiz);
- vbuf += rvsiz;
- vsiz -= rvsiz;
- node->recs.push_back(rec);
- node->size += rsiz;
- }
- if (vsiz != 0) {
- typename RecordArray::const_iterator rit = node->recs.begin();
- typename RecordArray::const_iterator ritend = node->recs.end();
- while (rit != ritend) {
- Record* rec = *rit;
- xfree(rec);
- ++rit;
- }
- delete node;
- return NOP;
- }
- node_ = node;
- return NOP;
- }
- LeafNode* node_;
- } visitor;
- if (!db_.accept(hbuf, hsiz, &visitor, false)) return NULL;
- LeafNode* node = visitor.pop();
- if (!node) return NULL;
- node->id = id;
- node->hot = false;
- node->dirty = false;
- node->dead = false;
- slot->warm->set(id, node, LeafCache::MLAST);
- cusage_ += node->size;
- return node;
- }
- /**
- * Check whether a record is in the range of a leaf node.
- * @param node the leaf node.
- * @param rec the record containing the key only.
- * @return true for in range, or false for out of range.
- */
- bool check_leaf_node_range(LeafNode* node, Record* rec) {
- _assert_(node && rec);
- RecordArray& recs = node->recs;
- if (recs.empty()) return false;
- Record* frec = recs.front();
- Record* lrec = recs.back();
- return !reccomp_(rec, frec) && !reccomp_(lrec, rec);
- }
- /**
- * Accept a visitor at a leaf node.
- * @param node the leaf node.
- * @param rec the record containing the key only.
- * @param visitor a visitor object.
- * @return true to reorganize the tree, or false if not.
- */
- bool accept_impl(LeafNode* node, Record* rec, Visitor* visitor) {
- _assert_(node && rec && visitor);
- bool reorg = false;
- RecordArray& recs = node->recs;
- typename RecordArray::iterator ritend = recs.end();
- typename RecordArray::iterator rit = std::lower_bound(recs.begin(), ritend, rec, reccomp_);
- if (rit != ritend && !reccomp_(rec, *rit)) {
- Record* rec = *rit;
- char* kbuf = (char*)rec + sizeof(*rec);
- size_t ksiz = rec->ksiz;
- size_t vsiz;
- const char* vbuf = visitor->visit_full(kbuf, ksiz, kbuf + ksiz, rec->vsiz, &vsiz);
- if (vbuf == Visitor::REMOVE) {
- size_t rsiz = sizeof(*rec) + rec->ksiz + rec->vsiz;
- count_ -= 1;
- cusage_ -= rsiz;
- node->size -= rsiz;
- node->dirty = true;
- xfree(rec);
- recs.erase(rit);
- if (recs.empty()) reorg = true;
- } else if (vbuf != Visitor::NOP) {
- int64_t diff = (int64_t)vsiz - (int64_t)rec->vsiz;
- cusage_ += diff;
- node->size += diff;
- node->dirty = true;
- if (vsiz > rec->vsiz) {
- *rit = (Record*)xrealloc(rec, sizeof(*rec) + rec->ksiz + vsiz);
- rec = *rit;
- kbuf = (char*)rec + sizeof(*rec);
- }
- std::memcpy(kbuf + rec->ksiz, vbuf, vsiz);
- rec->vsiz = vsiz;
- if (node->size > psiz_ && recs.size() > 1) reorg = true;
- }
- } else {
- const char* kbuf = (char*)rec + sizeof(*rec);
- size_t ksiz = rec->ksiz;
- size_t vsiz;
- const char* vbuf = visitor->visit_empty(kbuf, ksiz, &vsiz);
- if (vbuf != Visitor::NOP && vbuf != Visitor::REMOVE) {
- size_t rsiz = sizeof(*rec) + ksiz + vsiz;
- count_ += 1;
- cusage_ += rsiz;
- node->size += rsiz;
- node->dirty = true;
- rec = (Record*)xmalloc(rsiz);
- rec->ksiz = ksiz;
- rec->vsiz = vsiz;
- char* dbuf = (char*)rec + sizeof(*rec);
- std::memcpy(dbuf, kbuf, ksiz);
- std::memcpy(dbuf + ksiz, vbuf, vsiz);
- recs.insert(rit, rec);
- if (node->size > psiz_ && recs.size() > 1) reorg = true;
- }
- }
- return reorg;
- }
- /**
- * Devide a leaf node into two.
- * @param node the leaf node.
- * @return the created node, or NULL on failure.
- */
- LeafNode* divide_leaf_node(LeafNode* node) {
- _assert_(node);
- LeafNode* newnode = create_leaf_node(node->id, node->next);
- if (newnode->next > 0) {
- LeafNode* nextnode = load_leaf_node(newnode->next, false);
- if (!nextnode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing leaf node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)newnode->next);
- return NULL;
- }
- nextnode->prev = newnode->id;
- nextnode->dirty = true;
- }
- node->next = newnode->id;
- node->dirty = true;
- RecordArray& recs = node->recs;
- typename RecordArray::iterator mid = recs.begin() + recs.size() / 2;
- typename RecordArray::iterator rit = mid;
- typename RecordArray::iterator ritend = recs.end();
- RecordArray& newrecs = newnode->recs;
- while (rit != ritend) {
- Record* rec = *rit;
- newrecs.push_back(rec);
- size_t rsiz = sizeof(*rec) + rec->ksiz + rec->vsiz;
- node->size -= rsiz;
- newnode->size += rsiz;
- ++rit;
- }
- escape_cursors(node->id, node->next, *mid);
- recs.erase(mid, ritend);
- return newnode;
- }
- /**
- * Open the inner cache.
- */
- void create_inner_cache() {
- _assert_(true);
- int64_t bnum = (bnum_ / AVGWAY) / SLOTNUM + 1;
- if (bnum < INT8MAX) bnum = INT8MAX;
- bnum = nearbyprime(bnum);
- for (int32_t i = 0; i < SLOTNUM; i++) {
- islots_[i].warm = new InnerCache(bnum);
- }
- }
- /**
- * Close the inner cache.
- */
- void delete_inner_cache() {
- _assert_(true);
- for (int32_t i = SLOTNUM - 1; i >= 0; i--) {
- InnerSlot* slot = islots_ + i;
- delete slot->warm;
- }
- }
- /**
- * Remove all inner nodes from the inner cache.
- * @param save whether to save dirty nodes.
- * @return true on success, or false on failure.
- */
- bool flush_inner_cache(bool save) {
- _assert_(true);
- bool err = false;
- for (int32_t i = SLOTNUM - 1; i >= 0; i--) {
- InnerSlot* slot = islots_ + i;
- typename InnerCache::Iterator it = slot->warm->begin();
- typename InnerCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- InnerNode* node = it.value();
- ++it;
- if (!flush_inner_node(node, save)) err = true;
- }
- }
- return !err;
- }
- /**
- * Flush a part of the inner cache.
- * @param slot a slot of inner nodes.
- * @return true on success, or false on failure.
- */
- bool flush_inner_cache_part(InnerSlot* slot) {
- _assert_(slot);
- bool err = false;
- if (slot->warm->count() > 0) {
- InnerNode* node = slot->warm->first_value();
- if (!flush_inner_node(node, true)) err = true;
- }
- return !err;
- }
- /**
- * Clean all of the inner cache.
- * @return true on success, or false on failure.
- */
- bool clean_inner_cache() {
- _assert_(true);
- bool err = false;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- InnerSlot* slot = islots_ + i;
- ScopedMutex lock(&slot->lock);
- typename InnerCache::Iterator it = slot->warm->begin();
- typename InnerCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- InnerNode* node = it.value();
- if (!save_inner_node(node)) err = true;
- ++it;
- }
- }
- return !err;
- }
- /**
- * Create a new inner node.
- * @param heir the ID of the child before the first link.
- * @return the created inner node.
- */
- InnerNode* create_inner_node(int64_t heir) {
- _assert_(true);
- InnerNode* node = new InnerNode;
- node->id = ++icnt_ + INIDBASE;
- node->heir = heir;
- node->links.reserve(DEFIINUM);
- node->size = sizeof(int64_t);
- node->dirty = true;
- node->dead = false;
- int32_t sidx = node->id % SLOTNUM;
- InnerSlot* slot = islots_ + sidx;
- slot->warm->set(node->id, node, InnerCache::MLAST);
- cusage_ += node->size;
- return node;
- }
- /**
- * Remove an inner node from the cache.
- * @param node the inner node.
- * @param save whether to save dirty node.
- * @return true on success, or false on failure.
- */
- bool flush_inner_node(InnerNode* node, bool save) {
- _assert_(node);
- bool err = false;
- if (save && !save_inner_node(node)) err = true;
- typename LinkArray::const_iterator lit = node->links.begin();
- typename LinkArray::const_iterator litend = node->links.end();
- while (lit != litend) {
- Link* link = *lit;
- xfree(link);
- ++lit;
- }
- int32_t sidx = node->id % SLOTNUM;
- InnerSlot* slot = islots_ + sidx;
- slot->warm->remove(node->id);
- cusage_ -= node->size;
- delete node;
- return !err;
- }
- /**
- * Save a inner node.
- * @param node the inner node.
- * @return true on success, or false on failure.
- */
- bool save_inner_node(InnerNode* node) {
- _assert_(true);
- if (!node->dirty) return true;
- bool err = false;
- char hbuf[NUMBUFSIZ];
- size_t hsiz = write_key(hbuf, INPREFIX, node->id - INIDBASE);
- if (node->dead) {
- if (!db_.remove(hbuf, hsiz) && db_.error().code() != Error::NOREC) err = true;
- } else {
- char* rbuf = new char[node->size];
- char* wp = rbuf;
- wp += writevarnum(wp, node->heir);
- typename LinkArray::const_iterator lit = node->links.begin();
- typename LinkArray::const_iterator litend = node->links.end();
- while (lit != litend) {
- Link* link = *lit;
- wp += writevarnum(wp, link->child);
- wp += writevarnum(wp, link->ksiz);
- char* dbuf = (char*)link + sizeof(*link);
- std::memcpy(wp, dbuf, link->ksiz);
- wp += link->ksiz;
- ++lit;
- }
- if (!db_.set(hbuf, hsiz, rbuf, wp - rbuf)) err = true;
- delete[] rbuf;
- }
- node->dirty = false;
- return !err;
- }
- /**
- * Load an inner node.
- * @param id the ID number of the inner node.
- * @return the loaded inner node.
- */
- InnerNode* load_inner_node(int64_t id) {
- _assert_(id > 0);
- int32_t sidx = id % SLOTNUM;
- InnerSlot* slot = islots_ + sidx;
- ScopedMutex lock(&slot->lock);
- InnerNode** np = slot->warm->get(id, InnerCache::MLAST);
- if (np) return *np;
- char hbuf[NUMBUFSIZ];
- size_t hsiz = write_key(hbuf, INPREFIX, id - INIDBASE);
- class VisitorImpl : public DB::Visitor {
- public:
- explicit VisitorImpl() : node_(NULL) {}
- InnerNode* pop() {
- return node_;
- }
- private:
- const char* visit_full(const char* kbuf, size_t ksiz,
- const char* vbuf, size_t vsiz, size_t* sp) {
- uint64_t heir;
- size_t step = readvarnum(vbuf, vsiz, &heir);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- InnerNode* node = new InnerNode;
- node->size = sizeof(int64_t);
- node->heir = heir;
- while (vsiz > 1) {
- uint64_t child;
- step = readvarnum(vbuf, vsiz, &child);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- uint64_t rksiz;
- step = readvarnum(vbuf, vsiz, &rksiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- if (vsiz < rksiz) break;
- Link* link = (Link*)xmalloc(sizeof(*link) + rksiz);
- link->child = child;
- link->ksiz = rksiz;
- char* dbuf = (char*)link + sizeof(*link);
- std::memcpy(dbuf, vbuf, rksiz);
- vbuf += rksiz;
- vsiz -= rksiz;
- node->links.push_back(link);
- node->size += sizeof(*link) + rksiz;
- }
- if (vsiz != 0) {
- typename LinkArray::const_iterator lit = node->links.begin();
- typename LinkArray::const_iterator litend = node->links.end();
- while (lit != litend) {
- Link* link = *lit;
- xfree(link);
- ++lit;
- }
- delete node;
- return NOP;
- }
- node_ = node;
- return NOP;
- }
- InnerNode* node_;
- } visitor;
- if (!db_.accept(hbuf, hsiz, &visitor, false)) return NULL;
- InnerNode* node = visitor.pop();
- if (!node) return NULL;
- node->id = id;
- node->dirty = false;
- node->dead = false;
- slot->warm->set(id, node, InnerCache::MLAST);
- cusage_ += node->size;
- return node;
- }
- /**
- * Search the B+ tree.
- * @param link the link containing the key only.
- * @param prom whether to promote the warm cache.
- * @param hist the array of visiting history.
- * @param hnp the pointer to the variable into which the number of the history is assigned.
- * @return the corresponding leaf node, or NULL on failure.
- */
- LeafNode* search_tree(Link* link, bool prom, int64_t* hist, int32_t* hnp) {
- _assert_(link && hist && hnp);
- int64_t id = root_;
- int32_t hnum = 0;
- while (id > INIDBASE) {
- InnerNode* node = load_inner_node(id);
- if (!node) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing inner node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)id);
- return NULL;
- }
- hist[hnum++] = id;
- const LinkArray& links = node->links;
- typename LinkArray::const_iterator litbeg = links.begin();
- typename LinkArray::const_iterator litend = links.end();
- typename LinkArray::const_iterator lit = std::upper_bound(litbeg, litend, link, linkcomp_);
- if (lit == litbeg) {
- id = node->heir;
- } else {
- --lit;
- Link* link = *lit;
- id = link->child;
- }
- }
- *hnp = hnum;
- return load_leaf_node(id, prom);
- }
- /**
- * Reorganize the B+ tree.
- * @param node a leaf node.
- * @param hist the array of visiting history.
- * @param hnum the number of the history.
- * @return true on success, or false on failure.
- */
- bool reorganize_tree(LeafNode* node, int64_t* hist, int32_t hnum) {
- _assert_(node && hist && hnum >= 0);
- if (node->size > psiz_ && node->recs.size() > 1) {
- LeafNode* newnode = divide_leaf_node(node);
- if (!newnode) return false;
- if (node->id == last_) last_ = newnode->id;
- int64_t heir = node->id;
- int64_t child = newnode->id;
- Record* rec = *newnode->recs.begin();
- char* dbuf = (char*)rec + sizeof(*rec);
- int32_t ksiz = rec->ksiz;
- char* kbuf = new char[ksiz];
- std::memcpy(kbuf, dbuf, ksiz);
- while (true) {
- if (hnum < 1) {
- InnerNode* inode = create_inner_node(heir);
- add_link_inner_node(inode, child, kbuf, ksiz);
- root_ = inode->id;
- delete[] kbuf;
- break;
- }
- int64_t parent = hist[--hnum];
- InnerNode* inode = load_inner_node(parent);
- if (!inode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing inner node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)parent);
- delete[] kbuf;
- return false;
- }
- add_link_inner_node(inode, child, kbuf, ksiz);
- delete[] kbuf;
- LinkArray& links = inode->links;
- if (inode->size <= psiz_ || links.size() <= INLINKMIN) break;
- typename LinkArray::iterator litbeg = links.begin();
- typename LinkArray::iterator mid = litbeg + links.size() / 2;
- Link* link = *mid;
- InnerNode* newinode = create_inner_node(link->child);
- heir = inode->id;
- child = newinode->id;
- char* dbuf = (char*)link + sizeof(*link);
- ksiz = link->ksiz;
- kbuf = new char[ksiz];
- std::memcpy(kbuf, dbuf, ksiz);
- typename LinkArray::iterator lit = mid + 1;
- typename LinkArray::iterator litend = links.end();
- while (lit != litend) {
- link = *lit;
- char* dbuf = (char*)link + sizeof(*link);
- add_link_inner_node(newinode, link->child, dbuf, link->ksiz);
- ++lit;
- }
- int32_t num = newinode->links.size();
- for (int32_t i = 0; i <= num; i++) {
- Link* link = links.back();
- size_t rsiz = sizeof(*link) + link->ksiz;
- cusage_ -= rsiz;
- inode->size -= rsiz;
- xfree(link);
- links.pop_back();
- }
- inode->dirty = true;
- }
- } else if (node->recs.empty() && hnum > 0) {
- if (!escape_cursors(node->id, node->next)) return false;
- InnerNode* inode = load_inner_node(hist[--hnum]);
- if (!inode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing inner node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)hist[hnum]);
- return false;
- }
- if (sub_link_tree(inode, node->id, hist, hnum)) {
- if (node->prev > 0) {
- LeafNode* tnode = load_leaf_node(node->prev, false);
- if (!tnode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)node->prev);
- return false;
- }
- tnode->next = node->next;
- tnode->dirty = true;
- if (last_ == node->id) last_ = node->prev;
- }
- if (node->next > 0) {
- LeafNode* tnode = load_leaf_node(node->next, false);
- if (!tnode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)node->next);
- return false;
- }
- tnode->prev = node->prev;
- tnode->dirty = true;
- if (first_ == node->id) first_ = node->next;
- }
- node->dead = true;
- }
- }
- return true;
- }
- /**
- * Add a link to a inner node.
- * @param node the inner node.
- * @param child the ID number of the child.
- * @param kbuf the pointer to the key region.
- * @param ksiz the size of the key region.
- */
- void add_link_inner_node(InnerNode* node, int64_t child, const char* kbuf, size_t ksiz) {
- _assert_(node && kbuf);
- size_t rsiz = sizeof(Link) + ksiz;
- Link* link = (Link*)xmalloc(rsiz);
- link->child = child;
- link->ksiz = ksiz;
- char* dbuf = (char*)link + sizeof(*link);
- std::memcpy(dbuf, kbuf, ksiz);
- LinkArray& links = node->links;
- typename LinkArray::iterator litend = links.end();
- typename LinkArray::iterator lit = std::upper_bound(links.begin(), litend, link, linkcomp_);
- links.insert(lit, link);
- node->size += rsiz;
- node->dirty = true;
- cusage_ += rsiz;
- }
- /**
- * Subtract a link from the B+ tree.
- * @param node the inner node.
- * @param child the ID number of the child.
- * @param hist the array of visiting history.
- * @param hnum the number of the history.
- * @return true on success, or false on failure.
- */
- bool sub_link_tree(InnerNode* node, int64_t child, int64_t* hist, int32_t hnum) {
- _assert_(node && hist && hnum >= 0);
- node->dirty = true;
- LinkArray& links = node->links;
- typename LinkArray::iterator lit = links.begin();
- typename LinkArray::iterator litend = links.end();
- if (node->heir == child) {
- if (!links.empty()) {
- Link* link = *lit;
- node->heir = link->child;
- xfree(link);
- links.erase(lit);
- return true;
- } else if (hnum > 0) {
- InnerNode* pnode = load_inner_node(hist[--hnum]);
- if (!pnode) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing inner node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)hist[hnum]);
- return false;
- }
- node->dead = true;
- return sub_link_tree(pnode, node->id, hist, hnum);
- }
- node->dead = true;
- root_ = child;
- while (child > INIDBASE) {
- node = load_inner_node(child);
- if (!node) {
- set_error(_KCCODELINE_, Error::BROKEN, "missing inner node");
- db_.report(_KCCODELINE_, Logger::WARN, "id=%lld", (long long)child);
- return false;
- }
- if (node->dead) {
- child = node->heir;
- root_ = child;
- } else {
- child = 0;
- }
- }
- return false;
- }
- while (lit != litend) {
- Link* link = *lit;
- if (link->child == child) {
- xfree(link);
- links.erase(lit);
- return true;
- }
- ++lit;
- }
- set_error(_KCCODELINE_, Error::BROKEN, "invalid tree");
- return false;
- }
- /**
- * Dump the meta data into the file.
- * @return true on success, or false on failure.
- */
- bool dump_meta() {
- _assert_(true);
- char head[HEADSIZ];
- std::memset(head, 0, sizeof(head));
- char* wp = head;
- if (reccomp_.comp == LEXICALCOMP) {
- *(uint8_t*)(wp++) = 0x10;
- } else if (reccomp_.comp == DECIMALCOMP) {
- *(uint8_t*)(wp++) = 0x11;
- } else if (reccomp_.comp == LEXICALDESCCOMP) {
- *(uint8_t*)(wp++) = 0x18;
- } else if (reccomp_.comp == DECIMALDESCCOMP) {
- *(uint8_t*)(wp++) = 0x19;
- } else {
- *(uint8_t*)(wp++) = 0xff;
- }
- wp = head + MOFFNUMS;
- uint64_t num = hton64(psiz_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(root_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(first_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(last_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(lcnt_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(icnt_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(count_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- num = hton64(bnum_);
- std::memcpy(wp, &num, sizeof(num));
- wp += sizeof(num);
- std::memcpy(wp, "\x0a\x42\x6f\x6f\x66\x79\x21\x0a", sizeof(num));
- wp += sizeof(num);
- if (!db_.set(KCPDBMETAKEY, sizeof(KCPDBMETAKEY) - 1, head, sizeof(head))) return false;
- trlcnt_ = lcnt_;
- trcount_ = count_;
- return true;
- }
- /**
- * Load the meta data from the file.
- * @return true on success, or false on failure.
- */
- bool load_meta() {
- _assert_(true);
- char head[HEADSIZ];
- int32_t hsiz = db_.get(KCPDBMETAKEY, sizeof(KCPDBMETAKEY) - 1, head, sizeof(head));
- if (hsiz < 0) return false;
- if (hsiz != sizeof(head)) {
- set_error(_KCCODELINE_, Error::BROKEN, "invalid meta data record");
- db_.report(_KCCODELINE_, Logger::WARN, "hsiz=%d", hsiz);
- return false;
- }
- const char* rp = head;
- if (*(uint8_t*)rp == 0x10) {
- reccomp_.comp = LEXICALCOMP;
- linkcomp_.comp = LEXICALCOMP;
- } else if (*(uint8_t*)rp == 0x11) {
- reccomp_.comp = DECIMALCOMP;
- linkcomp_.comp = DECIMALCOMP;
- } else if (*(uint8_t*)rp == 0x18) {
- reccomp_.comp = LEXICALDESCCOMP;
- linkcomp_.comp = LEXICALDESCCOMP;
- } else if (*(uint8_t*)rp == 0x19) {
- reccomp_.comp = DECIMALDESCCOMP;
- linkcomp_.comp = DECIMALDESCCOMP;
- } else if (*(uint8_t*)rp == 0xff) {
- if (!reccomp_.comp) {
- set_error(_KCCODELINE_, Error::INVALID, "the custom comparator is not given");
- return false;
- }
- linkcomp_.comp = reccomp_.comp;
- } else {
- set_error(_KCCODELINE_, Error::BROKEN, "comparator is invalid");
- return false;
- }
- rp = head + MOFFNUMS;
- uint64_t num;
- std::memcpy(&num, rp, sizeof(num));
- psiz_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- root_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- first_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- last_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- lcnt_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- icnt_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- count_ = ntoh64(num);
- rp += sizeof(num);
- std::memcpy(&num, rp, sizeof(num));
- bnum_ = ntoh64(num);
- rp += sizeof(num);
- trlcnt_ = lcnt_;
- trcount_ = count_;
- return true;
- }
- /**
- * Caluculate the total number of nodes in the leaf cache.
- * @return the total number of nodes in the leaf cache.
- */
- int64_t calc_leaf_cache_count() {
- _assert_(true);
- int64_t sum = 0;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- LeafSlot* slot = lslots_ + i;
- sum += slot->warm->count();
- sum += slot->hot->count();
- }
- return sum;
- }
- /**
- * Caluculate the amount of memory usage of the leaf cache.
- * @return the amount of memory usage of the leaf cache.
- */
- int64_t calc_leaf_cache_size() {
- _assert_(true);
- int64_t sum = 0;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- LeafSlot* slot = lslots_ + i;
- typename LeafCache::Iterator it = slot->warm->begin();
- typename LeafCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- LeafNode* node = it.value();
- sum += node->size;
- ++it;
- }
- it = slot->hot->begin();
- itend = slot->hot->end();
- while (it != itend) {
- LeafNode* node = it.value();
- sum += node->size;
- ++it;
- }
- }
- return sum;
- }
- /**
- * Caluculate the total number of nodes in the inner cache.
- * @return the total number of nodes in the inner cache.
- */
- int64_t calc_inner_cache_count() {
- _assert_(true);
- int64_t sum = 0;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- InnerSlot* slot = islots_ + i;
- sum += slot->warm->count();
- }
- return sum;
- }
- /**
- * Caluculate the amount of memory usage of the inner cache.
- * @return the amount of memory usage of the inner cache.
- */
- int64_t calc_inner_cache_size() {
- _assert_(true);
- int64_t sum = 0;
- for (int32_t i = 0; i < SLOTNUM; i++) {
- InnerSlot* slot = islots_ + i;
- typename InnerCache::Iterator it = slot->warm->begin();
- typename InnerCache::Iterator itend = slot->warm->end();
- while (it != itend) {
- InnerNode* node = it.value();
- sum += node->size;
- ++it;
- }
- }
- return sum;
- }
- /**
- * Disable all cursors.
- */
- void disable_cursors() {
- _assert_(true);
- if (curs_.empty()) return;
- typename CursorList::const_iterator cit = curs_.begin();
- typename CursorList::const_iterator citend = curs_.end();
- while (cit != citend) {
- Cursor* cur = *cit;
- if (cur->kbuf_) cur->clear_position();
- ++cit;
- }
- }
- /**
- * Escape cursors on a divided leaf node.
- * @param src the ID of the source node.
- * @param dest the ID of the destination node.
- * @param rec the pivot record.
- * @return true on success, or false on failure.
- */
- void escape_cursors(int64_t src, int64_t dest, Record* rec) {
- _assert_(src > 0 && dest >= 0 && rec);
- if (curs_.empty()) return;
- typename CursorList::const_iterator cit = curs_.begin();
- typename CursorList::const_iterator citend = curs_.end();
- while (cit != citend) {
- Cursor* cur = *cit;
- if (cur->lid_ == src) {
- char* dbuf = (char*)rec + sizeof(*rec);
- if (reccomp_.comp->compare(cur->kbuf_, cur->ksiz_, dbuf, rec->ksiz) >= 0)
- cur->lid_ = dest;
- }
- ++cit;
- }
- }
- /**
- * Escape cursors on a removed leaf node.
- * @param src the ID of the source node.
- * @param dest the ID of the destination node.
- * @return true on success, or false on failure.
- */
- bool escape_cursors(int64_t src, int64_t dest) {
- _assert_(src > 0 && dest >= 0);
- if (curs_.empty()) return true;
- bool err = false;
- typename CursorList::const_iterator cit = curs_.begin();
- typename CursorList::const_iterator citend = curs_.end();
- while (cit != citend) {
- Cursor* cur = *cit;
- if (cur->lid_ == src) {
- cur->clear_position();
- if (!cur->set_position(dest) && db_.error().code() != Error::NOREC) err = true;
- }
- ++cit;
- }
- return !err;
- }
- /**
- * Recalculate the count data.
- * @return true on success, or false on failure.
- */
- bool recalc_count() {
- _assert_(true);
- if (!load_meta()) return false;
- bool err = false;
- std::set<int64_t> ids;
- std::set<int64_t> prevs;
- std::set<int64_t> nexts;
- class VisitorImpl : public DB::Visitor {
- public:
- explicit VisitorImpl(std::set<int64_t>* ids,
- std::set<int64_t>* prevs, std::set<int64_t>* nexts) :
- ids_(ids), prevs_(prevs), nexts_(nexts), count_(0) {}
- int64_t count() {
- return count_;
- }
- private:
- const char* visit_full(const char* kbuf, size_t ksiz,
- const char* vbuf, size_t vsiz, size_t* sp) {
- if (ksiz < 2 || ksiz >= NUMBUFSIZ || kbuf[0] != LNPREFIX) return NOP;
- kbuf++;
- ksiz--;
- char tkbuf[NUMBUFSIZ];
- std::memcpy(tkbuf, kbuf, ksiz);
- tkbuf[ksiz] = '\0';
- int64_t id = atoih(tkbuf);
- uint64_t prev;
- size_t step = readvarnum(vbuf, vsiz, &prev);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- uint64_t next;
- step = readvarnum(vbuf, vsiz, &next);
- if (step < 1) return NOP;
- vbuf += step;
- vsiz -= step;
- ids_->insert(id);
- if (prev > 0) prevs_->insert(prev);
- if (next > 0) nexts_->insert(next);
- while (vsiz > 1) {
- uint64_t rksiz;
- step = readvarnum(vbuf, vsiz, &rksiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- uint64_t rvsiz;
- step = readvarnum(vbuf, vsiz, &rvsiz);
- if (step < 1) break;
- vbuf += step;
- vsiz -= step;
- if (vsiz < rksiz + rvsiz) break;
- vbuf += rksiz;
- vsiz -= rksiz;
- vbuf += rvsiz;
- vsiz -= rvsiz;
- count_++;
- }
- return NOP;
- }
- std::set<int64_t>* ids_;
- std::set<int64_t>* prevs_;
- std::set<int64_t>* nexts_;
- int64_t count_;
- } visitor(&ids, &prevs, &nexts);
- if (!db_.iterate(&visitor, false)) err = true;
- int64_t count = visitor.count();
- db_.report(_KCCODELINE_, Logger::WARN, "recalculated the record count from %lld to %lld",
- (long long)count_, (long long)count);
- std::set<int64_t>::iterator iitend = ids.end();
- std::set<int64_t>::iterator nit = nexts.begin();
- std::set<int64_t>::iterator nitend = nexts.end();
- while (nit != nitend) {
- if (ids.find(*nit) == ids.end()) {
- db_.report(_KCCODELINE_, Logger::WARN, "detected missing leaf: %lld", (long long)*nit);
- count = INT64MAX;
- }
- ++nit;
- }
- std::set<int64_t>::iterator pit = prevs.begin();
- std::set<int64_t>::iterator pitend = prevs.end();
- while (pit != pitend) {
- if (ids.find(*pit) == iitend) {
- db_.report(_KCCODELINE_, Logger::WARN, "detected missing leaf: %lld", (long long)*pit);
- count = INT64MAX;
- }
- ++pit;
- }
- count_ = count;
- if (!dump_meta()) err = true;
- return !err;
- }
- /**
- * Reorganize the database file.
- * @param mode the connection mode of the internal database.
- * @return true on success, or false on failure.
- */
- bool reorganize_file(uint32_t mode) {
- _assert_(true);
- if (!load_meta()) {
- if (reccomp_.comp) {
- linkcomp_.comp = reccomp_.comp;
- } else {
- reccomp_.comp = LEXICALCOMP;
- linkcomp_.comp = LEXICALCOMP;
- }
- }
- const std::string& path = db_.path();
- const std::string& npath = path + File::EXTCHR + KCPDBTMPPATHEXT;
- PlantDB tdb;
- tdb.tune_comparator(reccomp_.comp);
- if (!tdb.open(npath, OWRITER | OCREATE | OTRUNCATE)) {
- set_error(_KCCODELINE_, tdb.error().code(), "opening the destination failed");
- return false;
- }
- db_.report(_KCCODELINE_, Logger::WARN, "reorganizing the database");
- bool err = false;
- create_leaf_cache();
- create_inner_cache();
- DB::Cursor* cur = db_.cursor();
- cur->jump();
- char* kbuf;
- size_t ksiz;
- while (!err && (kbuf = cur->get_key(&ksiz)) != NULL) {
- if (*kbuf == LNPREFIX) {
- int64_t id = std::strtol(kbuf + 1, NULL, 16);
- if (id > 0 && id < INIDBASE) {
- LeafNode* node = load_leaf_node(id, false);
- if (node) {
- const RecordArray& recs = node->recs;
- typename RecordArray::const_iterator rit = recs.begin();
- typename RecordArray::const_iterator ritend = recs.end();
- while (rit != ritend) {
- Record* rec = *rit;
- char* dbuf = (char*)rec + sizeof(*rec);
- if (!tdb.set(dbuf, rec->ksiz, dbuf + rec->ksiz, rec->vsiz)) {
- set_error(_KCCODELINE_, tdb.error().code(),
- "opening the destination failed");
- err = true;
- }
- ++rit;
- }
- flush_leaf_node(node, false);
- }
- }
- }
- delete[] kbuf;
- cur->step();
- }
- delete cur;
- delete_inner_cache();
- delete_leaf_cache();
- if (!tdb.close()) {
- set_error(_KCCODELINE_, tdb.error().code(), "opening the destination failed");
- err = true;
- }
- if (DBTYPE == TYPETREE) {
- if (File::rename(npath, path)) {
- if (!db_.close()) err = true;
- if (!db_.open(path, mode)) err = true;
- } else {
- set_error(_KCCODELINE_, Error::SYSTEM, "renaming the destination failed");
- err = true;
- }
- File::remove(npath);
- } else if (DBTYPE == TYPEFOREST) {
- const std::string& tpath = npath + File::EXTCHR + KCPDBTMPPATHEXT;
- File::remove_recursively(tpath);
- if (File::rename(path, tpath)) {
- if (File::rename(npath, path)) {
- if (!db_.close()) err = true;
- if (!db_.open(path, mode)) err = true;
- } else {
- set_error(_KCCODELINE_, Error::SYSTEM, "renaming the destination failed");
- File::rename(tpath, path);
- err = true;
- }
- } else {
- set_error(_KCCODELINE_, Error::SYSTEM, "renaming the source failed");
- err = true;
- }
- File::remove_recursively(tpath);
- File::remove_recursively(npath);
- } else {
- BASEDB udb;
- if (!err && udb.open(npath, OREADER)) {
- if (writer_) {
- if (!db_.clear()) err = true;
- } else {
- if (!db_.close()) err = true;
- uint32_t tmode = (mode & ~OREADER) | OWRITER | OCREATE | OTRUNCATE;
- if (!db_.open(path, tmode)) err = true;
- }
- cur = udb.cursor();
- cur->jump();
- const char* vbuf;
- size_t vsiz;
- while (!err && (kbuf = cur->get(&ksiz, &vbuf, &vsiz)) != NULL) {
- if (!db_.set(kbuf, ksiz, vbuf, vsiz)) err = true;
- delete[] kbuf;
- cur->step();
- }
- delete cur;
- if (writer_) {
- if (!db_.synchronize(false, NULL)) err = true;
- } else {
- if (!db_.close()) err = true;
- if (!db_.open(path, mode)) err = true;
- }
- if (!udb.close()) {
- set_error(_KCCODELINE_, udb.error().code(), "closing the destination failed");
- err = true;
- }
- } else {
- set_error(_KCCODELINE_, udb.error().code(), "opening the destination failed");
- err = true;
- }
- File::remove_recursively(npath);
- }
- return !err;
- }
- /**
- * Begin transaction.
- * @param hard true for physical synchronization with the device, or false for logical
- * synchronization with the file system.
- * @return true on success, or false on failure.
- */
- bool begin_transaction_impl(bool hard) {
- _assert_(true);
- if (!clean_leaf_cache()) return false;
- if (!clean_inner_cache()) return false;
- int32_t idx = trclock_++ % SLOTNUM;
- LeafSlot* lslot = lslots_ + idx;
- if (lslot->warm->count() + lslot->hot->count() > 1) flush_leaf_cache_part(lslot);
- InnerSlot* islot = islots_ + idx;
- if (islot->warm->count() > 1) flush_inner_cache_part(islot);
- if ((trlcnt_ != lcnt_ || count_ != trcount_) && !dump_meta()) return false;
- if (!db_.begin_transaction(hard)) return false;
- return true;
- }
- /**
- * Commit transaction.
- * @return true on success, or false on failure.
- */
- bool commit_transaction() {
- _assert_(true);
- bool err = false;
- if (!clean_leaf_cache()) return false;
- if (!clean_inner_cache()) return false;
- if ((trlcnt_ != lcnt_ || count_ != trcount_) && !dump_meta()) err = true;
- if (!db_.end_transaction(true)) return false;
- return !err;
- }
- /**
- * Abort transaction.
- * @return true on success, or false on failure.
- */
- bool abort_transaction() {
- _assert_(true);
- bool err = false;
- flush_leaf_cache(false);
- flush_inner_cache(false);
- if (!db_.end_transaction(false)) err = true;
- if (!load_meta()) err = true;
- disable_cursors();
- return !err;
- }
- /**
- * Fix auto transaction for the B+ tree.
- * @return true on success, or false on failure.
- */
- bool fix_auto_transaction_tree() {
- _assert_(true);
- if (!db_.begin_transaction(autosync_)) return false;
- bool err = false;
- if (!clean_leaf_cache()) err = true;
- if (!clean_inner_cache()) err = true;
- size_t cnum = ATRANCNUM / SLOTNUM;
- int32_t idx = trclock_++ % SLOTNUM;
- LeafSlot* lslot = lslots_ + idx;
- if (lslot->warm->count() + lslot->hot->count() > cnum) flush_leaf_cache_part(lslot);
- InnerSlot* islot = islots_ + idx;
- if (islot->warm->count() > cnum) flush_inner_cache_part(islot);
- if (!dump_meta()) err = true;
- if (!db_.end_transaction(true)) err = true;
- return !err;
- }
- /**
- * Fix auto transaction for a leaf.
- * @return true on success, or false on failure.
- */
- bool fix_auto_transaction_leaf(LeafNode* node) {
- _assert_(node);
- bool err = false;
- if (!save_leaf_node(node)) err = true;
- return !err;
- }
- /**
- * Fix auto synchronization.
- * @return true on success, or false on failure.
- */
- bool fix_auto_synchronization() {
- _assert_(true);
- bool err = false;
- if (!flush_leaf_cache(true)) err = true;
- if (!flush_inner_cache(true)) err = true;
- if (!dump_meta()) err = true;
- if (!db_.synchronize(true, NULL)) err = true;
- return !err;
- }
- /**
- * Write the key pattern into a buffer.
- * @param kbuf the destination buffer.
- * @param pc the prefix character.
- * @param id the ID number of the page.
- * @return the size of the key pattern.
- */
- size_t write_key(char* kbuf, int32_t pc, int64_t num) {
- _assert_(kbuf && num >= 0);
- char* wp = kbuf;
- *(wp++) = pc;
- bool hit = false;
- for (size_t i = 0; i < sizeof(num); i++) {
- uint8_t c = num >> ((sizeof(num) - 1 - i) * 8);
- uint8_t h = c >> 4;
- if (h < 10) {
- if (hit || h != 0) {
- *(wp++) = '0' + h;
- hit = true;
- }
- } else {
- *(wp++) = 'A' - 10 + h;
- hit = true;
- }
- uint8_t l = c & 0xf;
- if (l < 10) {
- if (hit || l != 0) {
- *(wp++) = '0' + l;
- hit = true;
- }
- } else {
- *(wp++) = 'A' - 10 + l;
- hit = true;
- }
- }
- return wp - kbuf;
- }
- /** Dummy constructor to forbid the use. */
- PlantDB(const PlantDB&);
- /** Dummy Operator to forbid the use. */
- PlantDB& operator =(const PlantDB&);
- /** The internal meta operation trigger. */
- MetaTrigger* mtrigger_;
- /** The open mode. */
- uint32_t omode_;
- /** The flag for writer. */
- bool writer_;
- /** The flag for auto transaction. */
- bool autotran_;
- /** The flag for auto synchronization. */
- bool autosync_;
- /** The internal database. */
- BASEDB db_;
- /** The cursor objects. */
- CursorList curs_;
- /** The alignment power. */
- uint8_t apow_;
- /** The free block pool power. */
- uint8_t fpow_;
- /** The options. */
- uint8_t opts_;
- /** The bucket number. */
- int64_t bnum_;
- /** The page size. */
- int32_t psiz_;
- /** The capacity of page cache. */
- int64_t pccap_;
- /** The root node. */
- int64_t root_;
- /** The first node. */
- int64_t first_;
- /** The last node. */
- int64_t last_;
- /** The count of leaf nodes. */
- int64_t lcnt_;
- /** The count of inner nodes. */
- int64_t icnt_;
- /** The record number. */
- AtomicInt64 count_;
- /** The cache memory usage. */
- AtomicInt64 cusage_;
- /** The Slots of leaf nodes. */
- LeafSlot lslots_[SLOTNUM];
- /** The Slots of inner nodes. */
- InnerSlot islots_[SLOTNUM];
- /** The record comparator. */
- RecordComparator reccomp_;
- /** The link comparator. */
- LinkComparator linkcomp_;
- /** The flag whether in transaction. */
- bool tran_;
- /** The logical clock for transaction. */
- int64_t trclock_;
- /** The leaf count history for transaction. */
- int64_t trlcnt_;
- /** The record count history for transaction. */
- int64_t trcount_;
-};
-
-
-} // common namespace
-
-#endif // duplication check
-
-// END OF FILE