Fix freelist allocate().

pull/34/head
Ben Johnson 2014-05-19 14:11:32 -06:00
parent 782ead0dbf
commit 12b36fe70c
5 changed files with 16 additions and 12 deletions

View File

@ -319,10 +319,10 @@ func TestDB_Consistency(t *testing.T) {
assert.Equal(t, "free", p.Type) assert.Equal(t, "free", p.Type)
} }
if p, _ := tx.Page(4); assert.NotNil(t, p) { if p, _ := tx.Page(4); assert.NotNil(t, p) {
assert.Equal(t, "freelist", p.Type) assert.Equal(t, "leaf", p.Type) // root leaf
} }
if p, _ := tx.Page(5); assert.NotNil(t, p) { if p, _ := tx.Page(5); assert.NotNil(t, p) {
assert.Equal(t, "leaf", p.Type) // root leaf assert.Equal(t, "freelist", p.Type)
} }
p, _ := tx.Page(6) p, _ := tx.Page(6)
assert.Nil(t, p) assert.Nil(t, p)

View File

@ -55,7 +55,7 @@ func (f *freelist) allocate(n int) pgid {
if (i + 1) == n { if (i + 1) == n {
f.ids = f.ids[i+1:] f.ids = f.ids[i+1:]
} else { } else {
copy(f.ids[i-1:], f.ids[i+n-1:]) copy(f.ids[i-n+1:], f.ids[i+1:])
f.ids = f.ids[:len(f.ids)-n] f.ids = f.ids[:len(f.ids)-n]
} }
return initial return initial
@ -111,7 +111,7 @@ func (f *freelist) read(p *page) {
ids := ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[0:p.count] ids := ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[0:p.count]
f.ids = make([]pgid, len(ids)) f.ids = make([]pgid, len(ids))
copy(f.ids, ids) copy(f.ids, ids)
sort.Sort(pgids(ids)) sort.Sort(pgids(f.ids))
} }
// write writes the page ids onto a freelist page. All free and pending ids are // write writes the page ids onto a freelist page. All free and pending ids are

View File

@ -29,9 +29,9 @@ func TestFreelist_release(t *testing.T) {
f.free(102, &page{id: 39}) f.free(102, &page{id: 39})
f.release(100) f.release(100)
f.release(101) f.release(101)
assert.Equal(t, f.ids, []pgid{13, 12, 9}) assert.Equal(t, []pgid{9, 12, 13}, f.ids)
f.release(102) f.release(102)
assert.Equal(t, f.ids, []pgid{39, 13, 12, 9}) assert.Equal(t, []pgid{9, 12, 13, 39}, f.ids)
} }
// Ensure that a freelist can find contiguous blocks of pages. // Ensure that a freelist can find contiguous blocks of pages.
@ -44,6 +44,10 @@ func TestFreelist_allocate(t *testing.T) {
assert.Equal(t, 7, int(f.allocate(1))) assert.Equal(t, 7, int(f.allocate(1)))
assert.Equal(t, 0, int(f.allocate(0))) assert.Equal(t, 0, int(f.allocate(0)))
assert.Equal(t, []pgid{9, 18}, f.ids) assert.Equal(t, []pgid{9, 18}, f.ids)
assert.Equal(t, 9, int(f.allocate(1)))
assert.Equal(t, 18, int(f.allocate(1)))
assert.Equal(t, 0, int(f.allocate(1)))
assert.Equal(t, []pgid{}, f.ids)
} }
// Ensure that a freelist can deserialize from a freelist page. // Ensure that a freelist can deserialize from a freelist page.
@ -86,9 +90,9 @@ func TestFreelist_write(t *testing.T) {
// Ensure that the freelist is correct. // Ensure that the freelist is correct.
// All pages should be present and in reverse order. // All pages should be present and in reverse order.
assert.Equal(t, len(f2.ids), 5) assert.Equal(t, len(f2.ids), 5)
assert.Equal(t, f2.ids[0], pgid(39)) assert.Equal(t, f2.ids[0], pgid(3))
assert.Equal(t, f2.ids[1], pgid(28)) assert.Equal(t, f2.ids[1], pgid(11))
assert.Equal(t, f2.ids[2], pgid(12)) assert.Equal(t, f2.ids[2], pgid(12))
assert.Equal(t, f2.ids[3], pgid(11)) assert.Equal(t, f2.ids[3], pgid(28))
assert.Equal(t, f2.ids[4], pgid(3)) assert.Equal(t, f2.ids[4], pgid(39))
} }

View File

@ -133,4 +133,4 @@ type pgids []pgid
func (s pgids) Len() int { return len(s) } func (s pgids) Len() int { return len(s) }
func (s pgids) Swap(i, j int) { s[i], s[j] = s[j], s[i] } func (s pgids) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
func (s pgids) Less(i, j int) bool { return s[i] > s[j] } func (s pgids) Less(i, j int) bool { return s[i] < s[j] }

View File

@ -219,7 +219,7 @@ func TestTx_DeleteBucket(t *testing.T) {
db.Update(func(tx *Tx) error { db.Update(func(tx *Tx) error {
// Verify that the bucket's page is free. // Verify that the bucket's page is free.
assert.Equal(t, []pgid{5, 4}, db.freelist.all()) assert.Equal(t, []pgid{4, 5}, db.freelist.all())
// Create the bucket again and make sure there's not a phantom value. // Create the bucket again and make sure there's not a phantom value.
b, err := tx.CreateBucket([]byte("widgets")) b, err := tx.CreateBucket([]byte("widgets"))