From 5bfb29d1a8810b802ab117dc2fbc211ae213b32f Mon Sep 17 00:00:00 2001 From: Ilya Miheev Date: Wed, 12 Jun 2024 11:38:21 +0400 Subject: [PATCH 1/4] added valNP (no pointer value) argument to Txn obj tp increase performance (see GetFast for more details) --- mdbx/txn.go | 29 +++++++++++++++++++++++++---- mdbx/txn_test.go | 36 ++++++++++++++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 4 deletions(-) diff --git a/mdbx/txn.go b/mdbx/txn.go index fa1787b..1d7b51c 100644 --- a/mdbx/txn.go +++ b/mdbx/txn.go @@ -57,10 +57,11 @@ const ( // // See MDBX_txn. type Txn struct { - env *Env - _txn *C.MDBX_txn - key *C.MDBX_val - val *C.MDBX_val + env *Env + _txn *C.MDBX_txn + key *C.MDBX_val + valNP C.MDBX_val + val *C.MDBX_val errLogf func(format string, v ...interface{}) @@ -605,6 +606,26 @@ func (txn *Txn) Get(dbi DBI, key []byte) ([]byte, error) { return b, nil } +func (txn *Txn) GetFast(dbi DBI, key []byte) ([]byte, error) { + var k *C.char + if len(key) > 0 { + k = (*C.char)(unsafe.Pointer(&key[0])) + } + ret := C.mdbxgo_get( + txn._txn, C.MDBX_dbi(dbi), + k, C.size_t(len(key)), + &txn.valNP, + ) + err := operrno("mdbx_get", ret) + if err != nil { + txn.valNP = C.MDBX_val{} + return nil, err + } + b := castToBytes(&txn.valNP) + txn.valNP = C.MDBX_val{} + return b, nil +} + // Put stores an item in database dbi. // // See mdbx_put. diff --git a/mdbx/txn_test.go b/mdbx/txn_test.go index 60f0dce..65e426d 100644 --- a/mdbx/txn_test.go +++ b/mdbx/txn_test.go @@ -1299,6 +1299,42 @@ func BenchmarkTxn_Get(b *testing.B) { } } +func BenchmarkTxn_GetFast(b *testing.B) { + env, _ := setup(b) + + var db DBI + k := make([]byte, 8) + binary.BigEndian.PutUint64(k, uint64(1)) + + if err := env.Update(func(txn *Txn) (err error) { + db, err = txn.OpenRoot(0) + if err != nil { + return err + } + err = txn.Put(db, k, k, 0) + if err != nil { + return err + } + return nil + }); err != nil { + b.Errorf("dbi: %v", err) + return + } + + if err := env.View(func(txn *Txn) (err error) { + b.ResetTimer() + for i := 0; i < b.N; i++ { + _, err := txn.GetFast(db, k) + if err != nil { + return err + } + } + return nil + }); err != nil { + b.Errorf("put: %v", err) + } +} + func TestTxnEnvWarmup(t *testing.T) { env, _ := setup(t) From 6f9a27c0dbed2d4c0756aa30bca0e6e0fedd791b Mon Sep 17 00:00:00 2001 From: Ilya Miheev Date: Wed, 12 Jun 2024 12:03:56 +0400 Subject: [PATCH 2/4] replace txn.Get with txn.GetFast realization --- mdbx/txn.go | 24 ++---------------------- mdbx/txn_test.go | 36 ------------------------------------ 2 files changed, 2 insertions(+), 58 deletions(-) diff --git a/mdbx/txn.go b/mdbx/txn.go index 1d7b51c..2a15005 100644 --- a/mdbx/txn.go +++ b/mdbx/txn.go @@ -587,26 +587,6 @@ func (txn *Txn) bytes(val *C.MDBX_val) []byte { // // See mdbx_get. func (txn *Txn) Get(dbi DBI, key []byte) ([]byte, error) { - var k *C.char - if len(key) > 0 { - k = (*C.char)(unsafe.Pointer(&key[0])) - } - ret := C.mdbxgo_get( - txn._txn, C.MDBX_dbi(dbi), - k, C.size_t(len(key)), - txn.val, - ) - err := operrno("mdbx_get", ret) - if err != nil { - *txn.val = C.MDBX_val{} - return nil, err - } - b := castToBytes(txn.val) - *txn.val = C.MDBX_val{} - return b, nil -} - -func (txn *Txn) GetFast(dbi DBI, key []byte) ([]byte, error) { var k *C.char if len(key) > 0 { k = (*C.char)(unsafe.Pointer(&key[0])) @@ -618,11 +598,11 @@ func (txn *Txn) GetFast(dbi DBI, key []byte) ([]byte, error) { ) err := operrno("mdbx_get", ret) if err != nil { - txn.valNP = C.MDBX_val{} + txn.valNP.iov_base, txn.valNP.iov_len = nil, 0 return nil, err } b := castToBytes(&txn.valNP) - txn.valNP = C.MDBX_val{} + txn.valNP.iov_base, txn.valNP.iov_len = nil, 0 return b, nil } diff --git a/mdbx/txn_test.go b/mdbx/txn_test.go index 65e426d..60f0dce 100644 --- a/mdbx/txn_test.go +++ b/mdbx/txn_test.go @@ -1299,42 +1299,6 @@ func BenchmarkTxn_Get(b *testing.B) { } } -func BenchmarkTxn_GetFast(b *testing.B) { - env, _ := setup(b) - - var db DBI - k := make([]byte, 8) - binary.BigEndian.PutUint64(k, uint64(1)) - - if err := env.Update(func(txn *Txn) (err error) { - db, err = txn.OpenRoot(0) - if err != nil { - return err - } - err = txn.Put(db, k, k, 0) - if err != nil { - return err - } - return nil - }); err != nil { - b.Errorf("dbi: %v", err) - return - } - - if err := env.View(func(txn *Txn) (err error) { - b.ResetTimer() - for i := 0; i < b.N; i++ { - _, err := txn.GetFast(db, k) - if err != nil { - return err - } - } - return nil - }); err != nil { - b.Errorf("put: %v", err) - } -} - func TestTxnEnvWarmup(t *testing.T) { env, _ := setup(t) From a86986fef731675b4ab217e79261b586b795dcab Mon Sep 17 00:00:00 2001 From: Ilya Miheev Date: Wed, 12 Jun 2024 12:30:34 +0400 Subject: [PATCH 3/4] change type of txn.val from *C.MDBX_val to C.MDBX_val --- mdbx/cursor.go | 22 +++++++++++----------- mdbx/txn.go | 29 ++++++++++++++--------------- 2 files changed, 25 insertions(+), 26 deletions(-) diff --git a/mdbx/cursor.go b/mdbx/cursor.go index 8698396..991e2e7 100644 --- a/mdbx/cursor.go +++ b/mdbx/cursor.go @@ -159,7 +159,7 @@ func (c *Cursor) Get(setkey, setval []byte, op uint) (key, val []byte, err error } if err != nil { *c.txn.key = C.MDBX_val{} - *c.txn.val = C.MDBX_val{} + c.txn.val = C.MDBX_val{} return nil, nil, err } @@ -174,12 +174,12 @@ func (c *Cursor) Get(setkey, setval []byte, op uint) (key, val []byte, err error key = castToBytes(c.txn.key) } } - val = castToBytes(c.txn.val) + val = castToBytes(&c.txn.val) // Clear transaction storage record storage area for future use and to // prevent dangling references. *c.txn.key = C.MDBX_val{} - *c.txn.val = C.MDBX_val{} + c.txn.val = C.MDBX_val{} return key, val, nil } @@ -189,7 +189,7 @@ func (c *Cursor) Get(setkey, setval []byte, op uint) (key, val []byte, err error // // See mdb_cursor_get. func (c *Cursor) getVal0(op uint) error { - ret := C.mdbx_cursor_get(c._c, c.txn.key, c.txn.val, C.MDBX_cursor_op(op)) + ret := C.mdbx_cursor_get(c._c, c.txn.key, &c.txn.val, C.MDBX_cursor_op(op)) return operrno("mdbx_cursor_get", ret) } @@ -206,7 +206,7 @@ func (c *Cursor) getVal1(setkey []byte, op uint) error { c._c, k, C.size_t(len(setkey)), c.txn.key, - c.txn.val, + &c.txn.val, C.MDBX_cursor_op(op), ) return operrno("mdbx_cursor_get", ret) @@ -220,7 +220,7 @@ func (c *Cursor) getVal01(setval []byte, op uint) error { c._c, v, C.size_t(len(setval)), c.txn.key, - c.txn.val, + &c.txn.val, C.MDBX_cursor_op(op), ) return operrno("mdbx_cursor_get", ret) @@ -242,7 +242,7 @@ func (c *Cursor) getVal2(setkey, setval []byte, op uint) error { c._c, k, C.size_t(len(setkey)), v, C.size_t(len(setval)), - c.txn.key, c.txn.val, + c.txn.key, &c.txn.val, C.MDBX_cursor_op(op), ) return operrno("mdbx_cursor_get", ret) @@ -280,16 +280,16 @@ func (c *Cursor) PutReserve(key []byte, n int, flags uint) ([]byte, error) { ret := C.mdbxgo_cursor_put1( c._c, k, C.size_t(len(key)), - c.txn.val, + &c.txn.val, C.MDBX_put_flags_t(flags|C.MDBX_RESERVE), ) err := operrno("mdbx_cursor_put", ret) if err != nil { - *c.txn.val = C.MDBX_val{} + c.txn.val = C.MDBX_val{} return nil, err } - b := castToBytes(c.txn.val) - *c.txn.val = C.MDBX_val{} + b := castToBytes(&c.txn.val) + c.txn.val = C.MDBX_val{} return b, nil } diff --git a/mdbx/txn.go b/mdbx/txn.go index 2a15005..beeccbe 100644 --- a/mdbx/txn.go +++ b/mdbx/txn.go @@ -57,11 +57,10 @@ const ( // // See MDBX_txn. type Txn struct { - env *Env - _txn *C.MDBX_txn - key *C.MDBX_val - valNP C.MDBX_val - val *C.MDBX_val + env *Env + _txn *C.MDBX_txn + key *C.MDBX_val + val C.MDBX_val errLogf func(format string, v ...interface{}) @@ -94,14 +93,14 @@ func beginTxn(env *Env, parent *Txn, flags uint) (*Txn, error) { // In a write Txn we can use the shared, C-allocated key and value // allocated by env, and freed when it is closed. txn.key = env.ckey - txn.val = env.cval + txn.val = *env.cval } else { // It is not easy to share C.MDBX_val values in this scenario unless // there is a synchronized pool involved, which will increase // overhead. Further, allocating these values with C will add // overhead both here and when the values are freed. txn.key = new(C.MDBX_val) - txn.val = new(C.MDBX_val) + txn.val = C.MDBX_val{} } } else { // Because parent Txn objects cannot be used while a sub-Txn is active @@ -594,15 +593,15 @@ func (txn *Txn) Get(dbi DBI, key []byte) ([]byte, error) { ret := C.mdbxgo_get( txn._txn, C.MDBX_dbi(dbi), k, C.size_t(len(key)), - &txn.valNP, + &txn.val, ) err := operrno("mdbx_get", ret) if err != nil { - txn.valNP.iov_base, txn.valNP.iov_len = nil, 0 + txn.val = C.MDBX_val{} return nil, err } - b := castToBytes(&txn.valNP) - txn.valNP.iov_base, txn.valNP.iov_len = nil, 0 + b := castToBytes(&txn.val) + txn.val = C.MDBX_val{} return b, nil } @@ -638,16 +637,16 @@ func (txn *Txn) PutReserve(dbi DBI, key []byte, n int, flags uint) ([]byte, erro ret := C.mdbxgo_put1( txn._txn, C.MDBX_dbi(dbi), k, C.size_t(len(key)), - txn.val, + &txn.val, C.MDBX_put_flags_t(flags|C.MDBX_RESERVE), ) err := operrno("mdbx_put", ret) if err != nil { - *txn.val = C.MDBX_val{} + txn.val = C.MDBX_val{} return nil, err } - b := castToBytes(txn.val) - *txn.val = C.MDBX_val{} + b := castToBytes(&txn.val) + txn.val = C.MDBX_val{} return b, nil } From ca514b2384a494b31158408c8820a44753985363 Mon Sep 17 00:00:00 2001 From: Ilya Miheev Date: Wed, 12 Jun 2024 12:52:37 +0400 Subject: [PATCH 4/4] no longer takes pointer from env val --- mdbx/txn.go | 2 -- 1 file changed, 2 deletions(-) diff --git a/mdbx/txn.go b/mdbx/txn.go index beeccbe..92b27b8 100644 --- a/mdbx/txn.go +++ b/mdbx/txn.go @@ -93,14 +93,12 @@ func beginTxn(env *Env, parent *Txn, flags uint) (*Txn, error) { // In a write Txn we can use the shared, C-allocated key and value // allocated by env, and freed when it is closed. txn.key = env.ckey - txn.val = *env.cval } else { // It is not easy to share C.MDBX_val values in this scenario unless // there is a synchronized pool involved, which will increase // overhead. Further, allocating these values with C will add // overhead both here and when the values are freed. txn.key = new(C.MDBX_val) - txn.val = C.MDBX_val{} } } else { // Because parent Txn objects cannot be used while a sub-Txn is active