Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neoppod
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
1
Issues
1
List
Boards
Labels
Milestones
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
neoppod
Commits
57a61770
Commit
57a61770
authored
Aug 24, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
ed7ffaa2
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
51 additions
and
18 deletions
+51
-18
go/neo/client/cache.go
go/neo/client/cache.go
+36
-16
go/neo/client/cache_test.go
go/neo/client/cache_test.go
+14
-0
go/zodb/zodb.go
go/zodb/zodb.go
+1
-2
No files found.
go/neo/client/cache.go
View file @
57a61770
...
...
@@ -171,6 +171,36 @@ func isErrNoData(err error) bool {
}
func
(
c
*
Cache
)
Load
(
xid
zodb
.
Xid
)
(
data
[]
byte
,
tid
zodb
.
Tid
,
err
error
)
{
rce
,
rceNew
:=
c
.
lookupRCE
(
xid
)
// rce is already in cache - use it
if
!
rceNew
{
<-
rce
.
ready
c
.
gcMu
.
Lock
()
rce
.
inLRU
.
MoveBefore
(
&
c
.
lru
)
c
.
gcMu
.
Unlock
()
// rce is not in cache - this goroutine becomes responsible for loading it
}
else
{
c
.
loadRCE
(
rce
,
xid
)
}
return
rce
.
data
,
rce
.
serial
,
rce
.
userErr
(
xid
)
}
func
(
c
*
Cache
)
Prefetch
(
xid
zodb
.
Xid
)
{
rce
,
rceNew
:=
c
.
lookupRCE
(
xid
)
// spawn prefetch in the background if rce was not yet loaded
if
rceNew
{
go
c
.
loadRCE
(
rce
,
xid
)
}
}
// lookupRCE returns revCacheEntry corresponding to xid.
// rceNew indicates whether RCE is new and loading on it has not been initiated.
func
(
c
*
Cache
)
lookupRCE
(
xid
zodb
.
Xid
)
(
rce
*
revCacheEntry
,
rceNew
bool
)
{
// oid -> oce (oidCacheEntry) ; create new empty oce if not yet there
// exit with oce locked and cache.before read consistently
c
.
mu
.
RLock
()
...
...
@@ -196,9 +226,6 @@ func (c *Cache) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error) {
}
// oce, before -> rce (revCacheEntry)
var
rce
*
revCacheEntry
var
rceNew
bool
// whether we created rce anew
if
xid
.
TidBefore
{
l
:=
len
(
oce
.
rcev
)
i
:=
sort
.
Search
(
l
,
func
(
i
int
)
bool
{
...
...
@@ -245,17 +272,12 @@ func (c *Cache) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error) {
}
oce
.
Unlock
()
return
rce
,
rceNew
}
// entry was already in cache - use it
if
!
rceNew
{
<-
rce
.
ready
c
.
gcMu
.
Lock
()
rce
.
inLRU
.
MoveBefore
(
&
c
.
lru
)
c
.
gcMu
.
Unlock
()
return
rce
.
data
,
rce
.
serial
,
rce
.
userErr
(
xid
)
}
// entry was not in cache - this goroutine becomes responsible for loading it
// loadRCE performs data loading from database to RCE
func
(
c
*
Cache
)
loadRCE
(
rce
*
revCacheEntry
,
xid
zodb
.
Xid
)
{
oce
:=
rce
.
parent
data
,
serial
,
err
:=
c
.
loader
.
Load
(
xid
)
// normailize data/serial if it was error
...
...
@@ -284,7 +306,7 @@ func (c *Cache) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error) {
// rce was already dropped by merge / evicted
// (XXX recheck about evicted)
oce
.
Unlock
()
return
rce
.
data
,
rce
.
serial
,
rce
.
userErr
(
xid
)
return
}
// if rce & rceNext cover the same range -> drop rce
...
...
@@ -316,8 +338,6 @@ func (c *Cache) Load(xid zodb.Xid) (data []byte, tid zodb.Tid, err error) {
// XXX -> run gc
}
c
.
gcMu
.
Unlock
()
return
rce
.
data
,
rce
.
serial
,
rce
.
userErr
(
xid
)
}
// tryMerge tries to merge rce prev into next
...
...
go/neo/client/cache_test.go
View file @
57a61770
...
...
@@ -241,6 +241,20 @@ func TestCache(t *testing.T) {
ok1
(
rce1_b12
!=
rce1_b10
)
checkRCE
(
rce1_b12
,
12
,
9
,
world
,
nil
)
checkOCE
(
1
,
rce1_b4
,
rce1_b7
,
rce1_b8
,
rce1_b9
,
rce1_b12
)
// simulate case where <14 and <16 were loaded in parallel, both are ready
// but <14 takes oce lock first before <16 ans so <12 is not yet merged
// with <16 -> <12 and <14 should be merged into <16.
// (manually add rce1_b16 so it is not merged with <12)
rce1_b16
:=
oce1
.
newRevEntry
(
len
(
oce1
.
rcev
),
16
)
rce1_b16
.
serial
=
9
rce1_b16
.
data
=
world
close
(
rce1_b16
.
ready
)
// XXX
ok1
(
rce1_b16
.
loaded
())
checkOCE
(
1
,
rce1_b4
,
rce1_b7
,
rce1_b8
,
rce1_b9
,
rce1_b12
,
rce1_b16
)
// XXX launch load(<14) before <16.ready
}
type
Checker
struct
{
...
...
go/zodb/zodb.go
View file @
57a61770
...
...
@@ -155,8 +155,7 @@ type IStorage interface {
// TODO specify error when data not found
Load
(
xid
Xid
)
(
data
[]
byte
,
serial
Tid
,
err
error
)
// XXX -> StorageRecordInformation ?
// -> Prefetch(xid Xid) ...
// PrefetchBefore(oidv []Oid, beforeTid Tid) error (?)
// Prefetch(xid Xid) (no error)
// Store(oid Oid, serial Tid, data []byte, txn ITransaction) error
// XXX Restore ?
...
...
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