Commit 5b5dd3c2 authored by Kirill Smelkov's avatar Kirill Smelkov

.

parent d4889666
...@@ -44,7 +44,7 @@ func TODO(...interface{}) string { //TODOOK ...@@ -44,7 +44,7 @@ func TODO(...interface{}) string { //TODOOK
func use(...interface{}) {} func use(...interface{}) {}
func init() { func init() {
use(caller, dbg, TODO, isNil, (*Tree).dump, (*Tree).checkHit) //TODOOK use(caller, dbg, TODO, isNil, (*Tree).dump, (*Tree).checkHit, opPut) //TODOOK
} }
// ============================================================================ // ============================================================================
...@@ -143,10 +143,18 @@ func (t *Tree) dump() string { ...@@ -143,10 +143,18 @@ func (t *Tree) dump() string {
type treeOp int type treeOp int
const ( const (
opSet treeOp = iota opGet treeOp = iota
opSet
opDel opDel
) )
func opPut(written bool) treeOp {
if written {
return opSet
}
return opGet
}
// checkHit rescans t from root and checks that hit D, P, Kmin/Kmax and rest all match what they should // checkHit rescans t from root and checks that hit D, P, Kmin/Kmax and rest all match what they should
// it can be used after Set/Put/Delete to verify consistency // it can be used after Set/Put/Delete to verify consistency
func (t *Tree) checkHit(k interface{} /*K*/, op treeOp) { func (t *Tree) checkHit(k interface{} /*K*/, op treeOp) {
...@@ -207,6 +215,9 @@ loop: ...@@ -207,6 +215,9 @@ loop:
case *d: case *d:
switch op { switch op {
case opGet:
panic("TODO")
case opSet: case opSet:
if !ok { if !ok {
bad("key %v not found after set", k) bad("key %v not found after set", k)
......
...@@ -824,6 +824,7 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) { ...@@ -824,6 +824,7 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
// //
// modulo the differing return values. // modulo the differing return values.
func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists bool) (newV interface{} /*V*/, write bool)) (oldV interface{} /*V*/, written bool) { func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists bool) (newV interface{} /*V*/, write bool)) (oldV interface{} /*V*/, written bool) {
defer func () { t.checkHit(k, opPut(written)) }()
pi := -1 pi := -1
var p *x var p *x
q := t.r q := t.r
...@@ -840,8 +841,118 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -840,8 +841,118 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
return return
} }
// TODO handle t.hitD // check if we can do the update nearby previous change
i, ok := t.hitFind(k)
if i >= 0 {
dd := t.hitD
switch {
case ok:
newV, written = upd(dd.d[i].v, true)
if written {
dd.d[i].v = newV
}
t.hitDi = i
return
case dd.c < 2*kd:
newV, written = upd(newV, false)
if written {
t.insert(dd, i, k, newV)
} else {
t.hitDi = i
}
return
// here: need to overflow but we have to check: if overflowing would
// cause upper level overflow (splitX) -> we cannot overflow here -
// - need to do the usual scan from root to split index pages.
default:
p, pi := t.hitP, t.hitPi
if p != nil && p.c > 2*kx {
break
}
newV, written = upd(newV, false)
if written {
// NOTE overflow corrects hit Kmin, Kmax and Pi as needed
t.overflow(p, dd, pi, i, k, newV)
} else {
t.hitDi = i
}
return
}
}
// data page not quickly found - search and descent from root
t.hitKmin, t.hitKmax = xkey{}, xkey{} // initially [-∞, +∞)
t.hitPKmin, t.hitPKmax = xkey{}, xkey{}
for {
i, ok := t.find(q, k)
switch x := q.(type) {
case *x:
if ok {
i++
}
if x.c > 2*kx {
// NOTE splitX corrects hit Kmin and Kmax as needed
x, i = t.splitX(p, x, pi, i)
}
t.hitPKmin = t.hitKmin
t.hitPKmax = t.hitKmax
p = x
pi = i
q = p.x[pi].ch
if pi > 0 { // k=-∞ @-1
t.hitKmin.set(p.x[pi-1].k)
}
if pi < p.c { // k=+∞ @p.c
t.hitKmax.set(p.x[pi].k)
}
case *d:
// data page found - perform the update
t.hitP = p
t.hitPi = pi
switch {
case ok:
newV, written = upd(x.d[i].v, true)
if written {
x.d[i].v = newV
}
t.hitD, t.hitDi = x, i
default:
newV, written = upd(newV, false)
if !written {
t.hitD, t.hitDi = x, i
break
}
switch {
case x.c < 2*kd:
t.insert(x, i, k, newV)
default:
// NOTE overflow corrects hit Kmin, Kmax and Pi as needed
t.overflow(p, x, pi, i, k, newV)
}
}
return
}
}
/*
for { for {
i, ok := t.find(q, k) i, ok := t.find(q, k)
if ok { if ok {
...@@ -849,7 +960,6 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -849,7 +960,6 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
case *x: case *x:
i++ i++
if x.c > 2*kx { if x.c > 2*kx {
panic("TODO")
x, i = t.splitX(p, x, pi, i) x, i = t.splitX(p, x, pi, i)
} }
pi = i pi = i
...@@ -863,8 +973,6 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -863,8 +973,6 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
return return
} }
// XXX update hit
x.d[i].v = newV x.d[i].v = newV
} }
return return
...@@ -873,8 +981,7 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -873,8 +981,7 @@ 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 {
panic("TODO") x, i = t.splitX(p, x, pi, i)
//x, i = t.splitX(p, x, pi, i)
} }
pi = i pi = i
p = x p = x
...@@ -885,18 +992,16 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -885,18 +992,16 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
return return
} }
// XXX update hit
switch { switch {
case x.c < 2*kd: case x.c < 2*kd:
t.insert(x, i, k, newV) t.insert(x, i, k, newV)
default: default:
//t.overflow(p, x, pi, i, k, newV) t.overflow(p, x, pi, i, k, newV)
panic("TODO")
} }
return return
} }
} }
*/
} }
func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} /*V*/) { func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} /*V*/) {
......
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