Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
Kirill Smelkov
neo
Commits
51026eb3
Commit
51026eb3
authored
Aug 25, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
42a61864
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
79 additions
and
57 deletions
+79
-57
go/neo/client/cache.go
go/neo/client/cache.go
+72
-53
go/neo/client/cache_test.go
go/neo/client/cache_test.go
+2
-2
go/zodb/zodb.go
go/zodb/zodb.go
+5
-2
No files found.
go/neo/client/cache.go
View file @
51026eb3
...
...
@@ -177,7 +177,7 @@ func isErrNoData(err error) bool {
// Load loads data from database via cache.
//
// If data is already in cache cached content is returned.
func
(
c
*
Cache
)
Load
(
xid
zodb
.
Xid
)
(
data
[]
byte
,
tid
zodb
.
Tid
,
err
error
)
{
func
(
c
*
Cache
)
Load
(
xid
zodb
.
Xid
)
(
data
[]
byte
,
serial
zodb
.
Tid
,
err
error
)
{
rce
,
rceNew
:=
c
.
lookupRCE
(
xid
)
// rce is already in cache - use it
...
...
@@ -191,10 +191,21 @@ func (c *Cache) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error) {
}
else
{
// XXX use connection poll
// XXX or it should be cared by loader?
c
.
loadRCE
(
rce
,
xid
)
c
.
loadRCE
(
rce
,
xid
.
Oid
)
}
return
rce
.
data
,
rce
.
serial
,
rce
.
userErr
(
xid
)
if
rce
.
err
!=
nil
{
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
}
}
}
return
rce
.
data
,
rce
.
serial
,
nil
}
// Prefetch arranges for data to be eventually present in cache.
...
...
@@ -207,19 +218,30 @@ func (c *Cache) Prefetch(xid zodb.Xid) {
// XXX!rceNew -> adjust LRU?
// spawn
prefetch
in the background if rce was not yet loaded
// spawn
loading
in the background if rce was not yet loaded
if
rceNew
{
// XXX use connection poll
go
c
.
loadRCE
(
rce
,
xid
)
go
c
.
loadRCE
(
rce
,
xid
.
Oid
)
}
}
// lookupRCE returns revCacheEntry corresponding to xid.
//
// If xid indicates loadSerial query (xid.TidBefore=false) the 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
// initiated yet. rce should be loaded with loadRCE.
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
// exit with oce locked and cache.before read consistently
c
.
mu
.
RLock
()
...
...
@@ -245,20 +267,19 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
}
// oce, before -> rce (revCacheEntry)
if
xid
.
TidBefore
{
l
:=
len
(
oce
.
rcev
)
i
:=
sort
.
Search
(
l
,
func
(
i
int
)
bool
{
before
:=
oce
.
rcev
[
i
]
.
before
if
before
==
zodb
.
TidMax
{
before
=
cacheBefore
before_i
:=
oce
.
rcev
[
i
]
.
before
if
before_i
==
zodb
.
TidMax
{
before_i
=
cacheBefore
}
return
xid
.
Tid
<=
before
return
before
<=
before_i
})
switch
{
// not found - tid
> max(rcev.before) - insert new max entry
// not found - before
> max(rcev.before) - insert new max entry
case
i
==
l
:
rce
=
oce
.
newRevEntry
(
i
,
xid
.
Tid
)
rce
=
oce
.
newRevEntry
(
i
,
before
)
if
rce
.
before
==
cacheBefore
{
// FIXME better do this when the entry becomes loaded ?
// XXX vs concurrent invalidations?
...
...
@@ -267,32 +288,27 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
rceNew
=
true
// found:
// tid
<= rcev[i].before
// tid
> rcev[i-1].before
// before
<= rcev[i].before
// before
> rcev[i-1].before
// exact match - we already have entry for this before
case
xid
.
Tid
==
oce
.
rcev
[
i
]
.
before
:
case
before
==
oce
.
rcev
[
i
]
.
before
:
rce
=
oce
.
rcev
[
i
]
// non-exact match:
// - same entry if q(before) ∈ (serial, before]
// - we can also reuse this entry if q(before) < before and err="nodata"
case
oce
.
rcev
[
i
]
.
loaded
()
&&
(
(
oce
.
rcev
[
i
]
.
err
==
nil
&&
oce
.
rcev
[
i
]
.
serial
<
xid
.
Tid
)
||
(
isErrNoData
(
oce
.
rcev
[
i
]
.
err
)
&&
xid
.
Tid
<
oce
.
rcev
[
i
]
.
before
))
:
(
oce
.
rcev
[
i
]
.
err
==
nil
&&
oce
.
rcev
[
i
]
.
serial
<
before
)
||
(
isErrNoData
(
oce
.
rcev
[
i
]
.
err
)
&&
before
<
oce
.
rcev
[
i
]
.
before
))
:
rce
=
oce
.
rcev
[
i
]
// otherwise - insert new entry
default
:
rce
=
oce
.
newRevEntry
(
i
,
xid
.
Tid
)
rce
=
oce
.
newRevEntry
(
i
,
before
)
rceNew
=
true
}
// XXX serial -> revCacheEntry
}
else
{
// TODO
}
oce
.
Unlock
()
return
rce
,
rceNew
}
...
...
@@ -300,9 +316,13 @@ func (c *Cache) lookupRCE(xid zodb.Xid) (rce *revCacheEntry, rceNew bool) {
// loadRCE performs data loading from database into rce.
//
// rce must be new just created by lookupRCE() with returned rceNew=true.
func
(
c
*
Cache
)
loadRCE
(
rce
*
revCacheEntry
,
xid
zodb
.
Xid
)
{
// loading completion is signalled by closing rce.ready.
func
(
c
*
Cache
)
loadRCE
(
rce
*
revCacheEntry
,
oid
zodb
.
Oid
)
{
oce
:=
rce
.
parent
data
,
serial
,
err
:=
c
.
loader
.
Load
(
xid
)
data
,
serial
,
err
:=
c
.
loader
.
Load
(
zodb
.
Xid
{
Oid
:
oid
,
XTid
:
zodb
.
XTid
{
Tid
:
rce
.
before
,
TidBefore
:
true
},
})
// normalize data/serial if it was error
if
err
!=
nil
{
...
...
@@ -314,8 +334,7 @@ func (c *Cache) loadRCE(rce *revCacheEntry, xid zodb.Xid) {
rce
.
err
=
err
// verify db gives serial < before
if
rce
.
serial
>=
rce
.
before
{
// XXX loadSerial?
rce
.
errDB
(
xid
.
Oid
,
"load(<%v) -> %v"
,
rce
.
before
,
serial
)
rce
.
errDB
(
oid
,
"load(<%v) -> %v"
,
rce
.
before
,
serial
)
}
close
(
rce
.
ready
)
...
...
@@ -336,7 +355,7 @@ func (c *Cache) loadRCE(rce *revCacheEntry, xid zodb.Xid) {
// if rce & rceNext cover the same range -> drop rce
if
i
+
1
<
len
(
oce
.
rcev
)
{
rceNext
:=
oce
.
rcev
[
i
+
1
]
if
rceNext
.
loaded
()
&&
tryMerge
(
rce
,
rceNext
,
rce
,
xid
.
O
id
)
{
if
rceNext
.
loaded
()
&&
tryMerge
(
rce
,
rceNext
,
rce
,
o
id
)
{
// not δsize -= len(rce.data)
// tryMerge can change rce.data if consistency is broken
δsize
=
0
...
...
@@ -347,7 +366,7 @@ func (c *Cache) loadRCE(rce *revCacheEntry, xid zodb.Xid) {
// if rcePrev & rce cover the same range -> drop rcePrev
if
i
>
0
{
rcePrev
:=
oce
.
rcev
[
i
-
1
]
if
rcePrev
.
loaded
()
&&
tryMerge
(
rcePrev
,
rce
,
rce
,
xid
.
O
id
)
{
if
rcePrev
.
loaded
()
&&
tryMerge
(
rcePrev
,
rce
,
rce
,
o
id
)
{
δsize
-=
len
(
rcePrev
.
data
)
}
}
...
...
go/neo/client/cache_test.go
View file @
51026eb3
...
...
@@ -269,7 +269,7 @@ func TestCache(t *testing.T) {
checkOCE
(
1
,
rce1_b4
,
rce1_b7
,
rce1_b8
,
rce1_b10
,
rce1_b12
,
rce1_b14
,
rce1_b16
)
// (<14 also becomes ready and takes oce lock first, merging <12 and <14 into <16)
c
.
loadRCE
(
rce1_b14
,
xidlt
(
1
,
14
)
)
c
.
loadRCE
(
rce1_b14
,
1
)
checkRCE
(
rce1_b14
,
14
,
10
,
world
,
nil
)
checkRCE
(
rce1_b16
,
16
,
10
,
world
,
nil
)
checkRCE
(
rce1_b12
,
12
,
10
,
world
,
nil
)
...
...
@@ -304,7 +304,7 @@ func TestCache(t *testing.T) {
// <9 must be separate from <8 and <10 because it is IO error there
rce1_b9
,
new9
:=
c
.
lookupRCE
(
xidlt
(
1
,
9
))
ok1
(
new9
)
c
.
loadRCE
(
rce1_b9
,
xidlt
(
1
,
9
)
)
c
.
loadRCE
(
rce1_b9
,
1
)
checkRCE
(
rce1_b9
,
9
,
0
,
nil
,
ioerr
)
checkOCE
(
1
,
rce1_b4
,
rce1_b7
,
rce1_b8
,
rce1_b9
,
rce1_b10
,
rce1_b16
)
...
...
go/zodb/zodb.go
View file @
51026eb3
...
...
@@ -150,9 +150,12 @@ type IStorage interface {
LastOid
()
(
Oid
,
error
)
// LoadSerial and LoadBefore generalized into 1 Load (see Xid for details)
// TODO data []byte -> something allocated from slab ?
//
// XXX zodb.loadBefore() returns (data, serial, serial_next) -> add serial_next?
//
// XXX currently deleted data is returned as data=nil -- is it ok?
// TODO specify error when data not found
// TODO specify error when data not found -> ErrOidMissing | ErrXidMissing
// TODO data []byte -> something allocated from slab ?
Load
(
xid
Xid
)
(
data
[]
byte
,
serial
Tid
,
err
error
)
// XXX -> StorageRecordInformation ?
// Prefetch(xid Xid) (no error)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment