Commit 75dc3060 authored by Kirill Smelkov's avatar Kirill Smelkov

X zodb: Goodby XTid; Load always loads `At`; loadSerial is actually not needed

also:

- DataTid -> DataTidHint + 0 if there is no such hint.
parent ba61dad6
...@@ -36,6 +36,7 @@ import ( ...@@ -36,6 +36,7 @@ import (
"lab.nexedi.com/kirr/go123/xnet" "lab.nexedi.com/kirr/go123/xnet"
"lab.nexedi.com/kirr/neo/go/neo" "lab.nexedi.com/kirr/neo/go/neo"
"lab.nexedi.com/kirr/neo/go/neo/internal/common"
"lab.nexedi.com/kirr/neo/go/zodb" "lab.nexedi.com/kirr/neo/go/zodb"
"lab.nexedi.com/kirr/neo/go/xcommon/log" "lab.nexedi.com/kirr/neo/go/xcommon/log"
"lab.nexedi.com/kirr/neo/go/xcommon/task" "lab.nexedi.com/kirr/neo/go/xcommon/task"
...@@ -459,13 +460,11 @@ func (c *Client) _Load(ctx context.Context, xid zodb.Xid) (*zodb.Buf, zodb.Tid, ...@@ -459,13 +460,11 @@ func (c *Client) _Load(ctx context.Context, xid zodb.Xid) (*zodb.Buf, zodb.Tid,
// FIXME ^^^ slink.CloseAccept after really dialed (not to deadlock if // FIXME ^^^ slink.CloseAccept after really dialed (not to deadlock if
// S decides to send us something) // S decides to send us something)
req := neo.GetObject{Oid: xid.Oid} // on the wire it comes as "before", not "at"
if xid.TidBefore { req := neo.GetObject{
req.Serial = neo.INVALID_TID Oid: xid.Oid,
req.Tid = xid.Tid Tid: common.At2Before(xid.At),
} else { Serial: neo.INVALID_TID,
req.Serial = xid.Tid
req.Tid = neo.INVALID_TID
} }
resp := neo.AnswerObject{} resp := neo.AnswerObject{}
......
// Copyright (C) 2017 Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// option) any later version, as published by the Free Software Foundation.
//
// You can also Link and Combine this program with other software covered by
// the terms of any of the Free Software licenses or any of the Open Source
// Initiative approved licenses and Convey the resulting work. Corresponding
// source of such a combination shall include the source code for all other
// software used.
//
// This program is distributed WITHOUT ANY WARRANTY; without even the implied
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
//
// See COPYING file for full licensing terms.
// See https://www.nexedi.com/licensing for rationale and options.
// Package common provides internal bits shared in between NEO client and server packages.
package common
import (
"lab.nexedi.com/kirr/neo/go/zodb"
)
// At2Before converts at to before for ZODB load semantics taking edge cases into account.
//
// For most values it is
//
// before = at + 1 ; at < ∞
//
// but at ∞ (zodb.TidMax) it is just
//
// before = at ; at = ∞
func At2Before(at zodb.Tid) (before zodb.Tid) {
if at < zodb.TidMax {
return at + 1
} else {
// XXX do we need to care here also for at > zodb.TidMax (zodb.Tid is currently unsigned)
return zodb.TidMax
}
}
// Before2At is the reverse function to At2Before
func Before2At(before zodb.Tid) (at zodb.Tid) {
if before < zodb.TidMax {
// XXX before = 0 ?
return before - 1
} else {
// XXX before > zodb.TidMax (same as in At2Before) ?
return zodb.TidMax
}
}
...@@ -39,6 +39,7 @@ import ( ...@@ -39,6 +39,7 @@ import (
"lab.nexedi.com/kirr/neo/go/neo" "lab.nexedi.com/kirr/neo/go/neo"
"lab.nexedi.com/kirr/neo/go/neo/client" "lab.nexedi.com/kirr/neo/go/neo/client"
"lab.nexedi.com/kirr/neo/go/neo/internal/common"
"lab.nexedi.com/kirr/neo/go/zodb" "lab.nexedi.com/kirr/neo/go/zodb"
"lab.nexedi.com/kirr/neo/go/zodb/storage/fs1" "lab.nexedi.com/kirr/neo/go/zodb/storage/fs1"
...@@ -428,7 +429,7 @@ func TestMasterStorage(t *testing.T) { ...@@ -428,7 +429,7 @@ func TestMasterStorage(t *testing.T) {
// C starts loading first object ... // C starts loading first object ...
wg = &errgroup.Group{} wg = &errgroup.Group{}
xid1 := zodb.Xid{Oid: 1, XTid: zodb.XTid{Tid: zodb.TidMax, TidBefore: true}} xid1 := zodb.Xid{Oid: 1, At: zodb.TidMax}
buf1, serial1, err := zstor.Load(bg, xid1) buf1, serial1, err := zstor.Load(bg, xid1)
exc.Raiseif(err) exc.Raiseif(err)
gox(wg, func() { gox(wg, func() {
...@@ -462,7 +463,7 @@ func TestMasterStorage(t *testing.T) { ...@@ -462,7 +463,7 @@ func TestMasterStorage(t *testing.T) {
// ... -> GetObject(xid1) // ... -> GetObject(xid1)
tc.Expect(conntx("c:2", "s:3", 3, &neo.GetObject{ tc.Expect(conntx("c:2", "s:3", 3, &neo.GetObject{
Oid: xid1.Oid, Oid: xid1.Oid,
Tid: xid1.Tid, Tid: common.At2Before(xid1.At),
Serial: neo.INVALID_TID, Serial: neo.INVALID_TID,
})) }))
tc.Expect(conntx("s:3", "c:2", 3, &neo.AnswerObject{ tc.Expect(conntx("s:3", "c:2", 3, &neo.AnswerObject{
...@@ -503,32 +504,24 @@ func TestMasterStorage(t *testing.T) { ...@@ -503,32 +504,24 @@ func TestMasterStorage(t *testing.T) {
t.Fatalf("ziter.NextData: %v", err) t.Fatalf("ziter.NextData: %v", err)
} }
for _, tidBefore := range []bool{false, true} { // TODO also test GetObject(tid=ø, serial=...) which originate from loadSerial on py side
xid := zodb.Xid{Oid: datai.Oid} // {=,<}tid:oid xid := zodb.Xid{Oid: datai.Oid, At: datai.Tid}
xid.Tid = datai.Tid
xid.TidBefore = tidBefore
if tidBefore {
xid.Tid++
}
buf, tid, err := C.Load(bg, xid) buf, serial, err := C.Load(bg, xid)
if datai.Data != nil { if datai.Data != nil {
if !(bytes.Equal(buf.Data, datai.Data) && tid == datai.Tid && err == nil) { if !(bytes.Equal(buf.Data, datai.Data) && serial == datai.Tid && err == nil) {
t.Fatalf("load: %v:\nhave: %v %v %q\nwant: %v nil %q", t.Fatalf("load: %v:\nhave: %v %v %q\nwant: %v nil %q",
xid, tid, err, buf.Data, datai.Tid, datai.Data) xid, serial, err, buf.Data, datai.Tid, datai.Data)
} }
} else { } else {
// deleted // deleted
errWant := &zodb.ErrXidMissing{xid} errWant := &zodb.ErrXidMissing{xid}
if !(buf == nil && tid == 0 && reflect.DeepEqual(err, errWant)) { if !(buf == nil && serial == 0 && reflect.DeepEqual(err, errWant)) {
t.Fatalf("load: %v:\nhave: %v, %#v, %#v\nwant: %v, %#v, %#v", t.Fatalf("load: %v:\nhave: %v, %#v, %#v\nwant: %v, %#v, %#v",
xid, tid, err, buf, zodb.Tid(0), errWant, []byte(nil)) xid, serial, err, buf, zodb.Tid(0), errWant, []byte(nil))
}
} }
} }
} }
} }
...@@ -610,9 +603,7 @@ func benchmarkGetObject(b *testing.B, Mnet, Snet, Cnet xnet.Networker, benchit f ...@@ -610,9 +603,7 @@ func benchmarkGetObject(b *testing.B, Mnet, Snet, Cnet xnet.Networker, benchit f
b.Fatal(err) b.Fatal(err)
} }
xid1 := zodb.Xid{Oid: 1} xid1 := zodb.Xid{Oid: 1, At: zodb.TidMax}
xid1.Tid = zodb.TidMax
xid1.TidBefore = true
buf1, serial1, err := zstor.Load(ctx, xid1) buf1, serial1, err := zstor.Load(ctx, xid1)
if err != nil { if err != nil {
......
...@@ -31,6 +31,7 @@ import ( ...@@ -31,6 +31,7 @@ import (
"github.com/pkg/errors" "github.com/pkg/errors"
"lab.nexedi.com/kirr/neo/go/neo" "lab.nexedi.com/kirr/neo/go/neo"
"lab.nexedi.com/kirr/neo/go/neo/internal/common"
"lab.nexedi.com/kirr/neo/go/zodb" "lab.nexedi.com/kirr/neo/go/zodb"
"lab.nexedi.com/kirr/neo/go/xcommon/log" "lab.nexedi.com/kirr/neo/go/xcommon/log"
"lab.nexedi.com/kirr/neo/go/xcommon/task" "lab.nexedi.com/kirr/neo/go/xcommon/task"
...@@ -542,22 +543,29 @@ func (stor *Storage) serveClient1(ctx context.Context, req neo.Msg) (resp neo.Ms ...@@ -542,22 +543,29 @@ func (stor *Storage) serveClient1(ctx context.Context, req neo.Msg) (resp neo.Ms
case *neo.GetObject: case *neo.GetObject:
xid := zodb.Xid{Oid: req.Oid} xid := zodb.Xid{Oid: req.Oid}
if req.Serial != neo.INVALID_TID { if req.Serial != neo.INVALID_TID {
xid.Tid = req.Serial xid.At = req.Serial
xid.TidBefore = false
} else { } else {
xid.Tid = req.Tid xid.At = common.Before2At(req.Tid)
xid.TidBefore = true
} }
buf, tid, err := stor.zstor.Load(ctx, xid) buf, serial, err := stor.zstor.Load(ctx, xid)
if err != nil { if err != nil {
// translate err to NEO protocol error codes // translate err to NEO protocol error codes
return neo.ErrEncode(err) return neo.ErrEncode(err)
} }
// for loadSerial - check we have exact hit - else "nodata"
if req.Serial != neo.INVALID_TID {
if serial != req.Serial {
// XXX actually show in error it was strict "=" load
return neo.ErrEncode(&zodb.ErrXidMissing{xid})
}
}
return &neo.AnswerObject{ return &neo.AnswerObject{
Oid: xid.Oid, Oid: xid.Oid,
Serial: tid, Serial: serial,
Compression: false, Compression: false,
Data: buf, Data: buf,
......
...@@ -176,7 +176,6 @@ func zhash(ctx context.Context, url string, h hasher, useprefetch bool, bench, c ...@@ -176,7 +176,6 @@ func zhash(ctx context.Context, url string, h hasher, useprefetch bool, bench, c
if err != nil { if err != nil {
return err return err
} }
before := lastTid + 1 // XXX overflow ?
if false { if false {
defer profile.Start(profile.TraceProfile).Stop() defer profile.Start(profile.TraceProfile).Stop()
...@@ -190,7 +189,7 @@ func zhash(ctx context.Context, url string, h hasher, useprefetch bool, bench, c ...@@ -190,7 +189,7 @@ func zhash(ctx context.Context, url string, h hasher, useprefetch bool, bench, c
nread := 0 nread := 0
loop: loop:
for { for {
xid := zodb.Xid{Oid: oid, XTid: zodb.XTid{Tid: before, TidBefore: true}} xid := zodb.Xid{Oid: oid, At: lastTid}
if xid.Oid % nprefetch == 0 { if xid.Oid % nprefetch == 0 {
prefetchBlk(ctx, xid) prefetchBlk(ctx, xid)
} }
......
...@@ -43,9 +43,9 @@ type Cache struct { ...@@ -43,9 +43,9 @@ type Cache struct {
mu sync.RWMutex mu sync.RWMutex
// cache is fully synchronized with storage for transactions with tid < before. // cache is fully synchronized with storage for transactions with tid <= head.
// XXX clarify ^^^ (it means if revCacheEntry.before=∞ it is Cache.before) // XXX clarify ^^^ (it means if revCacheEntry.head=∞ it is Cache.head)
before zodb.Tid head zodb.Tid
entryMap map[zodb.Oid]*oidCacheEntry // oid -> oid's cache entries entryMap map[zodb.Oid]*oidCacheEntry // oid -> oid's cache entries
...@@ -64,7 +64,7 @@ type oidCacheEntry struct { ...@@ -64,7 +64,7 @@ type oidCacheEntry struct {
sync.Mutex sync.Mutex
// cached revisions in ascending order // cached revisions in ascending order
// [i].serial < [i].before <= [i+1].serial < [i+1].before // [i].serial <= [i].head < [i+1].serial <= [i+1].head
// //
// NOTE ^^^ .serial = 0 while loading is in progress // NOTE ^^^ .serial = 0 while loading is in progress
// NOTE ^^^ .serial = 0 if .err != nil // NOTE ^^^ .serial = 0 if .err != nil
...@@ -76,21 +76,21 @@ type revCacheEntry struct { ...@@ -76,21 +76,21 @@ type revCacheEntry struct {
parent *oidCacheEntry // oidCacheEntry holding us parent *oidCacheEntry // oidCacheEntry holding us
inLRU lruHead // in Cache.lru; protected by Cache.gcMu inLRU lruHead // in Cache.lru; protected by Cache.gcMu
// we know that loadBefore(oid, .before) will give this .serial:oid. // we know that load(oid, .head) will give this .serial:oid.
// //
// this is only what we currently know - not necessarily covering // this is only what we currently know - not necessarily covering
// whole correct range - e.g. if oid revisions in db are 1 and 5 if we // whole correct range - e.g. if oid revisions in db are 1 and 5 if we
// query db with loadBefore(3) on return we'll get serial=1 and // query db with load(@3) on return we'll get serial=1 and
// remember .before as 3. But for loadBefore(4) we have to redo // remember .head as 3. But for load(@4) we have to redo
// database query again. // database query again.
// //
// if .before=∞ here, that actually means before is cache.before // if .head=∞ here, that actually means head is cache.head
// ( this way we do not need to bump .before to next tid in many // ( this way we do not need to bump .head to next tid in many
// unchanged cache entries when a transaction invalidation comes ) // unchanged cache entries when a transaction invalidation comes )
// //
// .before can be > cache.before and still finite - that represents a // .head can be > cache.head and still finite - that represents a
// case when loadBefore with tid > cache.before was called. // case when load with tid > cache.head was called.
before zodb.Tid head zodb.Tid
// loading result: object (buf, serial) or error // loading result: object (buf, serial) or error
buf *zodb.Buf buf *zodb.Buf
...@@ -166,13 +166,6 @@ func (c *Cache) Load(ctx context.Context, xid zodb.Xid) (buf *zodb.Buf, serial z ...@@ -166,13 +166,6 @@ func (c *Cache) Load(ctx context.Context, xid zodb.Xid) (buf *zodb.Buf, serial z
return nil, 0, rce.userErr(xid) return nil, 0, rce.userErr(xid)
} }
// for loadSerial - check we have exact hit - else "nodata"
if !xid.TidBefore {
if rce.serial != xid.Tid {
return nil, 0, &zodb.ErrXidMissing{xid}
}
}
rce.buf.XIncref() rce.buf.XIncref()
return rce.buf, rce.serial, nil return rce.buf, rce.serial, nil
} }
...@@ -199,26 +192,15 @@ func (c *Cache) Prefetch(ctx context.Context, xid zodb.Xid) { ...@@ -199,26 +192,15 @@ func (c *Cache) Prefetch(ctx context.Context, xid zodb.Xid) {
// lookupRCE returns revCacheEntry corresponding to xid. // lookupRCE returns revCacheEntry corresponding to xid.
// //
// If xid indicates loadSerial query (xid.TidBefore=false) then rce will be
// lookuped and eventually loaded as if it was queried with <(serial+1).
// It is caller responsibility to check loadSerial cases for exact hits after
// rce will become ready.
//
// rceNew indicates whether rce is new and so loading on it has not been // rceNew indicates whether rce is new and so loading on it has not been
// initiated yet. If so the caller should proceed to loading rce via loadRCE. // initiated yet. If so the caller should proceed to loading rce via loadRCE.
func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) { func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
// loadSerial(serial) -> loadBefore(serial+1)
before := xid.Tid
if !xid.TidBefore {
before++ // XXX overflow
}
// oid -> oce (oidCacheEntry) ; create new empty oce if not yet there // oid -> oce (oidCacheEntry) ; create new empty oce if not yet there
// exit with oce locked and cache.before read consistently // exit with oce locked and cache.syncedTo read consistently
c.mu.RLock() c.mu.RLock()
oce := c.entryMap[xid.Oid] oce := c.entryMap[xid.Oid]
cacheBefore := c.before cacheHead := c.head
if oce != nil { if oce != nil {
oce.Lock() oce.Lock()
...@@ -232,51 +214,51 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) { ...@@ -232,51 +214,51 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
oce = &oidCacheEntry{} oce = &oidCacheEntry{}
c.entryMap[xid.Oid] = oce c.entryMap[xid.Oid] = oce
} }
cacheBefore = c.before // reload c.before because we relocked the cache cacheHead = c.head // reload c.head because we relocked the cache
oce.Lock() oce.Lock()
c.mu.Unlock() c.mu.Unlock()
} }
// oce, before -> rce (revCacheEntry) // oce, at -> rce (revCacheEntry)
l := len(oce.rcev) l := len(oce.rcev)
i := sort.Search(l, func(i int) bool { i := sort.Search(l, func(i int) bool {
before_i := oce.rcev[i].before head_i := oce.rcev[i].head
if before_i == zodb.TidMax { if head_i == zodb.TidMax {
before_i = cacheBefore head_i = cacheHead
} }
return before <= before_i return xid.At <= head_i
}) })
switch { switch {
// not found - before > max(rcev.before) - insert new max entry // not found - at > max(rcev.head) - insert new max entry
case i == l: case i == l:
rce = oce.newRevEntry(i, before) rce = oce.newRevEntry(i, xid.At)
if rce.before == cacheBefore { if rce.head == cacheHead {
// FIXME better do this when the entry becomes loaded ? // FIXME better do this when the entry becomes loaded ?
// XXX vs concurrent invalidations? // XXX vs concurrent invalidations?
rce.before = zodb.TidMax rce.head = zodb.TidMax
} }
rceNew = true rceNew = true
// found: // found:
// before <= rcev[i].before // at <= rcev[i].head
// before > rcev[i-1].before // at > rcev[i-1].head
// exact match - we already have entry for this before // exact match - we already have entry for this at
case before == oce.rcev[i].before: case xid.At == oce.rcev[i].head:
rce = oce.rcev[i] rce = oce.rcev[i]
// non-exact match: // non-exact match:
// - same entry if q(before) ∈ (serial, before] // - same entry if q(at) ∈ [serial, head]
// - we can also reuse this entry if q(before) < before and err="nodata" // - we can also reuse this entry if q(at) <= head and err="nodata"
case oce.rcev[i].loaded() && ( case oce.rcev[i].loaded() && (
(oce.rcev[i].err == nil && oce.rcev[i].serial < before) || (oce.rcev[i].err == nil && oce.rcev[i].serial <= xid.At) ||
(isErrNoData(oce.rcev[i].err) && before < oce.rcev[i].before)): (isErrNoData(oce.rcev[i].err) && xid.At <= oce.rcev[i].head)):
rce = oce.rcev[i] rce = oce.rcev[i]
// otherwise - insert new entry // otherwise - insert new entry
default: default:
rce = oce.newRevEntry(i, before) rce = oce.newRevEntry(i, xid.At)
rceNew = true rceNew = true
} }
...@@ -290,10 +272,7 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) { ...@@ -290,10 +272,7 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
// loading completion is signalled by closing rce.ready. // loading completion is signalled by closing rce.ready.
func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) { func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) {
oce := rce.parent oce := rce.parent
buf, serial, err := c.loader.Load(ctx, zodb.Xid{ buf, serial, err := c.loader.Load(ctx, zodb.Xid{At: rce.head, Oid: oid})
Oid: oid,
XTid: zodb.XTid{Tid: rce.before, TidBefore: true},
})
// normalize buf/serial if it was error // normalize buf/serial if it was error
if err != nil { if err != nil {
...@@ -305,16 +284,16 @@ func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) { ...@@ -305,16 +284,16 @@ func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) {
rce.serial = serial rce.serial = serial
rce.buf = buf rce.buf = buf
rce.err = err rce.err = err
// verify db gives serial < before // verify db gives serial <= head
if rce.serial >= rce.before { if rce.serial > rce.head {
rce.errDB(oid, "load(<%v) -> %v", rce.before, serial) rce.errDB(oid, "load(@%v) -> %v", rce.head, serial)
} }
close(rce.ready) close(rce.ready)
δsize := rce.buf.Len() δsize := rce.buf.Len()
// merge rce with adjacent entries in parent // merge rce with adjacent entries in parent
// ( e.g. loadBefore(3) and loadBefore(4) results in the same data loaded if // ( e.g. load(@3) and load(@4) results in the same data loaded if
// there are only revisions with serials 1 and 5 ) // there are only revisions with serials 1 and 5 )
oce.Lock() oce.Lock()
i := oce.find(rce) i := oce.find(rce)
...@@ -391,7 +370,7 @@ func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) { ...@@ -391,7 +370,7 @@ func (c *Cache) loadRCE(ctx context.Context, rce *revCacheEntry, oid zodb.Oid) {
// //
// both prev and next must be already loaded. // both prev and next must be already loaded.
// prev and next must come adjacent to each other in parent.rcev with // prev and next must come adjacent to each other in parent.rcev with
// prev.before < next.before . // prev.head < next.head .
// //
// cur must be one of either prev or next and indicates which rce is current // cur must be one of either prev or next and indicates which rce is current
// and so may be adjusted with consistency check error. // and so may be adjusted with consistency check error.
...@@ -406,31 +385,31 @@ func tryMerge(prev, next, cur *revCacheEntry, oid zodb.Oid) bool { ...@@ -406,31 +385,31 @@ func tryMerge(prev, next, cur *revCacheEntry, oid zodb.Oid) bool {
// can merge if consistent if // can merge if consistent if
// (if merging) // (if merging)
// //
// Pok Nok Ns < Pb Ps = Ns // Pok Nok Ns <= Ph Ps = Ns
// Pe Nok Ns < Pb Pe != "nodata" (e.g. it was IO loading error for P) // Pe Nok Ns <= Ph Pe != "nodata" (e.g. it was IO loading error for P)
// Pok Ne --- // Pok Ne ---
// Ne Pe (Pe="nodata") && (Ne="nodata") -> XXX vs deleteObject? // Ne Pe (Pe="nodata") && (Ne="nodata") -> XXX vs deleteObject?
// -> let deleted object actually read // -> let deleted object actually read
// -> as special non-error value // -> as special non-error value
// //
// b - before // h - head
// s - serial // s - serial
// e - error // e - error
if next.err == nil && next.serial < prev.before { if next.err == nil && next.serial <= prev.head {
// drop prev // drop prev
prev.parent.del(prev) prev.parent.del(prev)
// check consistency // check consistency
switch { switch {
case prev.err == nil && prev.serial != next.serial: case prev.err == nil && prev.serial != next.serial:
cur.errDB(oid, "load(<%v) -> %v; load(<%v) -> %v", cur.errDB(oid, "load(@%v) -> %v; load(@%v) -> %v",
prev.before, prev.serial, next.before, next.serial) prev.head, prev.serial, next.head, next.serial)
case prev.err != nil && !isErrNoData(prev.err): case prev.err != nil && !isErrNoData(prev.err):
if cur.err == nil { if cur.err == nil {
cur.errDB(oid, "load(<%v) -> %v; load(<%v) -> %v", cur.errDB(oid, "load(@%v) -> %v; load(@%v) -> %v",
prev.before, prev.err, next.before, next.serial) prev.head, prev.err, next.head, next.serial)
} }
} }
...@@ -532,13 +511,13 @@ func isErrNoData(err error) bool { ...@@ -532,13 +511,13 @@ func isErrNoData(err error) bool {
return true return true
} }
// newRevEntry creates new revCacheEntry with .before and inserts it into .rcev @i. // newRevEntry creates new revCacheEntry with .head and inserts it into .rcev @i.
// (if i == len(oce.rcev) - entry is appended) // (if i == len(oce.rcev) - entry is appended)
func (oce *oidCacheEntry) newRevEntry(i int, before zodb.Tid) *revCacheEntry { func (oce *oidCacheEntry) newRevEntry(i int, head zodb.Tid) *revCacheEntry {
rce := &revCacheEntry{ rce := &revCacheEntry{
parent: oce, parent: oce,
serial: 0, serial: 0,
before: before, head: head,
ready: make(chan struct{}), ready: make(chan struct{}),
} }
rce.inLRU.Init() // initially not on Cache.lru list rce.inLRU.Init() // initially not on Cache.lru list
...@@ -596,7 +575,7 @@ func (rce *revCacheEntry) loaded() bool { ...@@ -596,7 +575,7 @@ func (rce *revCacheEntry) loaded() bool {
// userErr returns error that, if any, needs to be returned to user from Cache.Load // userErr returns error that, if any, needs to be returned to user from Cache.Load
// //
// ( ErrXidMissing contains xid for which it is missing. In cache we keep such // ( ErrXidMissing contains xid for which it is missing. In cache we keep such
// xid with max .before but users need to get ErrXidMissing with their own query ) // xid with max .head but users need to get ErrXidMissing with their own query )
func (rce *revCacheEntry) userErr(xid zodb.Xid) error { func (rce *revCacheEntry) userErr(xid zodb.Xid) error {
switch e := rce.err.(type) { switch e := rce.err.(type) {
case *zodb.ErrXidMissing: case *zodb.ErrXidMissing:
......
...@@ -69,31 +69,22 @@ func bufSame(buf1, buf2 *zodb.Buf) bool { ...@@ -69,31 +69,22 @@ func bufSame(buf1, buf2 *zodb.Buf) bool {
func (stor *tStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, serial zodb.Tid, err error) { func (stor *tStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, serial zodb.Tid, err error) {
//fmt.Printf("> load(%v)\n", xid) //fmt.Printf("> load(%v)\n", xid)
//defer func() { fmt.Printf("< %v, %v, %v\n", buf.XData(), serial, err) }() //defer func() { fmt.Printf("< %v, %v, %v\n", buf.XData(), serial, err) }()
tid := xid.Tid
if !xid.TidBefore {
tid++ // XXX overflow?
}
datav := stor.dataMap[xid.Oid] datav := stor.dataMap[xid.Oid]
if datav == nil { if datav == nil {
return nil, 0, &zodb.ErrOidMissing{xid.Oid} return nil, 0, &zodb.ErrOidMissing{xid.Oid}
} }
// find max entry with .serial < tid // find max entry with .serial <= xid.At
n := len(datav) n := len(datav)
i := n - 1 - sort.Search(n, func(i int) bool { i := n - 1 - sort.Search(n, func(i int) bool {
v := datav[n - 1 - i].serial < tid v := datav[n - 1 - i].serial <= xid.At
//fmt.Printf("@%d -> %v (@%d; %v)\n", i, v, n - 1 -i, tid) //fmt.Printf("@%d -> %v (@%d; %v)\n", i, v, n - 1 -i, xid.At)
return v return v
}) })
//fmt.Printf("i: %d n: %d\n", i, n) //fmt.Printf("i: %d n: %d\n", i, n)
if i == -1 { if i == -1 {
// tid < all .serial - no such transaction // xid.At < all .serial - no such transaction
return nil, 0, &zodb.ErrXidMissing{xid}
}
// check we have exact match if it was loadSerial
if !xid.TidBefore && datav[i].serial != xid.Tid {
return nil, 0, &zodb.ErrXidMissing{xid} return nil, 0, &zodb.ErrXidMissing{xid}
} }
...@@ -106,12 +97,8 @@ func (stor *tStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, seri ...@@ -106,12 +97,8 @@ func (stor *tStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, seri
var ioerr = errors.New("input/output error") var ioerr = errors.New("input/output error")
func xidlt(oid zodb.Oid, tid zodb.Tid) zodb.Xid { func xidat(oid zodb.Oid, tid zodb.Tid) zodb.Xid {
return zodb.Xid{Oid: oid, XTid: zodb.XTid{Tid: tid, TidBefore: true}} return zodb.Xid{Oid: oid, At: tid}
}
func xideq(oid zodb.Oid, tid zodb.Tid) zodb.Xid {
return zodb.Xid{Oid: oid, XTid: zodb.XTid{Tid: tid, TidBefore: false}}
} }
// tracer which collects tracing events from all needed-for-tests sources // tracer which collects tracing events from all needed-for-tests sources
...@@ -181,11 +168,11 @@ func TestCache(t *testing.T) { ...@@ -181,11 +168,11 @@ func TestCache(t *testing.T) {
} }
} }
checkRCE := func(rce *revCacheEntry, before, serial zodb.Tid, buf *zodb.Buf, err error) { checkRCE := func(rce *revCacheEntry, head, serial zodb.Tid, buf *zodb.Buf, err error) {
t.Helper() t.Helper()
bad := &bytes.Buffer{} bad := &bytes.Buffer{}
if rce.before != before { if rce.head != head {
fmt.Fprintf(bad, "before:\n%s\n", pretty.Compare(before, rce.before)) fmt.Fprintf(bad, "head:\n%s\n", pretty.Compare(head, rce.head))
} }
if rce.serial != serial { if rce.serial != serial {
fmt.Fprintf(bad, "serial:\n%s\n", pretty.Compare(serial, rce.serial)) fmt.Fprintf(bad, "serial:\n%s\n", pretty.Compare(serial, rce.serial))
...@@ -246,174 +233,174 @@ func TestCache(t *testing.T) { ...@@ -246,174 +233,174 @@ func TestCache(t *testing.T) {
checkMRU(0) checkMRU(0)
// load <3 -> new rce entry // load @2 -> new rce entry
checkLoad(xidlt(1,3), nil, 0, &zodb.ErrXidMissing{xidlt(1,3)}) checkLoad(xidat(1,2), nil, 0, &zodb.ErrXidMissing{xidat(1,2)})
oce1 := c.entryMap[1] oce1 := c.entryMap[1]
ok1(len(oce1.rcev) == 1) ok1(len(oce1.rcev) == 1)
rce1_b3 := oce1.rcev[0] rce1_h2 := oce1.rcev[0]
checkRCE(rce1_b3, 3, 0, nil, &zodb.ErrXidMissing{xidlt(1,3)}) checkRCE(rce1_h2, 2, 0, nil, &zodb.ErrXidMissing{xidat(1,2)})
checkMRU(0, rce1_b3) checkMRU(0, rce1_h2)
// load <4 -> <3 merged with <4 // load @3 -> 2] merged with 3]
checkLoad(xidlt(1,4), nil, 0, &zodb.ErrXidMissing{xidlt(1,4)}) checkLoad(xidat(1,3), nil, 0, &zodb.ErrXidMissing{xidat(1,3)})
ok1(len(oce1.rcev) == 1) ok1(len(oce1.rcev) == 1)
rce1_b4 := oce1.rcev[0] rce1_h3 := oce1.rcev[0]
ok1(rce1_b4 != rce1_b3) // rce1_b3 was merged into rce1_b4 ok1(rce1_h3 != rce1_h2) // rce1_h2 was merged into rce1_h3
checkRCE(rce1_b4, 4, 0, nil, &zodb.ErrXidMissing{xidlt(1,4)}) checkRCE(rce1_h3, 3, 0, nil, &zodb.ErrXidMissing{xidat(1,3)})
checkMRU(0, rce1_b4) checkMRU(0, rce1_h3)
// load <2 -> <2 merged with <4 // load @1 -> 1] merged with 3]
checkLoad(xidlt(1,2), nil, 0, &zodb.ErrXidMissing{xidlt(1,2)}) checkLoad(xidat(1,1), nil, 0, &zodb.ErrXidMissing{xidat(1,1)})
ok1(len(oce1.rcev) == 1) ok1(len(oce1.rcev) == 1)
ok1(oce1.rcev[0] == rce1_b4) ok1(oce1.rcev[0] == rce1_h3)
checkRCE(rce1_b4, 4, 0, nil, &zodb.ErrXidMissing{xidlt(1,4)}) checkRCE(rce1_h3, 3, 0, nil, &zodb.ErrXidMissing{xidat(1,3)})
checkMRU(0, rce1_b4) checkMRU(0, rce1_h3)
// load <6 -> new rce entry with data // load @5 -> new rce entry with data
checkLoad(xidlt(1,6), b(hello), 4, nil) checkLoad(xidat(1,5), b(hello), 4, nil)
ok1(len(oce1.rcev) == 2) ok1(len(oce1.rcev) == 2)
rce1_b6 := oce1.rcev[1] rce1_h5 := oce1.rcev[1]
checkRCE(rce1_b6, 6, 4, b(hello), nil) checkRCE(rce1_h5, 5, 4, b(hello), nil)
checkOCE(1, rce1_b4, rce1_b6) checkOCE(1, rce1_h3, rce1_h5)
checkMRU(5, rce1_b6, rce1_b4) checkMRU(5, rce1_h5, rce1_h3)
// load <5 -> <5 merged with <6 // load @4 -> 4] merged with 5]
checkLoad(xidlt(1,5), b(hello), 4, nil) checkLoad(xidat(1,4), b(hello), 4, nil)
checkOCE(1, rce1_b4, rce1_b6) checkOCE(1, rce1_h3, rce1_h5)
checkMRU(5, rce1_b6, rce1_b4) checkMRU(5, rce1_h5, rce1_h3)
// load <7 -> <6 merged with <7 // load @6 -> 5] merged with 6]
checkLoad(xidlt(1,7), b(hello), 4, nil) checkLoad(xidat(1,6), b(hello), 4, nil)
ok1(len(oce1.rcev) == 2) ok1(len(oce1.rcev) == 2)
rce1_b7 := oce1.rcev[1] rce1_h6 := oce1.rcev[1]
ok1(rce1_b7 != rce1_b6) ok1(rce1_h6 != rce1_h5)
checkRCE(rce1_b7, 7, 4, b(hello), nil) checkRCE(rce1_h6, 6, 4, b(hello), nil)
checkOCE(1, rce1_b4, rce1_b7) checkOCE(1, rce1_h3, rce1_h6)
checkMRU(5, rce1_b7, rce1_b4) checkMRU(5, rce1_h6, rce1_h3)
// load <8 -> ioerr + new rce // load @7 -> ioerr + new rce
checkLoad(xidlt(1,8), nil, 0, ioerr) checkLoad(xidat(1,7), nil, 0, ioerr)
ok1(len(oce1.rcev) == 3) ok1(len(oce1.rcev) == 3)
rce1_b8 := oce1.rcev[2] rce1_h7 := oce1.rcev[2]
checkRCE(rce1_b8, 8, 0, nil, ioerr) checkRCE(rce1_h7, 7, 0, nil, ioerr)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8) checkOCE(1, rce1_h3, rce1_h6, rce1_h7)
checkMRU(5, rce1_b8, rce1_b7, rce1_b4) checkMRU(5, rce1_h7, rce1_h6, rce1_h3)
// load <10 -> ioerr + new rce (IO errors are not merged) // load @9 -> ioerr + new rce (IO errors are not merged)
checkLoad(xidlt(1,10), nil, 0, ioerr) checkLoad(xidat(1,9), nil, 0, ioerr)
ok1(len(oce1.rcev) == 4) ok1(len(oce1.rcev) == 4)
rce1_b10 := oce1.rcev[3] rce1_h9 := oce1.rcev[3]
checkRCE(rce1_b10, 10, 0, nil, ioerr) checkRCE(rce1_h9, 9, 0, nil, ioerr)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9)
checkMRU(5, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(5, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// load <11 -> new data rce, not merged with ioerr @<10 // load @10 -> new data rce, not merged with ioerr at 9]
checkLoad(xidlt(1,11), b(world), 10, nil) checkLoad(xidat(1,10), b(world), 10, nil)
ok1(len(oce1.rcev) == 5) ok1(len(oce1.rcev) == 5)
rce1_b11 := oce1.rcev[4] rce1_h10 := oce1.rcev[4]
checkRCE(rce1_b11, 11, 10, b(world), nil) checkRCE(rce1_h10, 10, 10, b(world), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b11) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h10)
checkMRU(12, rce1_b11, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(12, rce1_h10, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// load <12 -> <11 merged with <12 // load @11 -> 10] merged with 11]
checkLoad(xidlt(1,12), b(world), 10, nil) checkLoad(xidat(1,11), b(world), 10, nil)
ok1(len(oce1.rcev) == 5) ok1(len(oce1.rcev) == 5)
rce1_b12 := oce1.rcev[4] rce1_h11 := oce1.rcev[4]
ok1(rce1_b12 != rce1_b11) ok1(rce1_h11 != rce1_h10)
checkRCE(rce1_b12, 12, 10, b(world), nil) checkRCE(rce1_h11, 11, 10, b(world), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b12) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h11)
checkMRU(12, rce1_b12, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(12, rce1_h11, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// simulate case where <14 (α) and <16 (β) were loaded in parallel, both are ready // simulate case where 13] (α) and 15] (β) were loaded in parallel, both are ready
// but <14 (α) takes oce lock first before <16 and so <12 is not yet merged // but 13] (α) takes oce lock first before 15] and so 11] is not yet merged
// with <16 -> <12 and <14 should be merged into <16. // with 15] -> 11] and 13] should be merged into 15].
// (manually add rce1_b16 so it is not merged with <12) // (manually add rce1_h15 so it is not merged with 11])
rce1_b16, new16 := c.lookupRCE(xidlt(1,16)) rce1_h15, new15 := c.lookupRCE(xidat(1,15))
ok1(new16) ok1(new15)
rce1_b16.serial = 10 rce1_h15.serial = 10
rce1_b16.buf = mkbuf(world) rce1_h15.buf = mkbuf(world)
// here: first half of loadRCE(<16) before close(<16.ready) // here: first half of loadRCE(15]) before close(15].ready)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b12, rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h11, rce1_h15)
ok1(!rce1_b16.loaded()) ok1(!rce1_h15.loaded())
checkMRU(12, rce1_b12, rce1_b10, rce1_b8, rce1_b7, rce1_b4) // no <16 yet checkMRU(12, rce1_h11, rce1_h9, rce1_h7, rce1_h6, rce1_h3) // no 15] yet
// (lookup <14 while <16 is not yet loaded so <16 is not picked // (lookup 13] while 15] is not yet loaded so 15] is not picked
// automatically at lookup phase) // automatically at lookup phase)
rce1_b14, new14 := c.lookupRCE(xidlt(1,14)) rce1_h13, new13 := c.lookupRCE(xidat(1,13))
ok1(new14) ok1(new13)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b12, rce1_b14, rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h11, rce1_h13, rce1_h15)
checkMRU(12, rce1_b12, rce1_b10, rce1_b8, rce1_b7, rce1_b4) // no <14 and <16 yet checkMRU(12, rce1_h11, rce1_h9, rce1_h7, rce1_h6, rce1_h3) // no <14 and <16 yet
// (now <16 becomes ready but not yet takes oce lock) // (now 15] becomes ready but not yet takes oce lock)
close(rce1_b16.ready) close(rce1_h15.ready)
ok1(rce1_b16.loaded()) ok1(rce1_h15.loaded())
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b12, rce1_b14, rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h11, rce1_h13, rce1_h15)
checkMRU(12, rce1_b12, rce1_b10, rce1_b8, rce1_b7, rce1_b4) // no <14 and <16 yet checkMRU(12, rce1_h11, rce1_h9, rce1_h7, rce1_h6, rce1_h3) // no 13] and 15] yet
// (<14 also becomes ready and takes oce lock first, merging <12 and <14 into <16. // (13] also becomes ready and takes oce lock first, merging 11] and 13] into 15].
// <16 did not yet took oce lock so c.size is temporarily reduced and // 15] did not yet took oce lock so c.size is temporarily reduced and
// <16 is not yet on LRU list) // 15] is not yet on LRU list)
c.loadRCE(ctx, rce1_b14, 1) c.loadRCE(ctx, rce1_h13, 1)
checkRCE(rce1_b14, 14, 10, b(world), nil) checkRCE(rce1_h13, 13, 10, b(world), nil)
checkRCE(rce1_b16, 16, 10, b(world), nil) checkRCE(rce1_h15, 15, 10, b(world), nil)
checkRCE(rce1_b12, 12, 10, b(world), nil) checkRCE(rce1_h11, 11, 10, b(world), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15)
checkMRU(5 /*was 12*/, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(5 /*was 12*/, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// (<16 takes oce lock and updates c.size and LRU list) // (15] takes oce lock and updates c.size and LRU list)
rce1_b16.ready = make(chan struct{}) // so loadRCE could run rce1_h15.ready = make(chan struct{}) // so loadRCE could run
c.loadRCE(ctx, rce1_b16, 1) c.loadRCE(ctx, rce1_h15, 1)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15)
checkMRU(12, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(12, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// similar race in between <17 and <18 but now β (<18) takes oce lock first: // similar race in between 16] and 17] but now β (17]) takes oce lock first:
rce1_b17, new17 := c.lookupRCE(xidlt(1,17)) rce1_h16, new16 := c.lookupRCE(xidat(1,16))
ok1(new16)
rce1_h17, new17 := c.lookupRCE(xidat(1,17))
ok1(new17) ok1(new17)
rce1_b18, new18 := c.lookupRCE(xidlt(1,18))
ok1(new18) // (16] loads but not yet takes oce lock)
rce1_h16.serial = 16
// (<17 loads but not yet takes oce lock) rce1_h16.buf = mkbuf(zz)
rce1_b17.serial = 16 close(rce1_h16.ready)
rce1_b17.buf = mkbuf(zz) ok1(rce1_h16.loaded())
close(rce1_b17.ready) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15, rce1_h16, rce1_h17)
ok1(rce1_b17.loaded()) checkMRU(12, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3) // no 16] and 17] yet
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16, rce1_b17, rce1_b18)
checkMRU(12, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) // no <17 and <18 yet // (17] loads and takes oce lock first - merge 16] with 17])
c.loadRCE(ctx, rce1_h17, 1)
// (<18 loads and takes oce lock first - merge <17 with <18) checkRCE(rce1_h17, 17, 16, b(zz), nil)
c.loadRCE(ctx, rce1_b18, 1) checkRCE(rce1_h16, 16, 16, b(zz), nil)
checkRCE(rce1_b18, 18, 16, b(zz), nil) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15, rce1_h17)
checkRCE(rce1_b17, 17, 16, b(zz), nil) checkMRU(14, rce1_h17, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16, rce1_b18)
checkMRU(14, rce1_b18, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) // load @19 -> 17] merged with 19]
checkLoad(xidat(1,19), b(zz), 16, nil)
// load =19 -> <18 merged with <20
checkLoad(xideq(1,19), nil, 0, &zodb.ErrXidMissing{xideq(1,19)})
ok1(len(oce1.rcev) == 6) ok1(len(oce1.rcev) == 6)
rce1_b20 := oce1.rcev[5] rce1_h19 := oce1.rcev[5]
ok1(rce1_b20 != rce1_b18) ok1(rce1_h19 != rce1_h17)
checkRCE(rce1_b20, 20, 16, b(zz), nil) checkRCE(rce1_h19, 19, 16, b(zz), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16, rce1_b20) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15, rce1_h19)
checkMRU(14, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(14, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// load =20 -> new <21 // load @20 -> new 20]
checkLoad(xideq(1,20), b(www), 20, nil) checkLoad(xidat(1,20), b(www), 20, nil)
ok1(len(oce1.rcev) == 7) ok1(len(oce1.rcev) == 7)
rce1_b21 := oce1.rcev[6] rce1_h20 := oce1.rcev[6]
checkRCE(rce1_b21, 21, 20, b(www), nil) checkRCE(rce1_h20, 20, 20, b(www), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16, rce1_b20, rce1_b21) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15, rce1_h19, rce1_h20)
checkMRU(17, rce1_b21, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(17, rce1_h20, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// load =21 -> <21 merged with <22 // load @21 -> 20] merged with 21]
checkLoad(xideq(1,21), nil, 0, &zodb.ErrXidMissing{xideq(1,21)}) checkLoad(xidat(1,21), b(www), 20, nil)
ok1(len(oce1.rcev) == 7) ok1(len(oce1.rcev) == 7)
rce1_b22 := oce1.rcev[6] rce1_h21 := oce1.rcev[6]
ok1(rce1_b22 != rce1_b21) ok1(rce1_h21 != rce1_h20)
checkRCE(rce1_b22, 22, 20, b(www), nil) checkRCE(rce1_h21, 21, 20, b(www), nil)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b10, rce1_b16, rce1_b20, rce1_b22) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h9, rce1_h15, rce1_h19, rce1_h21)
checkMRU(17, rce1_b22, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(17, rce1_h21, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// ---- verify rce lookup for must be cached entries ---- // ---- verify rce lookup for must be cached entries ----
...@@ -435,61 +422,47 @@ func TestCache(t *testing.T) { ...@@ -435,61 +422,47 @@ func TestCache(t *testing.T) {
} }
} }
checkLookup(xidlt(1,20), rce1_b20) checkLookup(xidat(1,19), rce1_h19)
checkLookup(xideq(1,19), rce1_b20) checkLookup(xidat(1,18), rce1_h19)
checkLookup(xidlt(1,19), rce1_b20) checkLookup(xidat(1,17), rce1_h19)
checkLookup(xideq(1,18), rce1_b20) checkLookup(xidat(1,16), rce1_h19)
checkLookup(xidlt(1,18), rce1_b20) checkLookup(xidat(1,15), rce1_h15)
checkLookup(xideq(1,17), rce1_b20) checkLookup(xidat(1,14), rce1_h15)
checkLookup(xidlt(1,17), rce1_b20) checkLookup(xidat(1,11), rce1_h15)
checkLookup(xideq(1,16), rce1_b20) checkLookup(xidat(1,10), rce1_h15)
checkLookup(xidlt(1,16), rce1_b16) checkLookup(xidat(1,9), rce1_h9)
checkLookup(xideq(1,15), rce1_b16)
checkLookup(xidlt(1,15), rce1_b16) // 8] must be separate from 7] and 9] because it is IO error there
checkLookup(xideq(1,12), rce1_b16) rce1_h8, new8 := c.lookupRCE(xidat(1,8))
checkLookup(xidlt(1,12), rce1_b16) ok1(new8)
checkLookup(xideq(1,11), rce1_b16) c.loadRCE(ctx, rce1_h8, 1)
checkLookup(xidlt(1,11), rce1_b16) checkRCE(rce1_h8, 8, 0, nil, ioerr)
checkLookup(xideq(1,10), rce1_b16) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h8, rce1_h9, rce1_h15, rce1_h19, rce1_h21)
checkLookup(xidlt(1,10), rce1_b10) checkMRU(17, rce1_h8, rce1_h21, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
// <9 must be separate from <8 and <10 because it is IO error there checkLookup(xidat(1,8), rce1_h8)
rce1_b9, new9 := c.lookupRCE(xidlt(1,9)) checkLookup(xidat(1,7), rce1_h7)
ok1(new9)
c.loadRCE(ctx, rce1_b9, 1)
checkRCE(rce1_b9, 9, 0, nil, ioerr)
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b9, rce1_b10, rce1_b16, rce1_b20, rce1_b22)
checkMRU(17, rce1_b9, rce1_b22, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4)
checkLookup(xideq(1,8), rce1_b9)
checkLookup(xidlt(1,8), rce1_b8)
// have data exact and inexact hits // have data exact and inexact hits
checkLookup(xideq(1,7), rce1_b8) checkLookup(xidat(1,7), rce1_h7)
checkLookup(xidlt(1,7), rce1_b7) checkLookup(xidat(1,6), rce1_h6)
checkLookup(xideq(1,6), rce1_b7) checkLookup(xidat(1,5), rce1_h6)
checkLookup(xidlt(1,6), rce1_b7) checkLookup(xidat(1,4), rce1_h6)
checkLookup(xideq(1,5), rce1_b7)
checkLookup(xidlt(1,5), rce1_b7)
checkLookup(xideq(1,4), rce1_b7)
// nodata exact and inexact hits // nodata exact and inexact hits
checkLookup(xidlt(1,4), rce1_b4) checkLookup(xidat(1,3), rce1_h3)
checkLookup(xideq(1,3), rce1_b4) checkLookup(xidat(1,2), rce1_h3)
checkLookup(xidlt(1,3), rce1_b4) checkLookup(xidat(1,1), rce1_h3)
checkLookup(xideq(1,2), rce1_b4) checkLookup(xidat(1,0), rce1_h3)
checkLookup(xidlt(1,2), rce1_b4)
checkLookup(xideq(1,1), rce1_b4)
checkLookup(xidlt(1,1), rce1_b4)
// ---- verify how LRU changes for in-cache loads ---- // ---- verify how LRU changes for in-cache loads ----
checkMRU(17, rce1_b9, rce1_b22, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b7, rce1_b4) checkMRU(17, rce1_h8, rce1_h21, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h6, rce1_h3)
checkLoad(xidlt(1,7), b(hello), 4, nil) checkLoad(xidat(1,6), b(hello), 4, nil)
checkMRU(17, rce1_b7, rce1_b9, rce1_b22, rce1_b20, rce1_b16, rce1_b10, rce1_b8, rce1_b4) checkMRU(17, rce1_h6, rce1_h8, rce1_h21, rce1_h19, rce1_h15, rce1_h9, rce1_h7, rce1_h3)
checkLoad(xidlt(1,16), b(world), 10, nil) checkLoad(xidat(1,15), b(world), 10, nil)
checkMRU(17, rce1_b16, rce1_b7, rce1_b9, rce1_b22, rce1_b20, rce1_b10, rce1_b8, rce1_b4) checkMRU(17, rce1_h15, rce1_h6, rce1_h8, rce1_h21, rce1_h19, rce1_h9, rce1_h7, rce1_h3)
// ---- verify LRU eviction ---- // ---- verify LRU eviction ----
...@@ -511,46 +484,46 @@ func TestCache(t *testing.T) { ...@@ -511,46 +484,46 @@ func TestCache(t *testing.T) {
gcstart := &evCacheGCStart{c} gcstart := &evCacheGCStart{c}
gcfinish := &evCacheGCFinish{c} gcfinish := &evCacheGCFinish{c}
checkOCE(1, rce1_b4, rce1_b7, rce1_b8, rce1_b9, rce1_b10, rce1_b16, rce1_b20, rce1_b22) checkOCE(1, rce1_h3, rce1_h6, rce1_h7, rce1_h8, rce1_h9, rce1_h15, rce1_h19, rce1_h21)
checkMRU(17, rce1_b16, rce1_b7, rce1_b9, rce1_b22, rce1_b20, rce1_b10, rce1_b8, rce1_b4) checkMRU(17, rce1_h15, rce1_h6, rce1_h8, rce1_h21, rce1_h19, rce1_h9, rce1_h7, rce1_h3)
go c.SetSizeMax(16) // < c.size by 1 -> should trigger gc go c.SetSizeMax(16) // < c.size by 1 -> should trigger gc
tc.Expect(gcstart, gcfinish) tc.Expect(gcstart, gcfinish)
// evicted: // evicted:
// - <4 (lru.1, nodata, size=0) XXX ok to evict nodata & friends? // - 3] (lru.1, nodata, size=0) XXX ok to evict nodata & friends?
// - <8 (lru.2, ioerr, size=0) // - 7] (lru.2, ioerr, size=0)
// - <10 (lru.3, ioerr, size=0) // - 9] (lru.3, ioerr, size=0)
// - <20 (lru.4, zz, size=2) // - 19] (lru.4, zz, size=2)
checkOCE(1, rce1_b7, rce1_b9, rce1_b16, rce1_b22) checkOCE(1, rce1_h6, rce1_h8, rce1_h15, rce1_h21)
checkMRU(15, rce1_b16, rce1_b7, rce1_b9, rce1_b22) checkMRU(15, rce1_h15, rce1_h6, rce1_h8, rce1_h21)
// reload <20 -> <22 should be evicted // reload 19] -> 21] should be evicted
go c.Load(ctx, xidlt(1,20)) go c.Load(ctx, xidat(1,19))
tc.Expect(gcstart, gcfinish) tc.Expect(gcstart, gcfinish)
// - evicted <22 (lru.1, www, size=3) // - evicted 21] (lru.1, www, size=3)
// - loaded <20 (zz, size=2) // - loaded 19] (zz, size=2)
ok1(len(oce1.rcev) == 4) ok1(len(oce1.rcev) == 4)
rce1_b20_2 := oce1.rcev[3] rce1_h19_2 := oce1.rcev[3]
ok1(rce1_b20_2 != rce1_b20) ok1(rce1_h19_2 != rce1_h19)
checkRCE(rce1_b20_2, 20, 16, b(zz), nil) checkRCE(rce1_h19_2, 19, 16, b(zz), nil)
checkOCE(1, rce1_b7, rce1_b9, rce1_b16, rce1_b20_2) checkOCE(1, rce1_h6, rce1_h8, rce1_h15, rce1_h19_2)
checkMRU(14, rce1_b20_2, rce1_b16, rce1_b7, rce1_b9) checkMRU(14, rce1_h19_2, rce1_h15, rce1_h6, rce1_h8)
// load big <78 -> several rce must be evicted // load big 77] -> several rce must be evicted
go c.Load(ctx, xidlt(1,78)) go c.Load(ctx, xidat(1,77))
tc.Expect(gcstart, gcfinish) tc.Expect(gcstart, gcfinish)
// - evicted <9 (lru.1, ioerr, size=0) // - evicted 8] (lru.1, ioerr, size=0)
// - evicted <7 (lru.2, hello, size=5) // - evicted 6] (lru.2, hello, size=5)
// - evicted <16 (lru.3, world, size=7) // - evicted 15] (lru.3, world, size=7)
// - loaded <78 (big, size=10) // - loaded 77] (big, size=10)
ok1(len(oce1.rcev) == 2) ok1(len(oce1.rcev) == 2)
rce1_b78 := oce1.rcev[1] rce1_h77 := oce1.rcev[1]
checkRCE(rce1_b78, 78, 77, b(big), nil) checkRCE(rce1_h77, 77, 77, b(big), nil)
checkOCE(1, rce1_b20_2, rce1_b78) checkOCE(1, rce1_h19_2, rce1_h77)
checkMRU(12, rce1_b78, rce1_b20_2) checkMRU(12, rce1_h77, rce1_h19_2)
// sizeMax=0 evicts everything from cache // sizeMax=0 evicts everything from cache
go c.SetSizeMax(0) go c.SetSizeMax(0)
...@@ -561,7 +534,7 @@ func TestCache(t *testing.T) { ...@@ -561,7 +534,7 @@ func TestCache(t *testing.T) {
// and still loading works (because even if though rce's are evicted // and still loading works (because even if though rce's are evicted
// they stay live while someone user waits and uses it) // they stay live while someone user waits and uses it)
checkLoad(xidlt(1,5), b(hello), 4, nil) checkLoad(xidat(1,4), b(hello), 4, nil)
tc.Expect(gcstart, gcfinish) tc.Expect(gcstart, gcfinish)
checkOCE(1) checkOCE(1)
checkMRU(0) checkMRU(0)
......
...@@ -168,13 +168,8 @@ func (fs *FileStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, tid ...@@ -168,13 +168,8 @@ func (fs *FileStorage) Load(_ context.Context, xid zodb.Xid) (buf *zodb.Buf, tid
} }
func (fs *FileStorage) _Load(dh *DataHeader, xid zodb.Xid) (*zodb.Buf, zodb.Tid, error) { func (fs *FileStorage) _Load(dh *DataHeader, xid zodb.Xid) (*zodb.Buf, zodb.Tid, error) {
tidBefore := xid.XTid.Tid // search backwards for when we first have data record with tid satisfying xid.At
if !xid.XTid.TidBefore { for {
tidBefore++ // XXX recheck this is ok wrt overflow
}
// search backwards for when we first have data record with tid satisfying xid.XTid
for dh.Tid >= tidBefore {
err := dh.LoadPrevRev(fs.file) err := dh.LoadPrevRev(fs.file)
if err != nil { if err != nil {
if err == io.EOF { if err == io.EOF {
...@@ -186,11 +181,10 @@ func (fs *FileStorage) _Load(dh *DataHeader, xid zodb.Xid) (*zodb.Buf, zodb.Tid, ...@@ -186,11 +181,10 @@ func (fs *FileStorage) _Load(dh *DataHeader, xid zodb.Xid) (*zodb.Buf, zodb.Tid,
return nil, 0, err return nil, 0, err
} }
}
// found dh.Tid < tidBefore; check it really satisfies xid.XTid if dh.Tid <= xid.At {
if !xid.XTid.TidBefore && dh.Tid != xid.XTid.Tid { break
return nil, 0, &zodb.ErrXidMissing{Xid: xid} }
} }
// even if we will scan back via backpointers, the tid returned should // even if we will scan back via backpointers, the tid returned should
...@@ -274,7 +268,7 @@ func (zi *zIter) NextData(_ context.Context) (*zodb.DataInfo, error) { ...@@ -274,7 +268,7 @@ func (zi *zIter) NextData(_ context.Context) (*zodb.DataInfo, error) {
zi.datai.Tid = zi.iter.Datah.Tid zi.datai.Tid = zi.iter.Datah.Tid
// NOTE dh.LoadData() changes dh state while going through backpointers - // NOTE dh.LoadData() changes dh state while going through backpointers -
// - need to use separate dh because of this // - need to use separate dh because of this.
zi.dhLoading = zi.iter.Datah zi.dhLoading = zi.iter.Datah
if zi.dataBuf != nil { if zi.dataBuf != nil {
zi.dataBuf.Release() zi.dataBuf.Release()
...@@ -286,7 +280,11 @@ func (zi *zIter) NextData(_ context.Context) (*zodb.DataInfo, error) { ...@@ -286,7 +280,11 @@ func (zi *zIter) NextData(_ context.Context) (*zodb.DataInfo, error) {
} }
zi.datai.Data = zi.dataBuf.Data zi.datai.Data = zi.dataBuf.Data
zi.datai.DataTid = zi.dhLoading.Tid if zi.dhLoading.Tid != zi.datai.Tid {
zi.datai.DataTidHint = zi.dhLoading.Tid
} else {
zi.datai.DataTidHint = 0
}
return &zi.datai, nil return &zi.datai, nil
} }
......
...@@ -44,7 +44,7 @@ type txnEntry struct { ...@@ -44,7 +44,7 @@ type txnEntry struct {
Header DataHeader Header DataHeader
rawData []byte // what is on disk, e.g. it can be backpointer rawData []byte // what is on disk, e.g. it can be backpointer
userData []byte // data client should see on load; `sameAsRaw` means same as RawData userData []byte // data client should see on load; `sameAsRaw` means same as RawData
dataTid zodb.Tid // data tid client should see on iter; 0 means same as Header.Tid DataTidHint zodb.Tid // data tid client should see on iter
} }
var sameAsRaw = []byte{0} var sameAsRaw = []byte{0}
...@@ -58,15 +58,6 @@ func (txe *txnEntry) Data() []byte { ...@@ -58,15 +58,6 @@ func (txe *txnEntry) Data() []byte {
return data return data
} }
// DataTid returns data tid a client should see
func (txe *txnEntry) DataTid() zodb.Tid {
dataTid := txe.dataTid
if dataTid == 0 {
dataTid = txe.Header.Tid
}
return dataTid
}
// state of an object in the database for some particular revision // state of an object in the database for some particular revision
type objState struct { type objState struct {
tid zodb.Tid tid zodb.Tid
...@@ -137,29 +128,26 @@ func TestLoad(t *testing.T) { ...@@ -137,29 +128,26 @@ func TestLoad(t *testing.T) {
// XXX check Load finds data at correct .Pos / etc ? // XXX check Load finds data at correct .Pos / etc ?
// loadSerial // ~ loadSerial
xid := zodb.Xid{zodb.XTid{txh.Tid, false}, txh.Oid} xid := zodb.Xid{txh.Tid, txh.Oid}
checkLoad(t, fs, xid, objState{txh.Tid, txe.Data()}) checkLoad(t, fs, xid, objState{txh.Tid, txe.Data()})
// loadBefore // ~ loadBefore
xid = zodb.Xid{zodb.XTid{txh.Tid, true}, txh.Oid} xid = zodb.Xid{txh.Tid - 1, txh.Oid}
expect, ok := before[txh.Oid] expect, ok := before[txh.Oid]
if ok { if ok {
checkLoad(t, fs, xid, expect) checkLoad(t, fs, xid, expect)
} }
// loadBefore to get current record
xid.Tid += 1
checkLoad(t, fs, xid, objState{txh.Tid, txe.Data()})
before[txh.Oid] = objState{txh.Tid, txe.Data()} before[txh.Oid] = objState{txh.Tid, txe.Data()}
} }
} }
// loadBefore with TidMax // load at ∞ with TidMax
// XXX should we get "no such transaction" with at > head?
for oid, expect := range before { for oid, expect := range before {
xid := zodb.Xid{zodb.XTid{zodb.TidMax, true}, oid} xid := zodb.Xid{zodb.TidMax, oid}
checkLoad(t, fs, xid, expect) checkLoad(t, fs, xid, expect)
} }
} }
...@@ -268,8 +256,8 @@ func testIterate(t *testing.T, fs *FileStorage, tidMin, tidMax zodb.Tid, expectv ...@@ -268,8 +256,8 @@ func testIterate(t *testing.T, fs *FileStorage, tidMin, tidMax zodb.Tid, expectv
dataErrorf("data mismatch:\nhave %q\nwant %q", datai.Data, txe.Data()) dataErrorf("data mismatch:\nhave %q\nwant %q", datai.Data, txe.Data())
} }
if datai.DataTid != txe.DataTid() { if datai.DataTidHint != txe.DataTidHint {
dataErrorf("data tid mismatch: have %v; want %v", datai.DataTid, txe.DataTid()) dataErrorf("data tid hint mismatch: have %v; want %v", datai.DataTidHint, txe.DataTidHint)
} }
} }
} }
......
...@@ -119,11 +119,11 @@ def main(): ...@@ -119,11 +119,11 @@ def main():
datatid = "/* deleted */ 0" datatid = "/* deleted */ 0"
else: else:
data = "[]byte(%s)" % escapeqq(drec.data) data = "[]byte(%s)" % escapeqq(drec.data)
datatid = hex64(drec.data_txn) datatid = "/* copy from */ " + hex64(drec.data_txn)
else: else:
rawdata = drec.data rawdata = drec.data
data = "/* same as ^^^ */ sameAsRaw" data = "/* same as ^^^ */ sameAsRaw"
datatid = "/* same as ^^^ */ 0" datatid = "/* no copy */ 0"
emit("\t\t\t\t[]byte(%s)," % escapeqq(rawdata)) emit("\t\t\t\t[]byte(%s)," % escapeqq(rawdata))
emit("\t\t\t\t%s," % data) emit("\t\t\t\t%s," % data)
......
...@@ -44,7 +44,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -44,7 +44,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04s."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04s."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -74,7 +74,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -74,7 +74,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04U\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86Qss."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04U\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86Qss."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -87,7 +87,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -87,7 +87,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04f0.0q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04f0.0q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -117,7 +117,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -117,7 +117,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x07h\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x07h\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -130,7 +130,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -130,7 +130,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d0.1q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d0.1q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -160,7 +160,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -160,7 +160,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x07h\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x08h\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x07h\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x08h\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -173,7 +173,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -173,7 +173,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g0.2q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g0.2q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -203,7 +203,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -203,7 +203,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d0.3q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d0.3q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -233,7 +233,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -233,7 +233,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x07h\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x08h\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\th\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x07h\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x08h\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\th\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -246,7 +246,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -246,7 +246,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b0.4q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b0.4q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -276,7 +276,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -276,7 +276,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x07h\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x08h\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\th\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\nh\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x07h\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\x08h\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\th\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\nh\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -289,7 +289,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -289,7 +289,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04a0.5q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04a0.5q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -319,7 +319,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -319,7 +319,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b0.6q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b0.6q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -349,7 +349,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -349,7 +349,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04a0.7q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04a0.7q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -379,7 +379,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -379,7 +379,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x07h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\x08h\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\th\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\nh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0bh\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x07h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\x08h\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\th\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\nh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0bh\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -392,7 +392,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -392,7 +392,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e0.8q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e0.8q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -422,7 +422,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -422,7 +422,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e0.9q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e0.9q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -452,7 +452,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -452,7 +452,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.10q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.10q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -482,7 +482,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -482,7 +482,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.11q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.11q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -512,7 +512,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -512,7 +512,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05f0.12q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05f0.12q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -542,7 +542,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -542,7 +542,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.13q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.13q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -572,7 +572,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -572,7 +572,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.14q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.14q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -602,7 +602,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -602,7 +602,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e0.15q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e0.15q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -632,7 +632,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -632,7 +632,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.16q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.16q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -662,7 +662,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -662,7 +662,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05b0.17q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05b0.17q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -692,7 +692,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -692,7 +692,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05b0.18q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05b0.18q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -722,7 +722,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -722,7 +722,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.19q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a0.19q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -752,7 +752,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -752,7 +752,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d0.20q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d0.20q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -782,7 +782,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -782,7 +782,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d0.21q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d0.21q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -812,7 +812,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -812,7 +812,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x07q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x07q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -825,7 +825,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -825,7 +825,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.22q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.22q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -855,7 +855,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -855,7 +855,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.23q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.23q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -885,7 +885,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -885,7 +885,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.24q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.24q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -915,7 +915,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -915,7 +915,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x00\x00\x00\x00\x00\x00\x17\xd6"), []byte("\x00\x00\x00\x00\x00\x00\x17\xd6"),
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.22q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c0.22q\x02."),
0x0285cbaca4444480, /* copy from */ 0x0285cbaca4444480,
}, },
}, },
}, },
...@@ -945,7 +945,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -945,7 +945,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x00\x00\x00\x00\x00\x00\x0eB"), []byte("\x00\x00\x00\x00\x00\x00\x0eB"),
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.11q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g0.11q\x02."),
0x0285cbac70a3d733, /* copy from */ 0x0285cbac70a3d733,
}, },
}, },
}, },
...@@ -975,7 +975,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -975,7 +975,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x08q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x08q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\x06q\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -988,7 +988,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -988,7 +988,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x03c0*q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x03c0*q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1048,7 +1048,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1048,7 +1048,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.0q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.0q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1078,7 +1078,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1078,7 +1078,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e1.1q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04e1.1q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1108,7 +1108,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1108,7 +1108,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b1.2q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04b1.2q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1138,7 +1138,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1138,7 +1138,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g1.3q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g1.3q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1168,7 +1168,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1168,7 +1168,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g1.4q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04g1.4q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1198,7 +1198,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1198,7 +1198,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.5q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.5q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1228,7 +1228,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1228,7 +1228,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04f1.6q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04f1.6q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1258,7 +1258,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1258,7 +1258,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d1.7q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04d1.7q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1288,7 +1288,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1288,7 +1288,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.8q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.8q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1318,7 +1318,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1318,7 +1318,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.9q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x04c1.9q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1348,7 +1348,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1348,7 +1348,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.10q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.10q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1378,7 +1378,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1378,7 +1378,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.11q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.11q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1408,7 +1408,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1408,7 +1408,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c1.12q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05c1.12q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1438,7 +1438,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1438,7 +1438,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.13q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.13q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1468,7 +1468,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1468,7 +1468,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.14q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.14q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1498,7 +1498,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1498,7 +1498,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.15q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.15q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1528,7 +1528,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1528,7 +1528,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d1.16q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05d1.16q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1558,7 +1558,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1558,7 +1558,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.17q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.17q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1588,7 +1588,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1588,7 +1588,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05f1.18q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05f1.18q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1618,7 +1618,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1618,7 +1618,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.19q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.19q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1648,7 +1648,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1648,7 +1648,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.20q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.20q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1678,7 +1678,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1678,7 +1678,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.21q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05g1.21q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1708,7 +1708,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1708,7 +1708,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.22q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.22q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1738,7 +1738,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1738,7 +1738,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.23q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.23q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1768,7 +1768,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1768,7 +1768,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.24q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.24q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
...@@ -1798,7 +1798,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1798,7 +1798,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x00\x00\x00\x00\x00\x00(Z"), []byte("\x00\x00\x00\x00\x00\x00(Z"),
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.13q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05e1.13q\x02."),
0x0285cbad3a740e19, /* copy from */ 0x0285cbad3a740e19,
}, },
}, },
}, },
...@@ -1828,7 +1828,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1828,7 +1828,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x00\x00\x00\x00\x00\x000\x16"), []byte("\x00\x00\x00\x00\x00\x000\x16"),
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.23q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x05a1.23q\x02."),
0x0285cbad6962fd19, /* copy from */ 0x0285cbad6962fd19,
}, },
}, },
}, },
...@@ -1858,7 +1858,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1858,7 +1858,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x08q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\tq\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."), []byte("\x80\x02cpersistent.mapping\nPersistentMapping\nq\x01.\x80\x02}q\x02U\x04dataq\x03}q\x04(U\x01aU\x08\x00\x00\x00\x00\x00\x00\x00\x05q\x05czodbtools.test.gen_testdata\nObject\nq\x06\x86QU\x01cU\x08\x00\x00\x00\x00\x00\x00\x00\x08q\x07h\x06\x86QU\x01bU\x08\x00\x00\x00\x00\x00\x00\x00\x04q\x08h\x06\x86QU\x01eU\x08\x00\x00\x00\x00\x00\x00\x00\tq\th\x06\x86QU\x01dU\x08\x00\x00\x00\x00\x00\x00\x00\x02q\nh\x06\x86QU\x01gU\x08\x00\x00\x00\x00\x00\x00\x00\x03q\x0bh\x06\x86QU\x01fU\x08\x00\x00\x00\x00\x00\x00\x00\x01q\x0ch\x06\x86Qus."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
{ {
DataHeader{ DataHeader{
...@@ -1871,7 +1871,7 @@ var _1fs_dbEntryv = [...]dbEntry{ ...@@ -1871,7 +1871,7 @@ var _1fs_dbEntryv = [...]dbEntry{
}, },
[]byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x03e1*q\x02."), []byte("\x80\x02czodbtools.test.gen_testdata\nObject\nq\x01.\x80\x02U\x03e1*q\x02."),
/* same as ^^^ */ sameAsRaw, /* same as ^^^ */ sameAsRaw,
/* same as ^^^ */ 0, /* no copy */ 0,
}, },
}, },
}, },
......
...@@ -61,60 +61,26 @@ func (oid Oid) XFmtString(b []byte) []byte { ...@@ -61,60 +61,26 @@ func (oid Oid) XFmtString(b []byte) []byte {
return xfmt.AppendHex016(b, uint64(oid)) return xfmt.AppendHex016(b, uint64(oid))
} }
// bint converts bool to int with true => 1; false => 0.
//
// XXX place = ?
func bint(b bool) int {
if b {
return 1
} else {
return 0
}
}
// String converts xtid to string.
//
// Default xtid string representation is:
//
// - "=" or "<" character depending on whether xtid represents exact or "tid before" query
// - tid
//
// e.g.
//
// =0285cbac258bf266 - exactly 0285cbac258bf266
// <0285cbac258bf266 - before 0285cbac258bf266
//
// See also: ParseXTid.
func (xtid XTid) String() string {
// XXX also print "tid:" prefix ?
return fmt.Sprintf("%c%v", "=<"[bint(xtid.TidBefore)], xtid.Tid)
}
// String converts xid to string. // String converts xid to string.
// //
// Default xid string representation is: // Default xid string representation is:
// //
// - string of xtid // - string of at
// - ":" // - ":"
// - string of oid // - string of oid
// //
// e.g. // e.g.
// //
// =0285cbac258bf266:0000000000000001 - oid 1 at exactly 0285cbac258bf266 transaction // 0285cbac258bf266:0000000000000001 - oid 1 at first newest transaction changing it with tid <= 0285cbac258bf266
// <0285cbac258bf266:0000000000000001 - oid 1 at first newest transaction changing it with tid < 0285cbac258bf266
// //
// See also: ParseXid. // See also: ParseXid.
func (xid Xid) String() string { func (xid Xid) String() string {
return xid.XTid.String() + ":" + xid.Oid.String() return xid.At.String() + ":" + xid.Oid.String()
} }
/* TODO reenable? /* TODO reenable?
func (xtid XTid) XFmtString(b []byte) []byte {
b .C("=<"[bint(xtid.TidBefore)]) .V(xtid.Tid)
}
func (xid Xid) XFmtString(b xfmt.Buffer) xfmt.Buffer { func (xid Xid) XFmtString(b xfmt.Buffer) xfmt.Buffer {
b .V(xid.XTid) .C(':') .V(xid.Oid) b .V(xid.At) .C(':') .V(xid.Oid)
} }
*/ */
...@@ -151,55 +117,23 @@ func ParseOid(s string) (Oid, error) { ...@@ -151,55 +117,23 @@ func ParseOid(s string) (Oid, error) {
return Oid(x), err return Oid(x), err
} }
// ParseXTid parses xtid from string.
//
// See also: XTid.String .
func ParseXTid(s string) (XTid, error) {
if len(s) < 1 {
goto Error
}
{
var tidBefore bool
switch s[0] {
case '<':
tidBefore = true
case '=':
tidBefore = false
default:
goto Error
}
tid, err := ParseTid(s[1:])
if err != nil {
goto Error
}
return XTid{tid, tidBefore}, nil
}
Error:
return XTid{}, fmt.Errorf("xtid %q invalid", s)
}
// ParseXid parses xid from string. // ParseXid parses xid from string.
// //
// See also: Xid.String . // See also: Xid.String .
func ParseXid(s string) (Xid, error) { func ParseXid(s string) (Xid, error) {
xtids, oids, err := xstrings.Split2(s, ":") ats, oids, err := xstrings.Split2(s, ":")
if err != nil { if err != nil {
goto Error goto Error
} }
{ {
xtid, err1 := ParseXTid(xtids) at, err1 := ParseTid(ats)
oid, err2 := ParseOid(oids) oid, err2 := ParseOid(oids)
if err1 != nil || err2 != nil { if err1 != nil || err2 != nil {
goto Error goto Error
} }
return Xid{xtid, oid}, nil return Xid{at, oid}, nil
} }
Error: Error:
......
...@@ -45,30 +45,11 @@ func TestParseHex64(t *testing.T) { ...@@ -45,30 +45,11 @@ func TestParseHex64(t *testing.T) {
} }
} }
func TestParseXTid(t *testing.T) {
var testv = []struct {in string; xtid XTid; estr string} {
{"", XTid{}, `xtid "" invalid`},
{"a", XTid{}, `xtid "a" invalid`},
{"0123456789abcdef", XTid{}, `xtid "0123456789abcdef" invalid`}, // XXX or let it be < by default ?
{"z0123456789abcdef", XTid{}, `xtid "z0123456789abcdef" invalid`},
{"=0123456789abcdef", XTid{0x0123456789abcdef, false}, ""},
{"<0123456789abcdef", XTid{0x0123456789abcdef, true}, ""},
}
for _, tt := range testv {
xtid, err := ParseXTid(tt.in)
if !(xtid == tt.xtid && estr(err) == tt.estr) {
t.Errorf("parsextid: %v: test error:\nhave: %v %q\nwant: %v %q",
tt.in, xtid, err, tt.xtid, tt.estr)
}
}
}
func TestParseXid(t *testing.T) { func TestParseXid(t *testing.T) {
var testv = []struct {in string; xid Xid; estr string} { var testv = []struct {in string; xid Xid; estr string} {
{"", Xid{}, `xid "" invalid`}, {"", Xid{}, `xid "" invalid`},
{"a", Xid{}, `xid "a" invalid`}, {"a", Xid{}, `xid "a" invalid`},
{"0123456789abcdef", Xid{}, `xid "0123456789abcdef" invalid`}, // XXX or let it be < by default ? {"0123456789abcdef", Xid{}, `xid "0123456789abcdef" invalid`},
{"z0123456789abcdef", Xid{}, `xid "z0123456789abcdef" invalid`}, {"z0123456789abcdef", Xid{}, `xid "z0123456789abcdef" invalid`},
{"=0123456789abcdef", Xid{}, `xid "=0123456789abcdef" invalid`}, {"=0123456789abcdef", Xid{}, `xid "=0123456789abcdef" invalid`},
{"<0123456789abcdef", Xid{}, `xid "<0123456789abcdef" invalid`}, {"<0123456789abcdef", Xid{}, `xid "<0123456789abcdef" invalid`},
...@@ -76,8 +57,9 @@ func TestParseXid(t *testing.T) { ...@@ -76,8 +57,9 @@ func TestParseXid(t *testing.T) {
{"=0123456789abcdef|fedcba9876543210", Xid{}, `xid "=0123456789abcdef|fedcba9876543210" invalid`}, {"=0123456789abcdef|fedcba9876543210", Xid{}, `xid "=0123456789abcdef|fedcba9876543210" invalid`},
{"<0123456789abcdef|fedcba9876543210", Xid{}, `xid "<0123456789abcdef|fedcba9876543210" invalid`}, {"<0123456789abcdef|fedcba9876543210", Xid{}, `xid "<0123456789abcdef|fedcba9876543210" invalid`},
{"=0123456789abcdef:fedcba9876543210", Xid{XTid{0x0123456789abcdef, false}, 0xfedcba9876543210}, ""}, {"=0123456789abcdef:fedcba9876543210", Xid{}, `xid "=0123456789abcdef:fedcba9876543210" invalid`},
{"<0123456789abcdef:fedcba9876543210", Xid{XTid{0x0123456789abcdef, true}, 0xfedcba9876543210}, ""}, {"<0123456789abcdef:fedcba9876543210", Xid{}, `xid "<0123456789abcdef:fedcba9876543210" invalid`},
{"0123456789abcdef:fedcba9876543210", Xid{0x0123456789abcdef, 0xfedcba9876543210}, ""},
} }
for _, tt := range testv { for _, tt := range testv {
......
...@@ -34,10 +34,12 @@ import ( ...@@ -34,10 +34,12 @@ import (
// Tid is transaction identifier. // Tid is transaction identifier.
// //
// In ZODB transaction identifiers are unique 64-bit integers connected to time // In ZODB transaction identifiers are unique 64-bit integers corresponding to
// when corresponding transaction was created. // time when transaction in question was committed.
// //
// See also: XTid. // This way tid can also be used to specify whole database state constructed
// by all cumulated transaction changes from database beginning up to, and
// including, transaction specified by tid.
type Tid uint64 type Tid uint64
// ZODB/py defines maxtid to be max signed int64 since Jun 7 2016: // ZODB/py defines maxtid to be max signed int64 since Jun 7 2016:
...@@ -49,13 +51,29 @@ const TidMax Tid = 1<<63 - 1 // 0x7fffffffffffffff ...@@ -49,13 +51,29 @@ const TidMax Tid = 1<<63 - 1 // 0x7fffffffffffffff
// Oid is object identifier. // Oid is object identifier.
// //
// In ZODB objects are uniquely identified by 64-bit integer. // In ZODB objects are uniquely identified by 64-bit integer.
// Every object can have several revisions - each committed in different transaction. // An object can have several revisions - each committed in different transaction.
// The combination of object identifier and particular transaction (serial) // The combination of object identifier and particular transaction (serial)
// uniquely addresses corresponding data record. // uniquely addresses corresponding data record.
// //
// See also: Xid. // See also: Xid.
type Oid uint64 type Oid uint64
// Xid is "extended" oid - that fully specifies object and query for its revision.
//
// At specifies whole database state at which object identified with Oid should
// be looked up. The object revision is taken from latest transaction modifying
// the object with tid <= At.
//
// Note that Xids are not unique - the same object revision can be addressed
// with several xids.
//
// See also: Tid, Oid.
type Xid struct {
At Tid
Oid Oid
}
// TxnInfo is metadata information about one transaction. // TxnInfo is metadata information about one transaction.
type TxnInfo struct { type TxnInfo struct {
Tid Tid Tid Tid
...@@ -75,11 +93,16 @@ type DataInfo struct { ...@@ -75,11 +93,16 @@ type DataInfo struct {
Tid Tid Tid Tid
Data []byte // nil means: deleted XXX -> *Buf ? Data []byte // nil means: deleted XXX -> *Buf ?
// original tid data was committed at (e.g. in case of undo) // DataTidHint is optional hint from a storage that the same data was
// already originally committed in earlier transaction, for example in
// case of undo. It is 0 if there is no such hint.
//
// Storages are not obliged to provide this hint, and in particular it
// is valid for a storage to always return this as zero.
// //
// FIXME we don't really need this and this unnecessarily constraints interfaces. // In ZODB/py world this originates from
// originates from: https://github.com/zopefoundation/ZODB/commit/2b0c9aa4 // https://github.com/zopefoundation/ZODB/commit/2b0c9aa4.
DataTid Tid DataTidHint Tid
} }
// TxnStatus represents status of a transaction // TxnStatus represents status of a transaction
...@@ -92,22 +115,6 @@ const ( ...@@ -92,22 +115,6 @@ const (
) )
// XTid is "extended" transaction identifier.
//
// It defines a transaction for oid lookup - either exactly by serial, or by < beforeTid.
type XTid struct {
Tid
TidBefore bool // XXX merge into Tid itself (high bit) ?
}
// Xid is "extended" oid = oid + serial/beforeTid, completely specifying object address query.
type Xid struct {
XTid
Oid
}
// XXX add XidBefore() and XidSerial() as syntax convenience?
// ---- interfaces ---- // ---- interfaces ----
// ErrOidMissing is an error which tells that there is no such oid in the database at all // ErrOidMissing is an error which tells that there is no such oid in the database at all
...@@ -123,7 +130,7 @@ func (e ErrOidMissing) Error() string { ...@@ -123,7 +130,7 @@ func (e ErrOidMissing) Error() string {
} }
// ErrXidMissing is an error which tells that oid exists in the database, // ErrXidMissing is an error which tells that oid exists in the database,
// but there is no its revision satisfying xid.XTid search criteria. // but there is no its revision satisfying xid.At search criteria.
type ErrXidMissing struct { type ErrXidMissing struct {
Xid Xid Xid Xid
} }
...@@ -132,20 +139,17 @@ func (e *ErrXidMissing) Error() string { ...@@ -132,20 +139,17 @@ func (e *ErrXidMissing) Error() string {
return fmt.Sprintf("%v: no matching data record found", e.Xid) return fmt.Sprintf("%v: no matching data record found", e.Xid)
} }
// IStorage is the interface provided by ZODB storages // IStorage is the interface provided when a ZODB storage is opened
type IStorage interface { type IStorage interface {
// URL returns URL of this storage // URL returns URL of this storage
URL() string URL() string
// XXX also +StorageName() with storage driver name?
// Close closes storage // Close closes storage
Close() error Close() error
// LastTid returns the id of the last committed transaction. // LastTid returns the id of the last committed transaction.
// //
// If no transactions have been committed yet, LastTid returns Tid zero value. // If no transactions have been committed yet, LastTid returns 0.
LastTid(ctx context.Context) (Tid, error) LastTid(ctx context.Context) (Tid, error)
// LastOid returns highest object id of objects committed to storage. // LastOid returns highest object id of objects committed to storage.
...@@ -154,37 +158,51 @@ type IStorage interface { ...@@ -154,37 +158,51 @@ type IStorage interface {
// XXX ZODB/py does not define this in IStorage. // XXX ZODB/py does not define this in IStorage.
LastOid(ctx context.Context) (Oid, error) LastOid(ctx context.Context) (Oid, error)
// Load loads data from database. // Load loads object data addressed by xid from database.
// //
// The object to load is addressed by xid. // XXX currently deleted data is returned as buf.Data=nil -- is it ok?
// TODO specify error when data not found -> ErrOidMissing | ErrXidMissing
// //
// NOTE ZODB/py provides 2 entrypoints in IStorage: LoadSerial and // NOTE ZODB/py provides 2 entrypoints in IStorage for loading:
// LoadBefore. Load generalizes them into one (see Xid for details). // LoadSerial and LoadBefore but in ZODB/go we have only Load which is
// a bit different from both:
//
// - Load loads object data for object at database state specified by xid.At
// - LoadBefore loads object data for object at database state previous to xid.At
// it is thus equivalent to Load(..., xid.At-1)
// - LoadSerial loads object data from revision exactly modified
// by transaction with tid = xid.At.
// it is thus equivalent to Load(..., xid.At) with followup
// check that returned serial is exactly xid.At(*)
//
// (*) LoadSerial is used only in a few places in ZODB/py - mostly in
// conflict resolution code where plain Load semantic - without
// checking object was particularly modified at that revision - would
// suffice.
// //
// XXX zodb.loadBefore() returns (data, serial, serial_next) -> add serial_next? // XXX zodb.loadBefore() returns (data, serial, serial_next) -> add serial_next?
// XXX currently deleted data is returned as buf.Data=nil -- is it ok? Load(ctx context.Context, xid Xid) (buf *Buf, serial Tid, err error)
// TODO specify error when data not found -> ErrOidMissing | ErrXidMissing
Load(ctx context.Context, xid Xid) (buf *Buf, serial Tid, err error) // XXX -> DataInfo ?
// Prefetch(ctx, xid Xid) (no error) // Prefetch(ctx, xid Xid) (no error)
// TODO add invalidation channel (notify about changes made to DB not by us) // TODO: write mode
// Store(oid Oid, serial Tid, data []byte, txn ITransaction) error // Store(oid Oid, serial Tid, data []byte, txn ITransaction) error
// XXX Restore ? // KeepCurrent(oid Oid, serial Tid, txn ITransaction)
// CheckCurrentSerialInTransaction(oid Oid, serial Tid, txn ITransaction) // XXX naming
// TpcBegin(txn)
// TpcVote(txn)
// TpcFinish(txn, callback)
// TpcAbort(txn)
// TODO: invalidation channel (notify about changes made to DB not by us)
// TODO:
// tpc_begin(txn)
// tpc_vote(txn)
// tpc_finish(txn, callback) XXX clarify about callback
// tpc_abort(txn)
// TODO: History(ctx, oid, size=1) // TODO: History(ctx, oid, size=1)
// Iterate creates iterator to iterate storage in [tidMin, tidMax] range. // Iterate creates iterator to iterate storage in [tidMin, tidMax] range.
// //
// XXX allow iteration both ways (forward & backward) // TODO allow iteration both ways (forward & backward)
Iterate(tidMin, tidMax Tid) ITxnIterator // XXX ctx , error ? Iterate(tidMin, tidMax Tid) ITxnIterator // XXX ctx , error ?
} }
...@@ -210,7 +228,8 @@ type IDataIterator interface { ...@@ -210,7 +228,8 @@ type IDataIterator interface {
// Valid returns whether tid is in valid transaction identifiers range // Valid returns whether tid is in valid transaction identifiers range
func (tid Tid) Valid() bool { func (tid Tid) Valid() bool {
if 0 <= tid && tid <= TidMax { // NOTE 0 is invalid tid
if 0 < tid && tid <= TidMax {
return true return true
} else { } else {
return false return false
......
...@@ -62,7 +62,7 @@ func Dumpobj(ctx context.Context, w io.Writer, stor zodb.IStorage, xid zodb.Xid, ...@@ -62,7 +62,7 @@ func Dumpobj(ctx context.Context, w io.Writer, stor zodb.IStorage, xid zodb.Xid,
objInfo.Oid = xid.Oid objInfo.Oid = xid.Oid
objInfo.Tid = tid objInfo.Tid = tid
objInfo.Data = buf.Data objInfo.Data = buf.Data
objInfo.DataTid = tid // XXX generally wrong objInfo.DataTidHint = 0 // no copy detection at catobj - just dump raw content
d := dumper{W: w, HashOnly: hashOnly} d := dumper{W: w, HashOnly: hashOnly}
err = d.DumpData(&objInfo) err = d.DumpData(&objInfo)
......
...@@ -94,8 +94,8 @@ func (d *dumper) DumpData(datai *zodb.DataInfo) error { ...@@ -94,8 +94,8 @@ func (d *dumper) DumpData(datai *zodb.DataInfo) error {
case datai.Data == nil: case datai.Data == nil:
buf .S("delete") buf .S("delete")
case datai.Tid != datai.DataTid: case datai.DataTidHint != 0:
buf .S("from ") .V(&datai.DataTid) buf .S("from ") .V(&datai.DataTidHint)
default: default:
// XXX sha1 is hardcoded for now. Dump format allows other hashes. // XXX sha1 is hardcoded for now. Dump format allows other hashes.
......
...@@ -53,15 +53,13 @@ http://docs.pylonsproject.org/projects/zodburi/ ...@@ -53,15 +53,13 @@ http://docs.pylonsproject.org/projects/zodburi/
const helpXid = const helpXid =
`An object address for loading from ZODB should be specified as follows: `An object address for loading from ZODB should be specified as follows:
- "=" or "<" character depending on whether it is exact or "tid before" query
- tid - tid
- ":" - ":"
- oid - oid
for example for example
=0285cbac258bf266:0000000000000001 - oid 1 at exactly 0285cbac258bf266 transaction 0285cbac258bf266:0000000000000001 - oid 1 at first newest transaction changing it with tid <= 0285cbac258bf266
<0285cbac258bf266:0000000000000001 - oid 1 at first newest transaction changing it with tid < 0285cbac258bf266
` `
var helpTopics = prog.HelpRegistry{ var helpTopics = prog.HelpRegistry{
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment