feat(buffer): implement RFC buffer lifecycle with safety checks

Co-Authored-By: Virgil <virgil@lethean.io>
This commit is contained in:
Virgil 2026-04-03 19:11:30 +00:00
parent 7b75f95ee7
commit 530bfd55f9
3 changed files with 193 additions and 0 deletions

110
buffer.go Normal file
View file

@ -0,0 +1,110 @@
package cgo
import (
"runtime"
"sync/atomic"
"unsafe"
)
// Buffer owns byte memory that can be passed safely to C.
//
// buffer := NewBuffer(16)
// defer buffer.Free()
// n := buffer.CopyFrom([]byte("payload"))
// _ = buffer.Bytes()[:n]
type Buffer struct {
data []byte
length int
pointer unsafe.Pointer
freed atomic.Bool
isPinned bool
pinner runtime.Pinner
}
// NewBuffer allocates memory and pins it so Ptr can be used across C boundaries.
func NewBuffer(size int) *Buffer {
if size < 0 {
panic("cgo.NewBuffer: size must be non-negative")
}
data := make([]byte, size)
buffer := &Buffer{
data: data,
length: size,
}
if size > 0 {
buffer.pinner.Pin(&buffer.data[0])
buffer.isPinned = true
buffer.pointer = unsafe.Pointer(&buffer.data[0])
}
return buffer
}
// Free releases the pinned memory backing slice and marks the buffer as freed.
func (b *Buffer) Free() {
if b == nil {
return
}
if !b.freed.CompareAndSwap(false, true) {
panic("cgo.Buffer.Free: double-free detected")
}
if b.isPinned {
b.pinner.Unpin()
b.isPinned = false
}
b.pointer = nil
b.data = nil
}
// CopyFrom copies bytes from src into the buffer and returns bytes copied.
func (b *Buffer) CopyFrom(src []byte) int {
b.assertNotFreed()
if len(src) == 0 || b.length == 0 {
return 0
}
copied := len(src)
if copied > b.length {
copied = b.length
}
copy(b.data[:copied], src[:copied])
return copied
}
// Bytes returns a mutable byte slice backed by the buffer memory.
func (b *Buffer) Bytes() []byte {
b.assertNotFreed()
return b.data
}
// Ptr returns the raw pointer to the buffer.
func (b *Buffer) Ptr() unsafe.Pointer {
b.assertNotFreed()
return b.pointer
}
// Len returns the current buffer length.
func (b *Buffer) Len() int {
b.assertNotFreed()
return b.length
}
// IsFreed reports whether Free has already been called.
func (b *Buffer) IsFreed() bool {
return b.freed.Load()
}
func (b *Buffer) assertNotFreed() {
if b == nil {
panic("cgo.Buffer: use-after-free detected: buffer is nil")
}
if b.freed.Load() {
panic("cgo.Buffer: use-after-free detected")
}
}

80
buffer_test.go Normal file
View file

@ -0,0 +1,80 @@
package cgo
import "testing"
func TestBufferLifecycleAndCopy(t *testing.T) {
t.Parallel()
const capacity = 8
buffer := NewBuffer(capacity)
defer buffer.Free()
copied := buffer.CopyFrom([]byte{1, 2, 3, 4, 5})
if copied != 5 {
t.Fatalf("expected 5 bytes copied, got %d", copied)
}
if got := buffer.Len(); got != capacity {
t.Fatalf("expected buffer length %d, got %d", capacity, got)
}
if got, want := buffer.Bytes()[:5], []byte{1, 2, 3, 4, 5}; string(got) != string(want) {
t.Fatalf("expected bytes %v, got %v", want, got)
}
}
func TestBufferCopyClipsToCapacity(t *testing.T) {
t.Parallel()
buffer := NewBuffer(2)
defer buffer.Free()
copied := buffer.CopyFrom([]byte("abc"))
if copied != 2 {
t.Fatalf("expected 2 bytes copied, got %d", copied)
}
if got, want := buffer.Bytes(), []byte("ab"); string(got) != string(want) {
t.Fatalf("expected bytes %q, got %q", string(want), string(got))
}
}
func TestBufferDoubleFreePanics(t *testing.T) {
t.Parallel()
buffer := NewBuffer(1)
buffer.Free()
assertPanics(t, "double-free", func() {
buffer.Free()
})
}
func TestBufferUseAfterFreePanics(t *testing.T) {
t.Parallel()
buffer := NewBuffer(4)
buffer.Free()
assertPanics(t, "use-after-free", func() {
_ = buffer.Len()
})
assertPanics(t, "use-after-free", func() {
buffer.CopyFrom([]byte("x"))
})
assertPanics(t, "use-after-free", func() {
_ = buffer.Bytes()
})
assertPanics(t, "use-after-free", func() {
_ = buffer.Ptr()
})
}
func assertPanics(t *testing.T, want string, fn func()) {
t.Helper()
defer func() {
if r := recover(); r == nil {
t.Fatalf("expected panic for %s, got none", want)
}
}()
fn()
}

3
go.mod Normal file
View file

@ -0,0 +1,3 @@
module dappco.re/go/cgo
go 1.22