diff options
Diffstat (limited to 'plugins/Dbx_kyoto/src/kyotocabinet/kclangc.cc')
-rw-r--r-- | plugins/Dbx_kyoto/src/kyotocabinet/kclangc.cc | 1534 |
1 files changed, 1534 insertions, 0 deletions
diff --git a/plugins/Dbx_kyoto/src/kyotocabinet/kclangc.cc b/plugins/Dbx_kyoto/src/kyotocabinet/kclangc.cc new file mode 100644 index 0000000000..6b3a894d5c --- /dev/null +++ b/plugins/Dbx_kyoto/src/kyotocabinet/kclangc.cc @@ -0,0 +1,1534 @@ +/************************************************************************************************* + * C language binding + * 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/>. + *************************************************************************************************/ + + +#include "kcpolydb.h" +#include "kcdbext.h" +#include "kclangc.h" +#include "myconf.h" + +using namespace kyotocabinet; + +extern "C" { + + +/** The package version. */ +const char* const KCVERSION = VERSION; + + +/** Special pointer for no operation by the visiting function. */ +const char* const KCVISNOP = DB::Visitor::NOP; + + +/** Special pointer to remove the record by the visiting function. */ +const char* const KCVISREMOVE = DB::Visitor::REMOVE; + + +/** + * Allocate a region on memory. + */ +void* kcmalloc(size_t size) { + _assert_(size > 0 && size <= MEMMAXSIZ); + return new char[size]; +} + + +/** + * Release a region allocated in the library. + */ +void kcfree(void* ptr) { + _assert_(true); + delete[] (char*)ptr; +} + + +/** + * Get the time of day in seconds. + */ +double kctime(void) { + _assert_(true); + return kyotocabinet::time(); +} + + +/** + * Convert a string to an integer. + */ +int64_t kcatoi(const char* str) { + _assert_(str); + return kyotocabinet::atoi(str); +} + + +/** + * Convert a string with a metric prefix to an integer. + */ +int64_t kcatoix(const char* str) { + _assert_(str); + return kyotocabinet::atoix(str); +} + + +/** + * Convert a string to a real number. + */ +double kcatof(const char* str) { + _assert_(str); + return kyotocabinet::atof(str); +} + + +/** + * Get the hash value by MurMur hashing. + */ +uint64_t kchashmurmur(const void* buf, size_t size) { + _assert_(buf && size <= MEMMAXSIZ); + return kyotocabinet::hashmurmur(buf, size); +} + + +/** + * Get the hash value by FNV hashing. + */ +uint64_t kchashfnv(const void* buf, size_t size) { + _assert_(buf && size <= MEMMAXSIZ); + return kyotocabinet::hashfnv(buf, size); +} + + +/** + * Calculate the levenshtein distance of two regions. + */ +size_t kclevdist(const void* abuf, size_t asiz, const void* bbuf, size_t bsiz, int32_t utf) { + _assert_(abuf && asiz <= MEMMAXSIZ && bbuf && bsiz <= MEMMAXSIZ); + size_t dist; + if (utf) { + uint32_t astack[128]; + uint32_t* aary = asiz > sizeof(astack) / sizeof(*astack) ? new uint32_t[asiz] : astack; + size_t anum; + strutftoucs((const char*)abuf, asiz, aary, &anum); + uint32_t bstack[128]; + uint32_t* bary = bsiz > sizeof(bstack) / sizeof(*bstack) ? new uint32_t[bsiz] : bstack; + size_t bnum; + strutftoucs((const char*)bbuf, bsiz, bary, &bnum); + dist = strucsdist(aary, anum, bary, bnum); + if (bary != bstack) delete[] bary; + if (aary != astack) delete[] aary; + } else { + dist = memdist(abuf, asiz, bbuf, bsiz); + } + return dist; +} + + +/** + * Get the quiet Not-a-Number value. + */ +double kcnan() { + _assert_(true); + return kyotocabinet::nan(); +} + + +/** + * Get the positive infinity value. + */ +double kcinf() { + _assert_(true); + return kyotocabinet::inf(); +} + + +/** + * Check a number is a Not-a-Number value. + */ +int32_t kcchknan(double num) { + _assert_(true); + return kyotocabinet::chknan(num); +} + + +/** + * Check a number is an infinity value. + */ +int32_t kcchkinf(double num) { + _assert_(true); + return kyotocabinet::chkinf(num); +} + + +/** + * Get the readable string of an error code. + */ +const char* kcecodename(int32_t code) { + _assert_(true); + return BasicDB::Error::codename((BasicDB::Error::Code)code); +} + + +/** + * Create a database object. + */ +KCDB* kcdbnew(void) { + _assert_(true); + return (KCDB*)new PolyDB; +} + + +/** + * Destroy a database object. + */ +void kcdbdel(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + delete pdb; +} + + +/** + * Open a database file. + */ +int32_t kcdbopen(KCDB* db, const char* path, uint32_t mode) { + _assert_(db && path); + PolyDB* pdb = (PolyDB*)db; + return pdb->open(path, mode); +} + + +/** + * Close the database file. + */ +int32_t kcdbclose(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->close(); +} + + +/** + * Get the code of the last happened error. + */ +int32_t kcdbecode(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->error().code(); +} + + +/** + * Get the supplement message of the last happened error. + */ +const char* kcdbemsg(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->error().message(); +} + + +/** + * Accept a visitor to a record. + */ +int32_t kcdbaccept(KCDB* db, const char* kbuf, size_t ksiz, + KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq, int32_t writable) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + class VisitorImpl : public DB::Visitor { + public: + explicit VisitorImpl(KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq) : + fullproc_(fullproc), emptyproc_(emptyproc), opq_(opq) {} + const char* visit_full(const char* kbuf, size_t ksiz, + const char* vbuf, size_t vsiz, size_t* sp) { + if (!fullproc_) return NOP; + return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_); + } + const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) { + if (!emptyproc_) return NOP; + return emptyproc_(kbuf, ksiz, sp, opq_); + } + private: + KCVISITFULL fullproc_; + KCVISITEMPTY emptyproc_; + void* opq_; + }; + VisitorImpl visitor(fullproc, emptyproc, opq); + return pdb->accept(kbuf, ksiz, &visitor, writable); +} + + +/** + * Accept a visitor to multiple records at once. + */ +int32_t kcdbacceptbulk(KCDB* db, const KCSTR* keys, size_t knum, + KCVISITFULL fullproc, KCVISITEMPTY emptyproc, + void* opq, int32_t writable) { + _assert_(db && keys && knum <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> xkeys; + xkeys.reserve(knum); + for (size_t i = 0; i < knum; i++) { + xkeys.push_back(std::string(keys[i].buf, keys[i].size)); + } + class VisitorImpl : public DB::Visitor { + public: + explicit VisitorImpl(KCVISITFULL fullproc, KCVISITEMPTY emptyproc, void* opq) : + fullproc_(fullproc), emptyproc_(emptyproc), opq_(opq) {} + const char* visit_full(const char* kbuf, size_t ksiz, + const char* vbuf, size_t vsiz, size_t* sp) { + if (!fullproc_) return NOP; + return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_); + } + const char* visit_empty(const char* kbuf, size_t ksiz, size_t* sp) { + if (!emptyproc_) return NOP; + return emptyproc_(kbuf, ksiz, sp, opq_); + } + private: + KCVISITFULL fullproc_; + KCVISITEMPTY emptyproc_; + void* opq_; + }; + VisitorImpl visitor(fullproc, emptyproc, opq); + return pdb->accept_bulk(xkeys, &visitor, writable); +} + + +/** + * Iterate to accept a visitor for each record. + */ +int32_t kcdbiterate(KCDB* db, KCVISITFULL fullproc, void* opq, int32_t writable) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + class VisitorImpl : public DB::Visitor { + public: + explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {} + const char* visit_full(const char* kbuf, size_t ksiz, + const char* vbuf, size_t vsiz, size_t* sp) { + if (!fullproc_) return NOP; + return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_); + } + private: + KCVISITFULL fullproc_; + void* opq_; + }; + VisitorImpl visitor(fullproc, opq); + return pdb->iterate(&visitor, writable); +} + + +/** + * Scan each record in parallel. + */ +int32_t kcdbscanpara(KCDB* db, KCVISITFULL fullproc, void* opq, size_t thnum) { + _assert_(db && thnum <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + class VisitorImpl : public DB::Visitor { + public: + explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {} + const char* visit_full(const char* kbuf, size_t ksiz, + const char* vbuf, size_t vsiz, size_t* sp) { + if (!fullproc_) return NOP; + return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_); + } + private: + KCVISITFULL fullproc_; + void* opq_; + }; + VisitorImpl visitor(fullproc, opq); + return pdb->scan_parallel(&visitor, thnum); +} + + +/** + * Set the value of a record. + */ +int32_t kcdbset(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->set(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Add a record. + */ +int32_t kcdbadd(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->add(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Replace the value of a record. + */ +int32_t kcdbreplace(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->replace(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Append the value of a record. + */ +int32_t kcdbappend(KCDB* db, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->append(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Add a number to the numeric value of a record. + */ +int64_t kcdbincrint(KCDB* db, const char* kbuf, size_t ksiz, int64_t num, int64_t orig) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->increment(kbuf, ksiz, num, orig); +} + + +/** + * Add a number to the numeric value of a record. + */ +double kcdbincrdouble(KCDB* db, const char* kbuf, size_t ksiz, double num, double orig) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->increment_double(kbuf, ksiz, num, orig); +} + + +/** + * Perform compare-and-swap. + */ +int32_t kcdbcas(KCDB* db, const char* kbuf, size_t ksiz, + const char* ovbuf, size_t ovsiz, const char* nvbuf, size_t nvsiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->cas(kbuf, ksiz, ovbuf, ovsiz, nvbuf, nvsiz); +} + + +/** + * Remove a record. + */ +int32_t kcdbremove(KCDB* db, const char* kbuf, size_t ksiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->remove(kbuf, ksiz); +} + + +/** + * Retrieve the value of a record. + */ +char* kcdbget(KCDB* db, const char* kbuf, size_t ksiz, size_t* sp) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && sp); + PolyDB* pdb = (PolyDB*)db; + return pdb->get(kbuf, ksiz, sp); +} + + +/** + * Check the existence of a record. + */ +int32_t kcdbcheck(KCDB* db, const char* kbuf, size_t ksiz) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->check(kbuf, ksiz); +} + + +/** + * Retrieve the value of a record. + */ +int32_t kcdbgetbuf(KCDB* db, const char* kbuf, size_t ksiz, char* vbuf, size_t max) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && vbuf); + PolyDB* pdb = (PolyDB*)db; + return pdb->get(kbuf, ksiz, vbuf, max); +} + + +/** + * Retrieve the value of a record and remove it atomically. + */ +char* kcdbseize(KCDB* db, const char* kbuf, size_t ksiz, size_t* sp) { + _assert_(db && kbuf && ksiz <= MEMMAXSIZ && sp); + PolyDB* pdb = (PolyDB*)db; + return pdb->seize(kbuf, ksiz, sp); +} + + +/** + * Store records at once. + */ +int64_t kcdbsetbulk(KCDB* db, const KCREC* recs, size_t rnum, int32_t atomic) { + _assert_(db && recs && rnum <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::map<std::string, std::string> xrecs; + for (size_t i = 0; i < rnum; i++) { + const KCREC* rec = recs + i; + xrecs[std::string(rec->key.buf, rec->key.size)] = + std::string(rec->value.buf, rec->value.size); + } + return pdb->set_bulk(xrecs, atomic); +} + + +/** + * Remove records at once. + */ +int64_t kcdbremovebulk(KCDB* db, const KCSTR* keys, size_t knum, int32_t atomic) { + _assert_(db && keys && knum <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> xkeys; + xkeys.reserve(knum); + for (size_t i = 0; i < knum; i++) { + const KCSTR* key = keys + i; + xkeys.push_back(std::string(key->buf, key->size)); + } + return pdb->remove_bulk(xkeys, atomic); +} + + +/** + * Retrieve records at once. + */ +int64_t kcdbgetbulk(KCDB* db, const KCSTR* keys, size_t knum, KCREC* recs, int32_t atomic) { + _assert_(db && keys && knum <= MEMMAXSIZ && recs); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> xkeys; + xkeys.reserve(knum); + for (size_t i = 0; i < knum; i++) { + const KCSTR* key = keys + i; + xkeys.push_back(std::string(key->buf, key->size)); + } + std::map<std::string, std::string> xrecs; + if (pdb->get_bulk(xkeys, &xrecs, atomic) < 0) return -1; + std::map<std::string, std::string>::iterator it = xrecs.begin(); + std::map<std::string, std::string>::iterator itend = xrecs.end(); + size_t ridx = 0; + while (ridx < knum && it != itend) { + size_t ksiz = it->first.size(); + char* kbuf = new char[ksiz+1]; + std::memcpy(kbuf, it->first.data(), ksiz); + kbuf[ksiz] = '\0'; + size_t vsiz = it->second.size(); + char* vbuf = new char[vsiz+1]; + std::memcpy(vbuf, it->second.data(), vsiz); + vbuf[vsiz] = '\0'; + KCREC* rec = recs + (ridx++); + rec->key.buf = kbuf; + rec->key.size = ksiz; + rec->value.buf = vbuf; + rec->value.size = vsiz; + ++it; + } + return ridx; +} + + +/** + * Synchronize updated contents with the file and the device. + */ +int32_t kcdbsync(KCDB* db, int32_t hard, KCFILEPROC proc, void* opq) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + class FileProcessorImpl : public BasicDB::FileProcessor { + public: + explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {} + bool process(const std::string& path, int64_t count, int64_t size) { + if (!proc_) return true; + return proc_(path.c_str(), count, size, opq_); + } + private: + KCFILEPROC proc_; + void* opq_; + }; + FileProcessorImpl myproc(proc, opq); + return pdb->synchronize(hard, &myproc); +} + + +/** + * Occupy database by locking and do something meanwhile. + */ +int32_t kcdboccupy(KCDB* db, int32_t writable, KCFILEPROC proc, void* opq) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + class FileProcessorImpl : public BasicDB::FileProcessor { + public: + explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {} + bool process(const std::string& path, int64_t count, int64_t size) { + if (!proc_) return true; + return proc_(path.c_str(), count, size, opq_); + } + private: + KCFILEPROC proc_; + void* opq_; + }; + FileProcessorImpl myproc(proc, opq); + return pdb->occupy(writable, &myproc); +} + + +/** + * Create a copy of the database file. + */ +int32_t kcdbcopy(KCDB* db, const char* dest) { + _assert_(db && dest); + PolyDB* pdb = (PolyDB*)db; + return pdb->copy(dest); +} + + +/** + * Begin transaction. + */ +int32_t kcdbbegintran(KCDB* db, int32_t hard) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->begin_transaction(hard); +} + + +/** + * Try to begin transaction. + */ +int32_t kcdbbegintrantry(KCDB* db, int32_t hard) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->begin_transaction_try(hard); +} + + +/** + * End transaction. + */ +int32_t kcdbendtran(KCDB* db, int32_t commit) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->end_transaction(commit); +} + + +/** + * Remove all records. + */ +int32_t kcdbclear(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->clear(); +} + + +/** + * Dump records into a file. + */ +int32_t kcdbdumpsnap(KCDB* db, const char* dest) { + _assert_(db && dest); + PolyDB* pdb = (PolyDB*)db; + return pdb->dump_snapshot(dest); +} + + +/** + * Load records from a file. + */ +int32_t kcdbloadsnap(KCDB* db, const char* src) { + _assert_(db && src); + PolyDB* pdb = (PolyDB*)db; + return pdb->load_snapshot(src); +} + + +/** + * Get the number of records. + */ +int64_t kcdbcount(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->count(); +} + + +/** + * Get the size of the database file. + */ +int64_t kcdbsize(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return pdb->size(); +} + + +/** + * Get the path of the database file. + */ +char* kcdbpath(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + std::string path = pdb->path(); + size_t psiz = path.size(); + char* pbuf = new char[psiz+1]; + std::memcpy(pbuf, path.c_str(), psiz + 1); + return pbuf; +} + + +/** + * Get the miscellaneous status information. + */ +char* kcdbstatus(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + std::map<std::string, std::string> status; + if (!pdb->status(&status)) return NULL; + std::ostringstream obuf; + std::map<std::string, std::string>::iterator it = status.begin(); + std::map<std::string, std::string>::iterator itend = status.end(); + while (it != itend) { + obuf << it->first << "\t" << it->second << "\n"; + ++it; + } + std::string sstr = obuf.str(); + size_t ssiz = sstr.size(); + char* sbuf = new char[ssiz+1]; + std::memcpy(sbuf, sstr.c_str(), ssiz + 1); + return sbuf; +} + + +/** + * Get keys matching a prefix string. + */ +int64_t kcdbmatchprefix(KCDB* db, const char* prefix, char** strary, size_t max) { + _assert_(db && prefix && strary && max <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> strvec; + if (pdb->match_prefix(prefix, &strvec, max) == -1) return -1; + int64_t cnt = 0; + std::vector<std::string>::iterator it = strvec.begin(); + std::vector<std::string>::iterator itend = strvec.end(); + while (it != itend) { + size_t ksiz = it->size(); + char* kbuf = new char[ksiz+1]; + std::memcpy(kbuf, it->data(), ksiz); + kbuf[ksiz] = '\0'; + strary[cnt++] = kbuf; + ++it; + } + return cnt; +} + + +/** + * Get keys matching a regular expression string. + */ +int64_t kcdbmatchregex(KCDB* db, const char* regex, char** strary, size_t max) { + _assert_(db && regex && strary && max <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> strvec; + if (pdb->match_regex(regex, &strvec, max) == -1) return -1; + int64_t cnt = 0; + std::vector<std::string>::iterator it = strvec.begin(); + std::vector<std::string>::iterator itend = strvec.end(); + while (it != itend) { + size_t ksiz = it->size(); + char* kbuf = new char[ksiz+1]; + std::memcpy(kbuf, it->data(), ksiz); + kbuf[ksiz] = '\0'; + strary[cnt++] = kbuf; + ++it; + } + return cnt; +} + + +/** + * Get keys similar to a string in terms of the levenshtein distance. + */ +int64_t kcdbmatchsimilar(KCDB* db, const char* origin, uint32_t range, int32_t utf, + char** strary, size_t max) { + _assert_(db && origin && strary && max <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + std::vector<std::string> strvec; + if (pdb->match_similar(origin, range, utf, &strvec, max) == -1) return -1; + int64_t cnt = 0; + std::vector<std::string>::iterator it = strvec.begin(); + std::vector<std::string>::iterator itend = strvec.end(); + while (it != itend) { + size_t ksiz = it->size(); + char* kbuf = new char[ksiz+1]; + std::memcpy(kbuf, it->data(), ksiz); + kbuf[ksiz] = '\0'; + strary[cnt++] = kbuf; + ++it; + } + return cnt; +} + + +/** + * Merge records from other databases. + */ +int32_t kcdbmerge(KCDB* db, KCDB** srcary, size_t srcnum, uint32_t mode) { + _assert_(db && srcary && srcnum <= MEMMAXSIZ); + PolyDB* pdb = (PolyDB*)db; + return pdb->merge((BasicDB**)srcary, srcnum, (PolyDB::MergeMode)mode); +} + + +/** + * Create a cursor object. + */ +KCCUR* kcdbcursor(KCDB* db) { + _assert_(db); + PolyDB* pdb = (PolyDB*)db; + return (KCCUR*)pdb->cursor(); +} + + +/** + * Destroy a cursor object. + */ +void kccurdel(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + delete pcur; +} + + +/** + * Accept a visitor to the current record. + */ +int32_t kccuraccept(KCCUR* cur, KCVISITFULL fullproc, void* opq, + int32_t writable, int32_t step) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + class VisitorImpl : public DB::Visitor { + public: + explicit VisitorImpl(KCVISITFULL fullproc, void* opq) : fullproc_(fullproc), opq_(opq) {} + const char* visit_full(const char* kbuf, size_t ksiz, + const char* vbuf, size_t vsiz, size_t* sp) { + if (!fullproc_) return NOP; + return fullproc_(kbuf, ksiz, vbuf, vsiz, sp, opq_); + } + private: + KCVISITFULL fullproc_; + void* opq_; + }; + VisitorImpl visitor(fullproc, opq); + return pcur->accept(&visitor, writable, step); +} + + +/** + * Set the value of the current record. + */ +int32_t kccursetvalue(KCCUR* cur, const char* vbuf, size_t vsiz, int32_t step) { + _assert_(cur && vbuf && vsiz <= MEMMAXSIZ); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->set_value(vbuf, vsiz, step); +} + + +/** + * Remove the current record. + */ +int32_t kccurremove(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->remove(); +} + + +/** + * Get the key of the current record. + */ +char* kccurgetkey(KCCUR* cur, size_t* sp, int32_t step) { + _assert_(cur && sp); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->get_key(sp, step); +} + + +/** + * Get the value of the current record. + */ +char* kccurgetvalue(KCCUR* cur, size_t* sp, int32_t step) { + _assert_(cur && sp); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->get_value(sp, step); +} + + +/** + * Get a pair of the key and the value of the current record. + */ +char* kccurget(KCCUR* cur, size_t* ksp, const char** vbp, size_t* vsp, int32_t step) { + _assert_(cur && ksp && vbp && vsp); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->get(ksp, vbp, vsp, step); +} + + +/** + * Get a pair of the key and the value of the current record and remove it atomically. + */ +char* kccurseize(KCCUR* cur, size_t* ksp, const char** vbp, size_t* vsp) { + _assert_(cur && ksp && vbp && vsp); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->seize(ksp, vbp, vsp); +} + + +/** + * Jump the cursor to the first record. + */ +int32_t kccurjump(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->jump(); +} + + +/** + * Jump the cursor to a record. + */ +int32_t kccurjumpkey(KCCUR* cur, const char* kbuf, size_t ksiz) { + _assert_(cur && kbuf && ksiz <= MEMMAXSIZ); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->jump(kbuf, ksiz); +} + + +/** + * Jump the cursor to the last record for backward scan. + */ +int32_t kccurjumpback(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->jump_back(); +} + + +/** + * Jump the cursor to a record for backward scan. + */ +int32_t kccurjumpbackkey(KCCUR* cur, const char* kbuf, size_t ksiz) { + _assert_(cur && kbuf && ksiz <= MEMMAXSIZ); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->jump_back(kbuf, ksiz); +} + + +/** + * Step the cursor to the next record. + */ +int32_t kccurstep(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->step(); +} + + +/** + * Step the cursor to the previous record. + */ +int32_t kccurstepback(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->step_back(); +} + + +/** + * Get the database object. + */ +KCDB* kccurdb(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return (KCDB*)pcur->db(); +} + + +/** + * Get the code of the last happened error. + */ +int32_t kccurecode(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->error().code(); +} + + +/** + * Get the supplement message of the last happened error. + */ +const char* kccuremsg(KCCUR* cur) { + _assert_(cur); + PolyDB::Cursor* pcur = (PolyDB::Cursor*)cur; + return pcur->error().message(); +} + + +/** + * Create an index database object. + */ +KCIDX* kcidxnew(void) { + _assert_(true); + return (KCIDX*)new IndexDB; +} + + +/** + * Destroy a database object. + */ +void kcidxdel(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + delete idb; +} + + +/** + * Open a database file. + */ +int32_t kcidxopen(KCIDX* idx, const char* path, uint32_t mode) { + _assert_(idx && path); + IndexDB* idb = (IndexDB*)idx; + return idb->open(path, mode); +} + + +/** + * Close the database file. + */ +int32_t kcidxclose(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->close(); +} + + +/** + * Get the code of the last happened error. + */ +int32_t kcidxecode(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->error().code(); +} + + +/** + * Get the supplement message of the last happened error. + */ +const char* kcidxemsg(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->error().message(); +} + + +/** + * Set the value of a record. + */ +int32_t kcidxset(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + IndexDB* idb = (IndexDB*)idx; + return idb->set(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Add a record. + */ +int32_t kcidxadd(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + IndexDB* idb = (IndexDB*)idx; + return idb->add(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Replace the value of a record. + */ +int32_t kcidxreplace(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + IndexDB* idb = (IndexDB*)idx; + return idb->replace(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Append the value of a record. + */ +int32_t kcidxappend(KCIDX* idx, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + IndexDB* idb = (IndexDB*)idx; + return idb->append(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Remove a record. + */ +int32_t kcidxremove(KCIDX* idx, const char* kbuf, size_t ksiz) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ); + IndexDB* idb = (IndexDB*)idx; + return idb->remove(kbuf, ksiz); +} + + +/** + * Retrieve the value of a record. + */ +char* kcidxget(KCIDX* idx, const char* kbuf, size_t ksiz, size_t* sp) { + _assert_(idx && kbuf && ksiz <= MEMMAXSIZ && sp); + IndexDB* idb = (IndexDB*)idx; + return idb->get(kbuf, ksiz, sp); +} + + +/** + * Synchronize updated contents with the file and the device. + */ +int32_t kcidxsync(KCIDX* idx, int32_t hard, KCFILEPROC proc, void* opq) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + class FileProcessorImpl : public BasicDB::FileProcessor { + public: + explicit FileProcessorImpl(KCFILEPROC proc, void* opq) : proc_(proc), opq_(opq) {} + bool process(const std::string& path, int64_t count, int64_t size) { + if (!proc_) return true; + return proc_(path.c_str(), count, size, opq_); + } + private: + KCFILEPROC proc_; + void* opq_; + }; + FileProcessorImpl myproc(proc, opq); + return idb->synchronize(hard, &myproc); +} + + +/** + * Remove all records. + */ +int32_t kcidxclear(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->clear(); +} + + +/** + * Get the number of records. + */ +int64_t kcidxcount(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->count(); +} + + +/** + * Get the size of the database file. + */ +int64_t kcidxsize(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return idb->size(); +} + + +/** + * Get the path of the database file. + */ +char* kcidxpath(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + std::string path = idb->path(); + size_t psiz = path.size(); + char* pbuf = new char[psiz+1]; + std::memcpy(pbuf, path.c_str(), psiz + 1); + return pbuf; +} + + +/** + * Get the miscellaneous status information. + */ +char* kcidxstatus(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + std::map<std::string, std::string> status; + if (!idb->status(&status)) return NULL; + std::ostringstream obuf; + std::map<std::string, std::string>::iterator it = status.begin(); + std::map<std::string, std::string>::iterator itend = status.end(); + while (it != itend) { + obuf << it->first << "\t" << it->second << "\n"; + ++it; + } + std::string sstr = obuf.str(); + size_t ssiz = sstr.size(); + char* sbuf = new char[ssiz+1]; + std::memcpy(sbuf, sstr.c_str(), ssiz + 1); + return sbuf; +} + + +/** + * Reveal the inner database object. + */ +KCDB* kcidxrevealinnerdb(KCIDX* idx) { + _assert_(idx); + IndexDB* idb = (IndexDB*)idx; + return (KCDB*)idb->reveal_inner_db(); +} + + +/** + * Create a string hash map object. + */ +KCMAP* kcmapnew(size_t bnum) { + _assert_(true); + return (KCMAP*)new TinyHashMap(bnum); +} + + +/** + * Destroy a map object. + */ +void kcmapdel(KCMAP* map) { + _assert_(map); + TinyHashMap* thm = (TinyHashMap*)map; + delete thm; +} + + +/** + * Set the value of a record. + */ +void kcmapset(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + TinyHashMap* thm = (TinyHashMap*)map; + thm->set(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Add a record. + */ +int32_t kcmapadd(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + TinyHashMap* thm = (TinyHashMap*)map; + return thm->add(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Replace the value of a record. + */ +int32_t kcmapreplace(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + TinyHashMap* thm = (TinyHashMap*)map; + return thm->replace(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Append the value of a record. + */ +void kcmapappend(KCMAP* map, const char* kbuf, size_t ksiz, const char* vbuf, size_t vsiz) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ && vbuf && vsiz <= MEMMAXSIZ); + TinyHashMap* thm = (TinyHashMap*)map; + thm->append(kbuf, ksiz, vbuf, vsiz); +} + + +/** + * Remove a record. + */ +int32_t kcmapremove(KCMAP* map, const char* kbuf, size_t ksiz) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ); + TinyHashMap* thm = (TinyHashMap*)map; + return thm->remove(kbuf, ksiz); +} + + +/** + * Retrieve the value of a record. + */ +const char* kcmapget(KCMAP* map, const char* kbuf, size_t ksiz, size_t* sp) { + _assert_(map && kbuf && ksiz <= MEMMAXSIZ && sp); + TinyHashMap* thm = (TinyHashMap*)map; + return thm->get(kbuf, ksiz, sp); +} + + +/** + * Remove all records. + */ +void kcmapclear(KCMAP* map) { + _assert_(map); + TinyHashMap* thm = (TinyHashMap*)map; + thm->clear(); +} + + +/** + * Get the number of records. + */ +size_t kcmapcount(KCMAP* map) { + _assert_(map); + TinyHashMap* thm = (TinyHashMap*)map; + return thm->count(); +} + + +/** + * Create a string hash map iterator object. + */ +KCMAPITER* kcmapiterator(KCMAP* map) { + _assert_(map); + TinyHashMap* thm = (TinyHashMap*)map; + return (KCMAPITER*)new TinyHashMap::Iterator(thm); +} + + +/** + * Destroy an iterator object. + */ +void kcmapiterdel(KCMAPITER* iter) { + _assert_(iter); + TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter; + delete thmi; +} + + +/** + * Get the key of the current record. + */ +const char* kcmapitergetkey(KCMAPITER* iter, size_t* sp) { + _assert_(iter && sp); + TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter; + return thmi->get_key(sp); +} + + +/** + * Get the value of the current record. + */ +const char* kcmapitergetvalue(KCMAPITER* iter, size_t* sp) { + _assert_(iter && sp); + TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter; + return thmi->get_value(sp); +} + + +/** + * Get a pair of the key and the value of the current record. + */ +const char* kcmapiterget(KCMAPITER* iter, size_t* ksp, const char** vbp, size_t* vsp) { + _assert_(iter && ksp && vbp && vsp); + TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter; + return thmi->get(ksp, vbp, vsp); +} + + +/** + * Step the cursor to the next record. + */ +void kcmapiterstep(KCMAPITER* iter) { + _assert_(iter); + TinyHashMap::Iterator* thmi = (TinyHashMap::Iterator*)iter; + return thmi->step(); +} + + +/** + * Create a string hash map sorter object. + */ +KCMAPSORT* kcmapsorter(KCMAP* map) { + _assert_(map); + TinyHashMap* thm = (TinyHashMap*)map; + return (KCMAPSORT*)new TinyHashMap::Sorter(thm); +} + + +/** + * Destroy an sorter object. + */ +void kcmapsortdel(KCMAPSORT* sort) { + _assert_(sort); + TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort; + delete thms; +} + + +/** + * Get the key of the current record. + */ +const char* kcmapsortgetkey(KCMAPSORT* sort, size_t* sp) { + _assert_(sort && sp); + TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort; + return thms->get_key(sp); +} + + +/** + * Get the value of the current record. + */ +const char* kcmapsortgetvalue(KCMAPSORT* sort, size_t* sp) { + _assert_(sort && sp); + TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort; + return thms->get_value(sp); +} + + +/** + * Get a pair of the key and the value of the current record. + */ +const char* kcmapsortget(KCMAPSORT* sort, size_t* ksp, const char** vbp, size_t* vsp) { + _assert_(sort && ksp && vbp && vsp); + TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort; + return thms->get(ksp, vbp, vsp); +} + + +/** + * Step the cursor to the next record. + */ +void kcmapsortstep(KCMAPSORT* sort) { + _assert_(sort); + TinyHashMap::Sorter* thms = (TinyHashMap::Sorter*)sort; + return thms->step(); +} + + +/** + * Create a string array list object. + */ +KCLIST* kclistnew() { + _assert_(true); + return (KCLIST*)new TinyArrayList(); +} + + +/** + * Destroy a list object. + */ +void kclistdel(KCLIST* list) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + delete tal; +} + + +/** + * Insert a record at the bottom of the list. + */ +void kclistpush(KCLIST* list, const char* buf, size_t size) { + _assert_(list && buf && size <= MEMMAXSIZ); + TinyArrayList* tal = (TinyArrayList*)list; + tal->push(buf, size); +} + + +/** + * Remove a record at the bottom of the list. + */ +int32_t kclistpop(KCLIST* list) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + return tal->pop(); +} + + +/** + * Insert a record at the top of the list. + */ +void kclistunshift(KCLIST* list, const char* buf, size_t size) { + _assert_(list && buf && size <= MEMMAXSIZ); + TinyArrayList* tal = (TinyArrayList*)list; + tal->unshift(buf, size); +} + + +/** + * Remove a record at the top of the list. + */ +int32_t kclistshift(KCLIST* list) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + return tal->shift(); +} + + +/** + * Insert a record at the position of the given index of the list. + */ +void kclistinsert(KCLIST* list, const char* buf, size_t size, size_t idx) { + _assert_(list && buf && size <= MEMMAXSIZ); + TinyArrayList* tal = (TinyArrayList*)list; + tal->insert(buf, size, idx); +} + + +/** + * Remove a record at the position of the given index of the list. + */ +void kclistremove(KCLIST* list, size_t idx) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + return tal->remove(idx); +} + + +/** + * Retrieve a record at the position of the given index of the list. + */ +const char* kclistget(KCLIST* list, size_t idx, size_t* sp) { + _assert_(list && sp); + TinyArrayList* tal = (TinyArrayList*)list; + return tal->get(idx, sp); +} + + +/** + * Remove all records. + */ +void kclistclear(KCLIST* list) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + tal->clear(); +} + + +/** + * Get the number of records. + */ +size_t kclistcount(KCLIST* list) { + _assert_(list); + TinyArrayList* tal = (TinyArrayList*)list; + return tal->count(); +} + + +} + +// END OF FILE |