blob: 234f5d4337fab40c9631416252a8f51d352ef95e [file] [log] [blame]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package memory
import (
"sync/atomic"
"github.com/apache/arrow/go/arrow/internal/debug"
)
type Buffer struct {
refCount int64
buf []byte
length int
mutable bool
mem Allocator
}
// NewBufferBytes creates a fixed-size buffer from the specified data.
func NewBufferBytes(data []byte) *Buffer {
return &Buffer{refCount: 0, buf: data, length: len(data)}
}
// NewBuffer creates a mutable, resizable buffer with an Allocator for managing memory.
func NewResizableBuffer(mem Allocator) *Buffer {
return &Buffer{refCount: 1, mutable: true, mem: mem}
}
// Retain increases the reference count by 1.
func (b *Buffer) Retain() {
if b.mem != nil {
atomic.AddInt64(&b.refCount, 1)
}
}
// Release decreases the reference count by 1.
// When the reference count goes to zero, the memory is freed.
func (b *Buffer) Release() {
if b.mem != nil {
debug.Assert(atomic.LoadInt64(&b.refCount) > 0, "too many releases")
if atomic.AddInt64(&b.refCount, -1) == 0 {
b.mem.Free(b.buf)
b.buf, b.length = nil, 0
}
}
}
// Buf returns the slice of memory allocated by the Buffer, which is adjusted by calling Reserve.
func (b *Buffer) Buf() []byte { return b.buf }
// Bytes returns a slice of size Len, which is adjusted by calling Resize.
func (b *Buffer) Bytes() []byte { return b.buf[:b.length] }
func (b *Buffer) Mutable() bool { return b.mutable }
func (b *Buffer) Len() int { return b.length }
func (b *Buffer) Cap() int { return len(b.buf) }
func (b *Buffer) Reserve(capacity int) {
if capacity > len(b.buf) {
newCap := roundUpToMultipleOf64(capacity)
if len(b.buf) == 0 {
b.buf = b.mem.Allocate(newCap)
} else {
b.buf = b.mem.Reallocate(newCap, b.buf)
}
}
}
func (b *Buffer) Resize(newSize int) {
b.resize(newSize, true)
}
func (b *Buffer) ResizeNoShrink(newSize int) {
b.resize(newSize, false)
}
func (b *Buffer) resize(newSize int, shrink bool) {
if !shrink || newSize > b.length {
b.Reserve(newSize)
} else {
// Buffer is not growing, so shrink to the requested size without
// excess space.
newCap := roundUpToMultipleOf64(newSize)
if len(b.buf) != newCap {
if newSize == 0 {
b.mem.Free(b.buf)
b.buf = nil
} else {
b.buf = b.mem.Reallocate(newCap, b.buf)
}
}
}
b.length = newSize
}