Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
B
b
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
b
Commits
171c18f8
Commit
171c18f8
authored
Apr 11, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
6926aba3
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
147 additions
and
34 deletions
+147
-34
all_test.go
all_test.go
+5
-3
btree.go
btree.go
+142
-31
No files found.
all_test.go
View file @
171c18f8
...
@@ -223,11 +223,13 @@ func TestSetGet0(t *testing.T) {
...
@@ -223,11 +223,13 @@ func TestSetGet0(t *testing.T) {
}
}
func
TestSetGet1
(
t
*
testing
.
T
)
{
func
TestSetGet1
(
t
*
testing
.
T
)
{
const
N
=
40000
//
const N = 40000
//const N = 21
//const N = 21
//const N = 41
//const N = 41
//const N = 320
//const N = 320
for
_
,
x
:=
range
[]
int
{
0
,
-
1
,
0x555555
,
0xaaaaaa
,
0x333333
,
0xcccccc
,
0x314159
}
{
const
N
=
730
//for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x333333, 0xcccccc, 0x314159} {
for
_
,
x
:=
range
[]
int
{
0x333333
}
{
//for _, x := range []int{0x314159} {
//for _, x := range []int{0x314159} {
//for _, x := range []int{0} {
//for _, x := range []int{0} {
r
:=
TreeNew
(
cmp
)
r
:=
TreeNew
(
cmp
)
...
@@ -247,7 +249,7 @@ func TestSetGet1(t *testing.T) {
...
@@ -247,7 +249,7 @@ func TestSetGet1(t *testing.T) {
for
i
,
k
:=
range
a
{
for
i
,
k
:=
range
a
{
v
,
ok
:=
r
.
Get
(
k
)
v
,
ok
:=
r
.
Get
(
k
)
if
!
ok
{
if
!
ok
{
t
.
Fatal
(
i
,
k
,
v
,
ok
)
// XXX bug here
t
.
Fatal
(
i
,
k
,
v
,
ok
)
}
}
if
g
,
e
:=
v
.
(
int
),
k
^
x
;
g
!=
e
{
if
g
,
e
:=
v
.
(
int
),
k
^
x
;
g
!=
e
{
...
...
btree.go
View file @
171c18f8
...
@@ -107,7 +107,7 @@ type (
...
@@ -107,7 +107,7 @@ type (
hitPi
int
hitPi
int
hitKmin
xkey
// data page key range is [hitKmin, hitKmax)
hitKmin
xkey
// data page key range is [hitKmin, hitKmax)
hitKmax
xkey
hitKmax
xkey
hitPKmax
xkey
// Kmax for
hit range one level up XXX text
hitPKmax
xkey
// Kmax for
whole hitP
}
}
xe
struct
{
// x element
xe
struct
{
// x element
...
@@ -655,10 +655,109 @@ func (t *Tree) SeekLast() (e *Enumerator, err error) {
...
@@ -655,10 +655,109 @@ func (t *Tree) SeekLast() (e *Enumerator, err error) {
// Set sets the value associated with k.
// Set sets the value associated with k.
func
(
t
*
Tree
)
Set
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
func
(
t
*
Tree
)
Set
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
//dbg("--- PRE Set(%v, %v)\t(%v @%d, [%v, %v))\n%s", k, v, t.hitD, t.hitDi, t.hitKmin, t.hitKmax, t.dump())
dbg
(
"--- PRE Set(%v, %v)
\t
(%v @%d, [%v, %v) PKmax: %v)
\n
%s"
,
k
,
v
,
t
.
hitD
,
t
.
hitDi
,
t
.
hitKmin
,
t
.
hitKmax
,
t
.
hitPKmax
,
t
.
dump
())
//defer func() {
defer
func
()
{
// dbg("--- POST\n%s\n====\n", t.dump())
badHappenned
:=
false
//}()
bad
:=
func
(
s
string
,
va
...
interface
{})
{
dbg
(
s
,
va
...
)
badHappenned
=
true
}
println
()
if
t
.
hitDi
>=
0
{
if
t
.
hitD
.
d
[
t
.
hitDi
]
.
k
!=
k
{
bad
(
"hitD invalid: %v @%v"
,
t
.
hitD
,
t
.
hitDi
)
}
}
if
t
.
hitPi
>=
0
{
if
t
.
hitP
.
x
[
t
.
hitPi
]
.
ch
!=
t
.
hitD
{
bad
(
"hitP invalid: %v @%v"
,
t
.
hitP
,
t
.
hitPi
)
}
}
// rescan from root and check Kmin/Kmax and rest
q
:=
t
.
r
var
p
*
x
pi
:=
-
1
var
dd
*
d
var
i
int
var
ok
bool
var
hitKmin
,
hitKmax
xkey
var
hitPKmax
xkey
//dbg("k: %v", k)
loop
:
for
{
//dbg("p: %p: @%d %v", p, pi, p)
//dbg("q: %p: %v", q, q)
i
,
ok
=
t
.
find
(
q
,
k
)
//dbg("\t-> %v, %v", i, ok)
switch
x
:=
q
.
(
type
)
{
case
*
x
:
hitPKmax
=
hitKmax
// XXX recheck
p
=
x
pi
=
i
if
ok
{
pi
++
}
//dbg("\tpi -> %v", pi)
q
=
p
.
x
[
pi
]
.
ch
if
pi
<
p
.
c
{
//dbg("", p.x, pi)
hitKmax
.
set
(
p
.
x
[
pi
]
.
k
)
// XXX not sure or x[pi+1] ?
if
hitPKmax
.
kset
&&
t
.
cmp
(
hitKmax
.
k
,
hitPKmax
.
k
)
>=
0
{
bad
(
"hitKmax not ↓: %v -> %v"
,
hitPKmax
.
k
,
hitKmax
.
k
)
}
}
case
*
d
:
if
!
ok
{
bad
(
"key %v not found after set"
,
k
)
}
dd
=
x
break
loop
}
}
if
dd
!=
t
.
hitD
||
i
!=
t
.
hitDi
{
bad
(
"hitD mismatch: %v @%d ; want %v @%d"
,
t
.
hitD
,
t
.
hitDi
,
dd
,
i
)
}
if
p
!=
t
.
hitP
||
pi
!=
t
.
hitPi
{
bad
(
"hitP mismatch: %v @%d ; want %v @%d"
,
t
.
hitP
,
t
.
hitPi
,
p
,
pi
)
}
if
hitKmax
!=
t
.
hitKmax
{
bad
(
"hitKmax mismatch: %v ; want %v"
,
t
.
hitKmax
,
hitKmax
)
}
if
hitPKmax
!=
t
.
hitPKmax
{
bad
(
"hitPKmax mismatch: %v ; want %v"
,
t
.
hitPKmax
,
hitPKmax
)
}
v2
,
ok
:=
t
.
Get
(
k
)
if
!
ok
||
v2
!=
v
{
bad
(
"get(%v) -> %v, %v; want %v, %v"
,
k
,
v2
,
ok
,
v
,
true
)
}
if
badHappenned
{
panic
(
0
)
}
}()
defer
func
()
{
dbg
(
"--- POST
\n
%s
\n
====
\n
"
,
t
.
dump
())
}()
// check if we can do the update nearby previous change
// check if we can do the update nearby previous change
i
,
ok
:=
t
.
hitFind
(
k
)
i
,
ok
:=
t
.
hitFind
(
k
)
...
@@ -668,23 +767,24 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
...
@@ -668,23 +767,24 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
switch
{
switch
{
case
ok
:
case
ok
:
//
dbg("ok'")
dbg
(
"ok'"
)
dd
.
d
[
i
]
.
v
=
v
dd
.
d
[
i
]
.
v
=
v
t
.
hitDi
=
i
t
.
hitDi
=
i
return
return
case
dd
.
c
<
2
*
kd
:
case
dd
.
c
<
2
*
kd
:
//
dbg("insert'")
dbg
(
"insert'"
)
t
.
insert
(
dd
,
i
,
k
,
v
)
t
.
insert
(
dd
,
i
,
k
,
v
)
return
return
// here: need to overflow but we have to check: if overflowing
// here: need to overflow but we have to check: if overflowing
// would cause upper leve
r
overflow -> we cannot overflow here -
// would cause upper leve
l
overflow -> we cannot overflow here -
// - need to do the usual scan from root to split index pages.
// - need to do the usual scan from root to split index pages.
default
:
default
:
//break
p
,
pi
:=
t
.
hitP
,
t
.
hitPi
p
,
pi
:=
t
.
hitP
,
t
.
hitPi
if
p
==
nil
||
p
.
c
<=
2
*
kx
{
if
p
==
nil
||
p
.
c
<=
2
*
kx
{
// XXX < vs <=
//
dbg("overflow'")
dbg
(
"overflow'"
)
t
.
overflow
(
p
,
dd
,
pi
,
i
,
k
,
v
)
t
.
overflow
(
p
,
dd
,
pi
,
i
,
k
,
v
)
return
return
}
}
...
@@ -696,39 +796,48 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
...
@@ -696,39 +796,48 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
var
p
*
x
var
p
*
x
q
:=
t
.
r
q
:=
t
.
r
if
q
==
nil
{
if
q
==
nil
{
//
dbg("empty")
dbg
(
"empty"
)
z
:=
t
.
insert
(
btDPool
.
Get
()
.
(
*
d
),
0
,
k
,
v
)
// XXX update hit
z
:=
t
.
insert
(
btDPool
.
Get
()
.
(
*
d
),
0
,
k
,
v
)
// XXX update hit
t
.
r
,
t
.
first
,
t
.
last
=
z
,
z
,
z
t
.
r
,
t
.
first
,
t
.
last
=
z
,
z
,
z
return
return
}
}
var
hitKmin
,
hitKmax
xkey
// initially [-∞, +∞)
var
hitKmin
,
hitKmax
xkey
// initially [-∞, +∞)
var
hitPKmax
xkey
//
hitKmax state @ one level up
var
hitPKmax
xkey
//
Kmax for whole hitP
for
{
for
{
hitPKmax
=
hitKmax
i
,
ok
=
t
.
find
(
q
,
k
)
i
,
ok
=
t
.
find
(
q
,
k
)
switch
x
:=
q
.
(
type
)
{
switch
x
:=
q
.
(
type
)
{
case
*
x
:
case
*
x
:
//hitPKmax = hitKmax
if
x
.
c
>
2
*
kx
{
if
x
.
c
>
2
*
kx
{
x
,
i
=
t
.
splitX
(
p
,
x
,
pi
,
i
)
//x, i = t.splitX(p, x, pi, i)
dbg
(
"splitX"
)
x
,
i
,
p
,
pi
=
t
.
splitX
(
p
,
x
,
pi
,
i
)
}
}
if
ok
{
pi
=
i
+
1
if
pi
>=
0
&&
pi
<
p
.
c
{
}
else
{
hitPKmax
.
set
(
p
.
x
[
pi
]
.
k
)
pi
=
i
dbg
(
"hitPKmax: %v"
,
hitPKmax
)
}
}
p
=
x
p
=
x
pi
=
i
if
ok
{
pi
++
}
q
=
p
.
x
[
pi
]
.
ch
q
=
p
.
x
[
pi
]
.
ch
if
pi
>
0
{
// XXX also check < p.c ?
if
pi
>
0
{
// XXX also check < p.c ?
hitKmin
.
set
(
p
.
x
[
pi
-
1
]
.
k
)
hitKmin
.
set
(
p
.
x
[
pi
-
1
]
.
k
)
dbg
(
"hitKmin: %v"
,
hitKmin
)
}
}
if
pi
<
p
.
c
{
if
pi
<
p
.
c
{
hitKmax
.
set
(
p
.
x
[
pi
]
.
k
)
hitKmax
.
set
(
p
.
x
[
pi
]
.
k
)
dbg
(
"hitKmax: %v"
,
hitKmax
)
}
}
if
!
((
hitKmin
.
k
!=
nil
)
==
hitKmin
.
kset
)
{
if
!
((
hitKmin
.
k
!=
nil
)
==
hitKmin
.
kset
)
{
...
@@ -750,16 +859,16 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
...
@@ -750,16 +859,16 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
switch
{
switch
{
case
ok
:
case
ok
:
//
dbg("ok")
dbg
(
"ok"
)
x
.
d
[
i
]
.
v
=
v
x
.
d
[
i
]
.
v
=
v
t
.
hitD
,
t
.
hitDi
=
x
,
i
t
.
hitD
,
t
.
hitDi
=
x
,
i
case
x
.
c
<
2
*
kd
:
case
x
.
c
<
2
*
kd
:
//
dbg("insert")
dbg
(
"insert"
)
t
.
insert
(
x
,
i
,
k
,
v
)
t
.
insert
(
x
,
i
,
k
,
v
)
default
:
default
:
//
dbg("overflow")
dbg
(
"overflow"
)
// NOTE overflow will correct hit Kmin, Kmax, P and Pi as needed
// NOTE overflow will correct hit Kmin, Kmax, P and Pi as needed
t
.
overflow
(
p
,
x
,
pi
,
i
,
k
,
v
)
t
.
overflow
(
p
,
x
,
pi
,
i
,
k
,
v
)
}
}
...
@@ -806,7 +915,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
...
@@ -806,7 +915,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
switch
x
:=
q
.
(
type
)
{
switch
x
:=
q
.
(
type
)
{
case
*
x
:
case
*
x
:
if
x
.
c
>
2
*
kx
{
if
x
.
c
>
2
*
kx
{
x
,
i
=
t
.
splitX
(
p
,
x
,
pi
,
i
)
panic
(
"TODO"
)
x
,
i
,
_
,
_
=
t
.
splitX
(
p
,
x
,
pi
,
i
)
}
}
pi
=
i
+
1
pi
=
i
+
1
p
=
x
p
=
x
...
@@ -829,7 +939,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
...
@@ -829,7 +939,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
switch
x
:=
q
.
(
type
)
{
switch
x
:=
q
.
(
type
)
{
case
*
x
:
case
*
x
:
if
x
.
c
>
2
*
kx
{
if
x
.
c
>
2
*
kx
{
x
,
i
=
t
.
splitX
(
p
,
x
,
pi
,
i
)
panic
(
"TODO"
)
//x, i = t.splitX(p, x, pi, i)
}
}
pi
=
i
pi
=
i
p
=
x
p
=
x
...
@@ -913,7 +1024,7 @@ func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} /
...
@@ -913,7 +1024,7 @@ func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} /
}
}
}
}
func
(
t
*
Tree
)
splitX
(
p
*
x
,
q
*
x
,
pi
int
,
i
int
)
(
*
x
,
int
)
{
func
(
t
*
Tree
)
splitX
(
p
*
x
,
q
*
x
,
pi
int
,
i
int
)
(
*
x
,
int
,
*
x
,
int
)
{
t
.
ver
++
t
.
ver
++
r
:=
btXPool
.
Get
()
.
(
*
x
)
r
:=
btXPool
.
Get
()
.
(
*
x
)
copy
(
r
.
x
[
:
],
q
.
x
[
kx
+
1
:
])
copy
(
r
.
x
[
:
],
q
.
x
[
kx
+
1
:
])
...
@@ -928,11 +1039,11 @@ func (t *Tree) splitX(p *x, q *x, pi int, i int) (*x, int) {
...
@@ -928,11 +1039,11 @@ func (t *Tree) splitX(p *x, q *x, pi int, i int) (*x, int) {
switch
{
switch
{
case
i
<
kx
:
case
i
<
kx
:
return
q
,
i
return
q
,
i
,
p
,
pi
case
i
==
kx
:
case
i
==
kx
:
return
p
,
pi
return
p
,
pi
,
p
,
-
1
default
:
// i > kx
default
:
// i > kx
return
r
,
i
-
kx
-
1
return
r
,
i
-
kx
-
1
,
p
,
pi
+
1
}
}
}
}
...
@@ -945,11 +1056,11 @@ func (t *Tree) splitX(p *x, q *x, pi int, i int) (*x, int) {
...
@@ -945,11 +1056,11 @@ func (t *Tree) splitX(p *x, q *x, pi int, i int) (*x, int) {
switch
{
switch
{
case
i
<
kx
:
case
i
<
kx
:
return
q
,
i
return
q
,
i
,
nr
,
0
case
i
==
kx
:
case
i
==
kx
:
return
nr
,
0
return
nr
,
0
,
nr
,
-
1
default
:
// i > kx
default
:
// i > kx
return
r
,
i
-
kx
-
1
return
r
,
i
-
kx
-
1
,
nr
,
1
}
}
}
}
...
...
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