Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
W
wendelin.core
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
Kirill Smelkov
wendelin.core
Commits
a83a83b0
Commit
a83a83b0
authored
Oct 08, 2021
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
33abf5f4
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
32 additions
and
23 deletions
+32
-23
wcfs/internal/xbtree/δbtail.go
wcfs/internal/xbtree/δbtail.go
+32
-23
No files found.
wcfs/internal/xbtree/δbtail.go
View file @
a83a83b0
...
...
@@ -85,6 +85,7 @@ import (
"fmt"
"sort"
"strings"
"sync"
"lab.nexedi.com/kirr/go123/xerr"
"lab.nexedi.com/kirr/go123/xsync"
...
...
@@ -142,6 +143,16 @@ type ΔBtail struct {
// includes all changed objects, not only tracked ones.
δZtail
*
zodb
.
ΔTail
// handle to make connections to access database.
// TODO allow client to optionally provide zconnOld/zconnNew on e.g. Update()
db
*
zodb
.
DB
// to open connections to load new/old tree|buckets
// mu protects ΔBtail data _and_ all _ΔTtail data for all roots.
//
// NOTE: even though this lock is global, since _ΔTtail.vδT is updated
// via RCU, working with retrieved vδT snapshot does not need to hold the lock.
mu
sync
.
Mutex
vδBroots
[]
_ΔBroots
// [] (rev↑, roots changed in this rev)
byRoot
map
[
zodb
.
Oid
]
*
_ΔTtail
// {} root -> [] k/v change history; only for keys ∈ tracked subset
...
...
@@ -153,10 +164,6 @@ type ΔBtail struct {
// set of trees for which _ΔTtail.ktrackNew is non-empty
trackNewRoots
setOid
// handle to make connections to access database.
// TODO allow client to optionally provide zconnOld/zconnNew on e.g. Update()
db
*
zodb
.
DB
// to open connections to load new/old tree|buckets
}
// _ΔTtail represent tail of revisional changes to one BTree.
...
...
@@ -430,9 +437,10 @@ func (δBtail *ΔBtail) rebuildAll() (err error) {
//
// vδT is rebuilt if there are such not-yet-handled Track requests.
func
(
δBtail
*
ΔBtail
)
vδTSnapForTrackedKey
(
root
zodb
.
Oid
,
key
Key
)
(
vδT
[]
ΔTree
,
err
error
)
{
// XXX δBtail.lock
δBtail
.
mu
.
Lock
()
δTtail
:=
δBtail
.
byRoot
[
root
]
// must be there
if
δTtail
==
nil
{
δBtail
.
mu
.
Unlock
()
panicf
(
"δBtail: root<%s> not tracked"
,
root
)
}
...
...
@@ -444,31 +452,30 @@ func (δBtail *ΔBtail) vδTSnapForTrackedKey(root zodb.Oid, key Key) (vδT []Δ
if
!
inJobs
{
// key ∉ krebuildJobs -> it should be already in trackSet
vδT
=
δTtail
.
vδT
// XXX δBtail.unlock
δBtail
.
mu
.
Unlock
()
return
vδT
,
nil
}
// rebuild for root[key] is in progress -> wait for corresponding job to complete
// XXX δBtail.unlock
δBtail
.
mu
.
Unlock
()
<-
job
.
ready
if
job
.
err
==
nil
{
// XXX δBtail.lock
δBtail
.
mu
.
Lock
()
vδT
=
δTtail
.
vδT
// XXX δBtail.unlock
δBtail
.
mu
.
Unlock
()
}
return
vδT
,
job
.
err
}
// key ∈ ktrackNew -> this goroutine becomes responsible to start rebuilding vδT for it
//
launch
rebuild job for all keys queued in ktrackNew so far
//
run
rebuild job for all keys queued in ktrackNew so far
err
=
δTtail
.
_runRebuildJob
(
root
,
δBtail
)
if
err
!=
nil
{
return
nil
,
err
}
// XXX δBtail.lock
vδT
=
δTtail
.
vδT
// XXX δBtail.unlock
δBtail
.
mu
.
Unlock
()
return
vδT
,
nil
}
...
...
@@ -478,9 +485,10 @@ func (δBtail *ΔBtail) vδTSnapForTrackedKey(root zodb.Oid, key Key) (vδT []Δ
//
// vδT is rebuilt if there are such not-yet-handled Track requests.
func
(
δBtail
*
ΔBtail
)
vδTSnapForTracked
(
root
zodb
.
Oid
)
(
vδT
[]
ΔTree
,
err
error
)
{
// XXX δBtail.lock
δBtail
.
mu
.
Lock
()
δTtail
:=
δBtail
.
byRoot
[
root
]
// must be there
if
δTtail
==
nil
{
δBtail
.
mu
.
Unlock
()
panicf
(
"δBtail: root<%s> not tracked"
,
root
)
}
...
...
@@ -504,7 +512,8 @@ func (δBtail *ΔBtail) vδTSnapForTracked(root zodb.Oid) (vδT []ΔTree, err er
errJob
=
δTtail
.
_runRebuildJob
(
root
,
δBtail
)
}
// XXX δBtail.unlock
// wait for previous jobs to complete as well
δBtail
.
mu
.
Unlock
()
errWait
:=
wg
.
Wait
()
err
=
xerr
.
First
(
errJob
,
errWait
)
...
...
@@ -512,15 +521,18 @@ func (δBtail *ΔBtail) vδTSnapForTracked(root zodb.Oid) (vδT []ΔTree, err er
return
nil
,
err
}
// XXX δBtail.lock
// now it is ok to take the snapshot
δBtail
.
mu
.
Lock
()
vδT
=
δTtail
.
vδT
// XXX δBtail.unlock
δBtail
.
mu
.
Unlock
()
return
vδT
,
nil
}
// _runRebuildJob runs rebuild job for current .ktrackNew/.trackNew
// must be called with δBtail locked.
//
// must be called with δBtail.mu locked.
// returns with δBtail.mu locked.
func
(
δTtail
*
_ΔTtail
)
_runRebuildJob
(
root
zodb
.
Oid
,
δBtail
*
ΔBtail
)
(
err
error
)
{
// XXX errctx
job
:=
&
_RebuildJob
{
ready
:
make
(
chan
struct
{})}
...
...
@@ -536,11 +548,11 @@ func (δTtail *_ΔTtail) _runRebuildJob(root zodb.Oid, δBtail *ΔBtail) (err er
δTtail
.
krebuildJobs
.
SetRange
(
r
,
job
)
}
delete
(
δBtail
.
trackNewRoots
,
root
)
// XXX δBtail.unlock
// build δ(vδT) without the lock
δBtail
.
mu
.
Unlock
()
vδTnew
,
δtrackSet
,
err
:=
vδTBuild
(
root
,
trackNew
,
δBtail
.
δZtail
,
δBtail
.
db
)
// XXX δBtail.lock
δBtail
.
mu
.
Lock
()
// krebuildJobs -= rebuildKeys
for
_
,
r
:=
range
rebuildKeys
.
AllRanges
()
{
...
...
@@ -560,10 +572,7 @@ func (δTtail *_ΔTtail) _runRebuildJob(root zodb.Oid, δBtail *ΔBtail) (err er
// we are done
job
.
err
=
err
close
(
job
.
ready
)
// XXX δBtail.unlock
return
job
.
err
}
// rebuild1 rebuilds ΔBtail for single root.
...
...
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