Try to use reflect.SliceHeader correctly this time

pull/220/head
Josh Rickmar 2020-05-21 18:50:41 +00:00
parent 81f25783ae
commit 9034717d69
3 changed files with 38 additions and 13 deletions

View File

@ -267,18 +267,23 @@ func (f *freelist) read(p *page) {
} }
// If the page.count is at the max uint16 value (64k) then it's considered // If the page.count is at the max uint16 value (64k) then it's considered
// an overflow and the size of the freelist is stored as the first element. // an overflow and the size of the freelist is stored as the first element.
var idx, count uintptr = 0, uintptr(p.count) var idx, count = 0, int(p.count)
if count == 0xFFFF { if count == 0xFFFF {
idx = 1 idx = 1
count = uintptr(*(*pgid)(unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p)))) c := *(*pgid)(unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p)))
count = int(c)
if count < 0 {
panic(fmt.Sprintf("leading element count %d overflows int", c))
}
} }
// Copy the list of page ids from the freelist. // Copy the list of page ids from the freelist.
if count == 0 { if count == 0 {
f.ids = nil f.ids = nil
} else { } else {
ids := (*[maxAllocSize]pgid)(unsafeAdd( var ids []pgid
unsafe.Pointer(p), unsafe.Sizeof(*p)))[idx : idx+count : idx+count] data := unsafeIndex(unsafe.Pointer(p), unsafe.Sizeof(*p), unsafe.Sizeof(ids[0]), idx)
unsafeSlice(unsafe.Pointer(&ids), data, count)
// copy the ids, so we don't modify on the freelist page directly // copy the ids, so we don't modify on the freelist page directly
idsCopy := make([]pgid, count) idsCopy := make([]pgid, count)
@ -316,11 +321,15 @@ func (f *freelist) write(p *page) error {
p.count = uint16(l) p.count = uint16(l)
} else if l < 0xFFFF { } else if l < 0xFFFF {
p.count = uint16(l) p.count = uint16(l)
ids := (*[maxAllocSize]pgid)(unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p)))[:l:l] var ids []pgid
data := unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p))
unsafeSlice(unsafe.Pointer(&ids), data, l)
f.copyall(ids) f.copyall(ids)
} else { } else {
p.count = 0xFFFF p.count = 0xFFFF
ids := (*[maxAllocSize]pgid)(unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p)))[: l+1 : l+1] var ids []pgid
data := unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p))
unsafeSlice(unsafe.Pointer(&ids), data, l+1)
ids[0] = pgid(l) ids[0] = pgid(l)
f.copyall(ids[1:]) f.copyall(ids[1:])
} }

14
page.go
View File

@ -64,9 +64,10 @@ func (p *page) leafPageElements() []leafPageElement {
if p.count == 0 { if p.count == 0 {
return nil return nil
} }
const maxArraySize = maxAllocSize / unsafe.Sizeof(leafPageElement{}) var elems []leafPageElement
return (*[maxArraySize]leafPageElement)(unsafeIndex(unsafe.Pointer(p), unsafe.Sizeof(*p), data := unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p))
unsafe.Sizeof(leafPageElement{}), 0))[:p.count:p.count] unsafeSlice(unsafe.Pointer(&elems), data, int(p.count))
return elems
} }
// branchPageElement retrieves the branch node by index // branchPageElement retrieves the branch node by index
@ -80,9 +81,10 @@ func (p *page) branchPageElements() []branchPageElement {
if p.count == 0 { if p.count == 0 {
return nil return nil
} }
const maxArraySize = maxAllocSize / unsafe.Sizeof(branchPageElement{}) var elems []branchPageElement
return (*[maxArraySize]branchPageElement)(unsafeIndex(unsafe.Pointer(p), unsafe.Sizeof(*p), data := unsafeAdd(unsafe.Pointer(p), unsafe.Sizeof(*p))
unsafe.Sizeof(branchPageElement{}), 0))[:p.count:p.count] unsafeSlice(unsafe.Pointer(&elems), data, int(p.count))
return elems
} }
// dump writes n bytes of the page to STDERR as hex output. // dump writes n bytes of the page to STDERR as hex output.

View File

@ -1,6 +1,9 @@
package bbolt package bbolt
import "unsafe" import (
"reflect"
"unsafe"
)
func unsafeAdd(base unsafe.Pointer, offset uintptr) unsafe.Pointer { func unsafeAdd(base unsafe.Pointer, offset uintptr) unsafe.Pointer {
return unsafe.Pointer(uintptr(base) + offset) return unsafe.Pointer(uintptr(base) + offset)
@ -23,3 +26,14 @@ func unsafeByteSlice(base unsafe.Pointer, offset uintptr, i, j int) []byte {
// all), so this is believed to be correct. // all), so this is believed to be correct.
return (*[maxAllocSize]byte)(unsafeAdd(base, offset))[i:j:j] return (*[maxAllocSize]byte)(unsafeAdd(base, offset))[i:j:j]
} }
// unsafeSlice modifies the data, len, and cap of a slice variable pointed to by
// the slice parameter. This helper should be used over other direct
// manipulation of reflect.SliceHeader to prevent misuse, namely, converting
// from reflect.SliceHeader to a Go slice type.
func unsafeSlice(slice, data unsafe.Pointer, len int) {
s := (*reflect.SliceHeader)(slice)
s.Data = uintptr(data)
s.Cap = len
s.Len = len
}