2021-03-29 10:58:45 +07:00

648 lines
21 KiB
Go

package mdbx
/*
#include <stdlib.h>
#include <stdio.h>
#include "mdbxgo.h"
*/
import "C"
import (
"errors"
"os"
"runtime"
"sync"
"unsafe"
)
// success is a value returned from the LMDB API to indicate a successful call.
// The functions in this API this behavior and its use is not required.
const success = C.MDBX_SUCCESS
const (
// Flags for Env.Open.
//
// See mdbx_env_open
EnvDefaults = C.MDBX_ENV_DEFAULTS
LifoReclaim = C.MDBX_LIFORECLAIM
//FixedMap = C.MDBX_FIXEDMAP // Danger zone. Map memory at a fixed address.
NoSubdir = C.MDBX_NOSUBDIR // Argument to Open is a file, not a directory.
Accede = C.MDBX_ACCEDE
Coalesce = C.MDBX_COALESCE
Readonly = C.MDBX_RDONLY // Used in several functions to denote an object as readonly.
WriteMap = C.MDBX_WRITEMAP // Use a writable memory map.
NoMetaSync = C.MDBX_NOMETASYNC // Don't fsync metapage after commit.
UtterlyNoSync = C.MDBX_UTTERLY_NOSYNC
SafeNoSync = C.MDBX_SAFE_NOSYNC
Durable = C.MDBX_SYNC_DURABLE
NoTLS = C.MDBX_NOTLS // Danger zone. When unset reader locktable slots are tied to their thread.
//NoLock = C.MDBX_NOLOCK // Danger zone. LMDB does not use any locks.
NoReadahead = C.MDBX_NORDAHEAD // Disable readahead. Requires OS support.
NoMemInit = C.MDBX_NOMEMINIT // Disable LMDB memory initialization.
Exclusive = C.MDBX_EXCLUSIVE // Disable LMDB memory initialization.
)
const (
MinPageSize = C.MDBX_MIN_PAGESIZE
MaxPageSize = C.MDBX_MAX_PAGESIZE
MaxDbi = C.MDBX_MAX_DBI
)
// These flags are exclusively used in the Env.CopyFlags and Env.CopyFDFlags
// methods.
const (
// Flags for Env.CopyFlags
//
// See mdbx_env_copy2
CopyCompact = C.MDBX_CP_COMPACT // Perform compaction while copying
)
const (
AllowTxOverlap = C.MDBX_DBG_LEGACY_OVERLAP
)
const (
LogLvlFatal = C.MDBX_LOG_FATAL
LogLvlError = C.MDBX_LOG_ERROR
LogLvlWarn = C.MDBX_LOG_WARN
LogLvlNotice = C.MDBX_LOG_NOTICE
LogLvlVerbose = C.MDBX_LOG_VERBOSE
LogLvlDebug = C.MDBX_LOG_DEBUG
LogLvlTrace = C.MDBX_LOG_TRACE
LogLvlExtra = C.MDBX_LOG_EXTRA
LogLvlDoNotChange = C.MDBX_LOG_DONTCHANGE
)
const (
DbgAssert = C.MDBX_DBG_ASSERT
DbgAudit = C.MDBX_DBG_AUDIT
DbgJitter = C.MDBX_DBG_JITTER
DbgDump = C.MDBX_DBG_DUMP
DbgLegacyMultiOpen = C.MDBX_DBG_LEGACY_MULTIOPEN
DbgLegacyTxOverlap = C.MDBX_DBG_LEGACY_OVERLAP
DbgDoNotChange = C.MDBX_DBG_DONTCHANGE
)
const (
OptMaxDB = C.MDBX_opt_max_db
OptMaxReaders = C.MDBX_opt_max_readers
OptSyncBytes = C.MDBX_opt_sync_bytes
OptSyncPeriod = C.MDBX_opt_sync_period
OptRpAugmentLimit = C.MDBX_opt_rp_augment_limit
OptLooseLimit = C.MDBX_opt_loose_limit
OptDpReverseLimit = C.MDBX_opt_dp_reserve_limit
OptTxnDpLimit = C.MDBX_opt_txn_dp_limit
OptTxnDpInitial = C.MDBX_opt_txn_dp_initial
OptSpillMaxDenominator = C.MDBX_opt_spill_max_denominator
OptSpillMinDenominator = C.MDBX_opt_spill_min_denominator
OptSpillParent4ChildDenominator = C.MDBX_opt_spill_parent4child_denominator
)
var (
LoggerDoNotChange = C.MDBX_LOGGER_DONTCHANGE
)
// DBI is a handle for a database in an Env.
//
// See MDBX_dbi
type DBI C.MDBX_dbi
// Env is opaque structure for a database environment. A DB environment
// supports multiple databases, all residing in the same shared-memory map.
//
// See MDBX_env.
type Env struct {
_env *C.MDBX_env
// closeLock is used to allow the Txn finalizer to check if the Env has
// been closed, so that it may know if it must abort.
closeLock sync.RWMutex
ckey *C.MDBX_val
cval *C.MDBX_val
}
// NewEnv allocates and initializes a new Env.
//
// See mdbx_env_create.
func NewEnv() (*Env, error) {
env := new(Env)
ret := C.mdbx_env_create(&env._env)
if ret != success {
return nil, operrno("mdbx_env_create", ret)
}
env.ckey = (*C.MDBX_val)(C.malloc(C.size_t(unsafe.Sizeof(C.MDBX_val{}))))
env.cval = (*C.MDBX_val)(C.malloc(C.size_t(unsafe.Sizeof(C.MDBX_val{}))))
runtime.SetFinalizer(env, (*Env).Close)
return env, nil
}
// Open an environment handle. If this function fails Close() must be called to
// discard the Env handle. Open passes flags|NoTLS to mdbx_env_open.
//
// See mdbx_env_open.
func (env *Env) Open(path string, flags uint, mode os.FileMode) error {
cpath := C.CString(path)
defer C.free(unsafe.Pointer(cpath))
ret := C.mdbx_env_open(env._env, cpath, C.MDBX_env_flags_t(NoTLS|flags), C.mdbx_mode_t(mode))
return operrno("mdbx_env_open", ret)
}
var errNotOpen = errors.New("enivornment is not open")
var errNegSize = errors.New("negative size")
// FD returns the open file descriptor (or Windows file handle) for the given
// environment. An error is returned if the environment has not been
// successfully Opened (where C API just retruns an invalid handle).
//
// See mdbx_env_get_fd.
func (env *Env) FD() (uintptr, error) {
// fdInvalid is the value -1 as a uintptr, which is used by LMDB in the
// case that env has not been opened yet. the strange construction is done
// to avoid constant value overflow errors at compile time.
const fdInvalid = ^uintptr(0)
mf := new(C.mdbx_filehandle_t)
ret := C.mdbx_env_get_fd(env._env, mf)
err := operrno("mdbx_env_get_fd", ret)
if err != nil {
return 0, err
}
fd := uintptr(*mf)
if fd == fdInvalid {
return 0, errNotOpen
}
return fd, nil
}
// ReaderList dumps the contents of the reader lock table as text. Readers
// start on the second line as space-delimited fields described by the first
// line.
//
// See mdbx_reader_list.
//func (env *Env) ReaderList(fn func(string) error) error {
// ctx, done := newMsgFunc(fn)
// defer done()
// if fn == nil {
// ctx = 0
// }
//
// ret := C.mdbxgo_reader_list(env._env, C.size_t(ctx))
// if ret >= 0 {
// return nil
// }
// if ret < 0 && ctx != 0 {
// err := ctx.get().err
// if err != nil {
// return err
// }
// }
// return operrno("mdbx_reader_list", ret)
//}
// ReaderCheck clears stale entries from the reader lock table and returns the
// number of entries cleared.
//
// See mdbx_reader_check()
func (env *Env) ReaderCheck() (int, error) {
var _dead C.int
ret := C.mdbx_reader_check(env._env, &_dead)
return int(_dead), operrno("mdbx_reader_check", ret)
}
func (env *Env) close() bool {
if env._env == nil {
return false
}
env.closeLock.Lock()
C.mdbx_env_close(env._env)
env._env = nil
env.closeLock.Unlock()
C.free(unsafe.Pointer(env.ckey))
C.free(unsafe.Pointer(env.cval))
env.ckey = nil
env.cval = nil
return true
}
// Close shuts down the environment, releases the memory map, and clears the
// finalizer on env.
//
// See mdbx_env_close.
func (env *Env) Close() error {
if env.close() {
runtime.SetFinalizer(env, nil)
return nil
}
return errors.New("environment is already closed")
}
// CopyFD copies env to the the file descriptor fd.
//
// See mdbx_env_copyfd.
//func (env *Env) CopyFD(fd uintptr) error {
// ret := C.mdbx_env_copyfd(env._env, C.mdbx_filehandle_t(fd))
// return operrno("mdbx_env_copyfd", ret)
//}
// CopyFDFlag copies env to the file descriptor fd, with options.
//
// See mdbx_env_copyfd2.
//func (env *Env) CopyFDFlag(fd uintptr, flags uint) error {
// ret := C.mdbx_env_copyfd2(env._env, C.mdbx_filehandle_t(fd), C.uint(flags))
// return operrno("mdbx_env_copyfd2", ret)
//}
// Copy copies the data in env to an environment at path.
//
// See mdbx_env_copy.
//func (env *Env) Copy(path string) error {
// cpath := C.CString(path)
// defer C.free(unsafe.Pointer(cpath))
// ret := C.mdbx_env_copy(env._env, cpath)
// return operrno("mdbx_env_copy", ret)
//}
// CopyFlag copies the data in env to an environment at path created with flags.
//
// See mdbx_env_copy2.
//func (env *Env) CopyFlag(path string, flags uint) error {
// cpath := C.CString(path)
// defer C.free(unsafe.Pointer(cpath))
// ret := C.mdbx_env_copy2(env._env, cpath, C.uint(flags))
// return operrno("mdbx_env_copy2", ret)
//}
// Stat contains database status information.
//
// See MDBX_stat.
type Stat struct {
PSize uint // Size of a database page. This is currently the same for all databases.
Depth uint // Depth (height) of the B-tree
BranchPages uint64 // Number of internal (non-leaf) pages
LeafPages uint64 // Number of leaf pages
OverflowPages uint64 // Number of overflow pages
Entries uint64 // Number of data items
LastTxId uint64 // Transaction ID of commited last modification
}
// Stat returns statistics about the environment.
//
// See mdbx_env_stat.
func (env *Env) Stat() (*Stat, error) {
var _stat C.MDBX_stat
var ret C.int = C.mdbx_env_stat_ex(env._env, nil, &_stat, C.size_t(unsafe.Sizeof(_stat)))
if ret != success {
return nil, operrno("mdbx_env_stat_ex", ret)
}
stat := Stat{PSize: uint(_stat.ms_psize),
Depth: uint(_stat.ms_depth),
BranchPages: uint64(_stat.ms_branch_pages),
LeafPages: uint64(_stat.ms_leaf_pages),
OverflowPages: uint64(_stat.ms_overflow_pages),
Entries: uint64(_stat.ms_entries),
LastTxId: uint64(_stat.ms_mod_txnid)}
return &stat, nil
}
// EnvInfo contains information an environment.
//
// See MDBX_envinfo.
type EnvInfo struct {
MapSize int64 // Size of the data memory map
LastPNO int64 // ID of the last used page
Geo struct {
Lower uint64
Upper uint64
Current uint64
Shrink uint64
Grow uint64
}
LastTxnID int64 // ID of the last committed transaction
MaxReaders uint // maximum number of threads for the environment
NumReaders uint // maximum number of threads used in the environment
PageSize uint //
SystemPageSize uint //
AutoSyncThreshold uint //
SinceSyncSeconds16dot16 uint //
AutosyncPeriodSeconds16dot16 uint //
SinceReaderCheckSeconds16dot16 uint //
Flags uint //
}
// Info returns information about the environment.
//
// See mdbx_env_info.
func (env *Env) Info() (*EnvInfo, error) {
var _info C.MDBX_envinfo
var ret C.int
if err := env.View(func(txn *Txn) error {
ret = C.mdbx_env_info_ex(env._env, txn._txn, &_info, C.size_t(unsafe.Sizeof(_info)))
return nil
}); err != nil {
return nil, err
}
if ret != success {
return nil, operrno("mdbx_env_info", ret)
}
info := EnvInfo{
MapSize: int64(_info.mi_mapsize),
Geo: struct {
Lower uint64
Upper uint64
Current uint64
Shrink uint64
Grow uint64
}{
Lower: uint64(_info.mi_geo.lower),
Upper: uint64(_info.mi_geo.upper),
Current: uint64(_info.mi_geo.current),
Shrink: uint64(_info.mi_geo.shrink),
Grow: uint64(_info.mi_geo.grow),
},
LastPNO: int64(_info.mi_last_pgno),
LastTxnID: int64(_info.mi_recent_txnid),
MaxReaders: uint(_info.mi_maxreaders),
NumReaders: uint(_info.mi_numreaders),
PageSize: uint(_info.mi_dxb_pagesize),
SystemPageSize: uint(_info.mi_sys_pagesize),
AutoSyncThreshold: uint(_info.mi_autosync_threshold),
SinceSyncSeconds16dot16: uint(_info.mi_since_sync_seconds16dot16),
AutosyncPeriodSeconds16dot16: uint(_info.mi_autosync_period_seconds16dot16),
SinceReaderCheckSeconds16dot16: uint(_info.mi_since_reader_check_seconds16dot16),
Flags: uint(_info.mi_mode),
}
return &info, nil
}
// Sync flushes buffers to disk. If force is true a synchronous flush occurs
// and ignores any UtterlyNoSync or MapAsync flag on the environment.
//
// See mdbx_env_sync.
func (env *Env) Sync(force bool, nonblock bool) error {
ret := C.mdbx_env_sync_ex(env._env, C.bool(force), C.bool(nonblock))
return operrno("mdbx_env_sync_ex", ret)
}
// SetFlags sets flags in the environment.
//
// See mdbx_env_set_flags.
func (env *Env) SetFlags(flags uint) error {
ret := C.mdbx_env_set_flags(env._env, C.MDBX_env_flags_t(flags), true)
return operrno("mdbx_env_set_flags", ret)
}
// UnsetFlags clears flags in the environment.
//
// See mdbx_env_set_flags.
func (env *Env) UnsetFlags(flags uint) error {
ret := C.mdbx_env_set_flags(env._env, C.MDBX_env_flags_t(flags), false)
return operrno("mdbx_env_set_flags", ret)
}
// Flags returns the flags set in the environment.
//
// See mdbx_env_get_flags.
func (env *Env) Flags() (uint, error) {
var _flags C.uint
ret := C.mdbx_env_get_flags(env._env, &_flags)
if ret != success {
return 0, operrno("mdbx_env_get_flags", ret)
}
return uint(_flags), nil
}
func (env *Env) SetDebug(logLvl int, dbg int, logger *C.MDBX_debug_func) error {
ret := C.mdbx_setup_debug(C.MDBX_log_level_t(logLvl), C.MDBX_debug_flags_t(dbg), logger)
return operrno("mdbx_setup_debug", ret)
}
// Path returns the path argument passed to Open. Path returns a non-nil error
// if env.Open() was not previously called.
//
// See mdbx_env_get_path.
func (env *Env) Path() (string, error) {
var cpath *C.char
ret := C.mdbx_env_get_path(env._env, &cpath)
if ret != success {
return "", operrno("mdbx_env_get_path", ret)
}
if cpath == nil {
return "", errNotOpen
}
return C.GoString(cpath), nil
}
// SetMaxFreelistReuse sets the size of the environment memory map.
//
// Find a big enough contiguous page range for large values in freelist is hard
// just allocate new pages and even don't try to search if value is bigger than this limit.
// measured in pages
//func (env *Env) SetMaxFreelistReuse(pagesLimit uint) error {
// ret := C.mdbx_env_set_maxfree_reuse(env._env, C.uint(pagesLimit))
// return operrno("mdbx_env_set_maxfree_reuse", ret)
//}
// MaxFreelistReuse
//func (env *Env) MaxFreelistReuse() (uint, error) {
// var pages C.uint
// ret := C.mdbx_env_get_maxfree_reuse(env._env, &pages)
// return uint(pages), operrno("mdbx_env_get_maxreaders", ret)
//}
// SetMapSize sets the size of the environment memory map.
//
// See mdbx_env_set_mapsize.
//func (env *Env) SetMapSize(size int64) error {
// if size < 0 {
// return errNegSize
// }
// ret := C.mdbx_env_set_mapsize(env._env, C.size_t(size))
// return operrno("mdbx_env_set_mapsize", ret)
//}
func (env *Env) SetOption(option uint, value uint64) error {
ret := C.mdbx_env_set_option(env._env, C.MDBX_option_t(option), C.uint64_t(value))
return operrno("mdbx_env_set_option", ret)
}
func (env *Env) SetGeometry(sizeLower int, sizeNow int, sizeUpper int, growthStep int, shrinkThreshold int, pageSize int) error {
ret := C.mdbx_env_set_geometry(env._env,
C.intptr_t(sizeLower),
C.intptr_t(sizeNow),
C.intptr_t(sizeUpper),
C.intptr_t(growthStep),
C.intptr_t(shrinkThreshold),
C.intptr_t(pageSize))
return operrno("mdbx_env_set_geometry", ret)
}
// SetMaxReaders sets the maximum number of reader slots in the environment.
//
// See mdbx_env_set_maxreaders.
func (env *Env) SetMaxReaders(size int) error {
if size < 0 {
return errNegSize
}
ret := C.mdbx_env_set_maxreaders(env._env, C.uint(size))
return operrno("mdbx_env_set_maxreaders", ret)
}
// MaxReaders returns the maximum number of reader slots for the environment.
//
// See mdbx_env_get_maxreaders.
func (env *Env) MaxReaders() (int, error) {
var max C.uint
ret := C.mdbx_env_get_maxreaders(env._env, &max)
return int(max), operrno("mdbx_env_get_maxreaders", ret)
}
// MaxKeySize returns the maximum allowed length for a key.
//
// See mdbx_env_get_maxkeysize.
func (env *Env) MaxKeySize() int {
if env == nil {
return int(C.mdbx_env_get_maxkeysize_ex(nil, 0))
}
return int(C.mdbx_env_get_maxkeysize_ex(env._env, 0))
}
// SetMaxDBs sets the maximum number of named databases for the environment.
//
// See mdbx_env_set_maxdbs.
func (env *Env) SetMaxDBs(size int) error {
if size < 0 {
return errNegSize
}
ret := C.mdbx_env_set_maxdbs(env._env, C.MDBX_dbi(size))
return operrno("mdbx_env_set_maxdbs", ret)
}
// BeginTxn is an unsafe, low-level method to initialize a new transaction on
// env. The Txn returned by BeginTxn is unmanaged and must be terminated by
// calling either its Abort or Commit methods to ensure that its resources are
// released.
//
// BeginTxn does not call runtime.LockOSThread. Unless the Readonly flag is
// passed goroutines must call runtime.LockOSThread before calling BeginTxn and
// the returned Txn must not have its methods called from another goroutine.
// Failure to meet these restrictions can have undefined results that may
// include deadlocking your application.
//
// Instead of calling BeginTxn users should prefer calling the View and Update
// methods, which assist in management of Txn objects and provide OS thread
// locking required for write transactions.
//
// Unterminated transactions can adversly effect
// database performance and cause the database to grow until the map is full.
//
// See mdbx_txn_begin.
func (env *Env) BeginTxn(parent *Txn, flags uint) (*Txn, error) {
return beginTxn(env, parent, flags)
}
// RunTxn creates a new Txn and calls fn with it as an argument. Run commits
// the transaction if fn returns nil otherwise the transaction is aborted.
// Because RunTxn terminates the transaction goroutines should not retain
// references to it or its data after fn returns.
//
// RunTxn does not call runtime.LockOSThread. Unless the Readonly flag is
// passed the calling goroutine should ensure it is locked to its thread and
// any goroutines started by fn must not call methods on the Txn object it is
// passed.
//
// See mdbx_txn_begin.
func (env *Env) RunTxn(flags uint, fn TxnOp) error {
return env.run(false, flags, fn)
}
// View creates a readonly transaction with a consistent view of the
// environment and passes it to fn. View terminates its transaction after fn
// returns. Any error encountered by View is returned.
//
// Unlike with Update transactions, goroutines created by fn are free to call
// methods on the Txn passed to fn provided they are synchronized in their
// accesses (e.g. using a mutex or channel).
//
// Any call to Commit, Abort, Reset or Renew on a Txn created by View will
// panic.
func (env *Env) View(fn TxnOp) error {
return env.run(false, Readonly, fn)
}
// Update calls fn with a writable transaction. Update commits the transaction
// if fn returns a nil error otherwise Update aborts the transaction and
// returns the error.
//
// Update calls runtime.LockOSThread to lock the calling goroutine to its
// thread and until fn returns and the transaction has been terminated, at
// which point runtime.UnlockOSThread is called. If the calling goroutine is
// already known to be locked to a thread, use UpdateLocked instead to avoid
// premature unlocking of the goroutine.
//
// Neither Update nor UpdateLocked cannot be called safely from a goroutine
// where it isn't known if runtime.LockOSThread has been called. In such
// situations writes must either be done in a newly created goroutine which can
// be safely locked, or through a worker goroutine that accepts updates to
// apply and delivers transaction results using channels. See the package
// documentation and examples for more details.
//
// Goroutines created by the operation fn must not use methods on the Txn
// object that fn is passed. Doing so would have undefined and unpredictable
// results for your program (likely including data loss, deadlock, etc).
//
// Any call to Commit, Abort, Reset or Renew on a Txn created by Update will
// panic.
func (env *Env) Update(fn TxnOp) error {
return env.run(true, 0, fn)
}
// UpdateLocked behaves like Update but does not lock the calling goroutine to
// its thread. UpdateLocked should be used if the calling goroutine is already
// locked to its thread for another purpose.
//
// Neither Update nor UpdateLocked cannot be called safely from a goroutine
// where it isn't known if runtime.LockOSThread has been called. In such
// situations writes must either be done in a newly created goroutine which can
// be safely locked, or through a worker goroutine that accepts updates to
// apply and delivers transaction results using channels. See the package
// documentation and examples for more details.
//
// Goroutines created by the operation fn must not use methods on the Txn
// object that fn is passed. Doing so would have undefined and unpredictable
// results for your program (likely including data loss, deadlock, etc).
//
// Any call to Commit, Abort, Reset or Renew on a Txn created by UpdateLocked
// will panic.
func (env *Env) UpdateLocked(fn TxnOp) error {
return env.run(false, 0, fn)
}
func (env *Env) run(lock bool, flags uint, fn TxnOp) error {
if lock {
runtime.LockOSThread()
defer runtime.UnlockOSThread()
}
txn, err := beginTxn(env, nil, flags)
if err != nil {
return err
}
return txn.runOpTerm(fn)
}
// CloseDBI closes the database handle, db. Normally calling CloseDBI
// explicitly is not necessary.
//
// It is the caller's responsibility to serialize calls to CloseDBI.
//
// See mdbx_dbi_close.
func (env *Env) CloseDBI(db DBI) {
C.mdbx_dbi_close(env._env, C.MDBX_dbi(db))
}