1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2024-12-22 01:57:12 -05:00

refine buffer usage in kcp

This commit is contained in:
v2ray 2016-07-28 16:24:15 +02:00
parent 34b4fceaa4
commit 42efa5dde0
No known key found for this signature in database
GPG Key ID: 7251FFA14BB18169
10 changed files with 148 additions and 19 deletions

View File

@ -17,12 +17,12 @@ const (
// quickly.
type Buffer struct {
head []byte
pool *BufferPool
pool Pool
Value []byte
offset int
}
func CreateBuffer(container []byte, parent *BufferPool) *Buffer {
func CreateBuffer(container []byte, parent Pool) *Buffer {
b := new(Buffer)
b.head = container
b.pool = parent

View File

@ -4,6 +4,11 @@ import (
"sync"
)
type Pool interface {
Allocate() *Buffer
Free(*Buffer)
}
type BufferPool struct {
chain chan []byte
allocator *sync.Pool

View File

@ -0,0 +1,102 @@
package kcp
import (
"sync"
"github.com/v2ray/v2ray-core/common/alloc"
)
const (
NumDistro = 5
DistroSize = 1600
)
type Buffer struct {
sync.Mutex
buffer *alloc.Buffer
next int
released int
hold bool
distro [NumDistro]*alloc.Buffer
}
func NewBuffer() *Buffer {
b := &Buffer{
next: 0,
released: 0,
hold: true,
buffer: alloc.NewBuffer(),
}
for idx := range b.distro {
content := b.buffer.Value[idx*DistroSize : (idx+1)*DistroSize]
b.distro[idx] = alloc.CreateBuffer(content, b)
}
return b
}
func (this *Buffer) IsEmpty() bool {
this.Lock()
defer this.Unlock()
return this.next == NumDistro
}
func (this *Buffer) Allocate() *alloc.Buffer {
this.Lock()
defer this.Unlock()
if this.next == NumDistro {
return nil
}
b := this.distro[this.next]
this.next++
return b
}
func (this *Buffer) Free(b *alloc.Buffer) {
this.Lock()
defer this.Unlock()
this.released++
if !this.hold && this.released == this.next {
this.ReleaseBuffer()
}
}
func (this *Buffer) Release() {
this.Lock()
defer this.Unlock()
if this.next == this.released {
this.ReleaseBuffer()
}
this.hold = false
}
func (this *Buffer) ReleaseBuffer() {
this.buffer.Release()
this.buffer = nil
for idx := range this.distro {
this.distro[idx] = nil
}
}
var (
globalBuffer *Buffer
globalBufferAccess sync.Mutex
)
func AllocateBuffer() *alloc.Buffer {
globalBufferAccess.Lock()
defer globalBufferAccess.Unlock()
if globalBuffer == nil {
globalBuffer = NewBuffer()
}
b := globalBuffer.Allocate()
if globalBuffer.IsEmpty() {
globalBuffer.Release()
globalBuffer = nil
}
return b
}

View File

@ -0,0 +1,22 @@
package kcp_test
import (
"testing"
"github.com/v2ray/v2ray-core/testing/assert"
. "github.com/v2ray/v2ray-core/transport/internet/kcp"
)
func TestBuffer(t *testing.T) {
assert := assert.On(t)
b := NewBuffer()
for i := 0; i < NumDistro; i++ {
x := b.Allocate()
assert.Pointer(x).IsNotNil()
x.Release()
}
assert.Pointer(b.Allocate()).IsNil()
b.Release()
}

View File

@ -12,7 +12,7 @@ import (
func TestSimpleAuthenticator(t *testing.T) {
assert := assert.On(t)
buffer := alloc.NewBuffer().Clear()
buffer := alloc.NewLocalBuffer(512).Clear()
buffer.AppendBytes('a', 'b', 'c', 'd', 'e', 'f', 'g')
auth := NewSimpleAuthenticator()
@ -25,7 +25,7 @@ func TestSimpleAuthenticator(t *testing.T) {
func TestSimpleAuthenticator2(t *testing.T) {
assert := assert.On(t)
buffer := alloc.NewBuffer().Clear()
buffer := alloc.NewLocalBuffer(512).Clear()
buffer.AppendBytes('1', '2')
auth := NewSimpleAuthenticator()
@ -36,7 +36,7 @@ func TestSimpleAuthenticator2(t *testing.T) {
}
func BenchmarkSimpleAuthenticator(b *testing.B) {
buffer := alloc.NewBuffer().Clear()
buffer := alloc.NewLocalBuffer(2048).Clear()
buffer.Slice(0, 1024)
rand.Read(buffer.Value)

View File

@ -40,8 +40,8 @@ func TestRecivingQueue(t *testing.T) {
assert := assert.On(t)
queue := NewReceivingQueue(2)
queue.Put(alloc.NewSmallBuffer().Clear().AppendString("abcd"))
queue.Put(alloc.NewSmallBuffer().Clear().AppendString("efg"))
queue.Put(alloc.NewLocalBuffer(512).Clear().AppendString("abcd"))
queue.Put(alloc.NewLocalBuffer(512).Clear().AppendString("efg"))
assert.Bool(queue.IsFull()).IsTrue()
b := make([]byte, 1024)
@ -49,7 +49,7 @@ func TestRecivingQueue(t *testing.T) {
assert.Int(nBytes).Equals(7)
assert.String(string(b[:nBytes])).Equals("abcdefg")
queue.Put(alloc.NewSmallBuffer().Clear().AppendString("1"))
queue.Put(alloc.NewLocalBuffer(512).Clear().AppendString("1"))
queue.Close()
nBytes = queue.Read(b)
assert.Int(nBytes).Equals(0)

View File

@ -176,7 +176,7 @@ func ReadSegment(buf []byte) (Segment, []byte) {
if len(buf) < dataLen {
return nil, nil
}
seg.Data = alloc.NewSmallBuffer().Clear().Append(buf[:dataLen])
seg.Data = AllocateBuffer().Clear().Append(buf[:dataLen])
buf = buf[dataLen:]
return seg, buf

View File

@ -24,7 +24,7 @@ func TestDataSegment(t *testing.T) {
Timestamp: 3,
Number: 4,
SendingNext: 5,
Data: alloc.NewSmallBuffer().Clear().Append([]byte{'a', 'b', 'c', 'd'}),
Data: alloc.NewLocalBuffer(512).Clear().Append([]byte{'a', 'b', 'c', 'd'}),
}
nBytes := seg.ByteSize()

View File

@ -346,7 +346,7 @@ func (this *SendingWorker) Push(b []byte) int {
} else {
size = len(b)
}
this.queue.Push(alloc.NewSmallBuffer().Clear().Append(b[:size]))
this.queue.Push(AllocateBuffer().Clear().Append(b[:size]))
b = b[size:]
nBytes += size
}

View File

@ -13,10 +13,10 @@ func TestSendingQueue(t *testing.T) {
queue := NewSendingQueue(3)
seg0 := alloc.NewBuffer()
seg1 := alloc.NewBuffer()
seg2 := alloc.NewBuffer()
seg3 := alloc.NewBuffer()
seg0 := alloc.NewLocalBuffer(512)
seg1 := alloc.NewLocalBuffer(512)
seg2 := alloc.NewLocalBuffer(512)
seg3 := alloc.NewLocalBuffer(512)
assert.Bool(queue.IsEmpty()).IsTrue()
assert.Bool(queue.IsFull()).IsFalse()
@ -45,10 +45,10 @@ func TestSendingQueueClear(t *testing.T) {
queue := NewSendingQueue(3)
seg0 := alloc.NewBuffer()
seg1 := alloc.NewBuffer()
seg2 := alloc.NewBuffer()
seg3 := alloc.NewBuffer()
seg0 := alloc.NewLocalBuffer(512)
seg1 := alloc.NewLocalBuffer(512)
seg2 := alloc.NewLocalBuffer(512)
seg3 := alloc.NewLocalBuffer(512)
queue.Push(seg0)
assert.Bool(queue.IsEmpty()).IsFalse()