1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-01-20 08:16:55 -05:00

simplify receiving worker

This commit is contained in:
Darien Raymond 2016-08-24 23:51:53 +02:00
parent 5d20e3f70b
commit 2839ce7a88
No known key found for this signature in database
GPG Key ID: 7251FFA14BB18169
2 changed files with 35 additions and 104 deletions

View File

@ -55,68 +55,6 @@ func (this *ReceivingWindow) Advance() {
} }
} }
type ReceivingQueue struct {
start uint32
cap uint32
len uint32
data []*alloc.Buffer
}
func NewReceivingQueue(size uint32) *ReceivingQueue {
return &ReceivingQueue{
cap: size,
data: make([]*alloc.Buffer, size),
}
}
func (this *ReceivingQueue) IsEmpty() bool {
return this.len == 0
}
func (this *ReceivingQueue) IsFull() bool {
return this.len == this.cap
}
func (this *ReceivingQueue) Read(buf []byte) int {
if this.IsEmpty() {
return 0
}
totalBytes := 0
lenBuf := len(buf)
for !this.IsEmpty() && totalBytes < lenBuf {
payload := this.data[this.start]
nBytes, _ := payload.Read(buf)
buf = buf[nBytes:]
totalBytes += nBytes
if payload.IsEmpty() {
payload.Release()
this.data[this.start] = nil
this.start++
if this.start == this.cap {
this.start = 0
}
this.len--
if this.len == 0 {
this.start = 0
}
}
}
return totalBytes
}
func (this *ReceivingQueue) Put(payload *alloc.Buffer) {
this.data[(this.start+this.len)%this.cap] = payload
this.len++
}
func (this *ReceivingQueue) Close() {
for i := uint32(0); i < this.len; i++ {
this.data[(this.start+i)%this.cap].Release()
this.data[(this.start+i)%this.cap] = nil
}
}
type AckList struct { type AckList struct {
writer SegmentWriter writer SegmentWriter
timestamps []uint32 timestamps []uint32
@ -176,7 +114,7 @@ func (this *AckList) Flush(current uint32, rto uint32) {
type ReceivingWorker struct { type ReceivingWorker struct {
sync.RWMutex sync.RWMutex
conn *Connection conn *Connection
queue *ReceivingQueue leftOver *alloc.Buffer
window *ReceivingWindow window *ReceivingWindow
acklist *AckList acklist *AckList
updated bool updated bool
@ -185,10 +123,9 @@ type ReceivingWorker struct {
} }
func NewReceivingWorker(kcp *Connection) *ReceivingWorker { func NewReceivingWorker(kcp *Connection) *ReceivingWorker {
windowSize := effectiveConfig.GetReceivingWindowSize() windowSize := effectiveConfig.GetReceivingQueueSize()
worker := &ReceivingWorker{ worker := &ReceivingWorker{
conn: kcp, conn: kcp,
queue: NewReceivingQueue(effectiveConfig.GetReceivingQueueSize()),
window: NewReceivingWindow(windowSize), window: NewReceivingWindow(windowSize),
windowSize: windowSize, windowSize: windowSize,
} }
@ -218,27 +155,45 @@ func (this *ReceivingWorker) ProcessSegment(seg *DataSegment) {
if !this.window.Set(idx, seg) { if !this.window.Set(idx, seg) {
seg.Release() seg.Release()
} }
for !this.queue.IsFull() {
seg := this.window.RemoveFirst()
if seg == nil {
break
}
this.queue.Put(seg.Data)
seg.Data = nil
seg.Release()
this.window.Advance()
this.nextNumber++
this.updated = true
}
} }
func (this *ReceivingWorker) Read(b []byte) int { func (this *ReceivingWorker) Read(b []byte) int {
this.Lock() this.Lock()
defer this.Unlock() defer this.Unlock()
return this.queue.Read(b) total := 0
if this.leftOver != nil {
nBytes := copy(b, this.leftOver.Value)
if nBytes < this.leftOver.Len() {
this.leftOver.SliceFrom(nBytes)
return nBytes
}
this.leftOver.Release()
this.leftOver = nil
total += nBytes
}
for total < len(b) {
seg := this.window.RemoveFirst()
if seg == nil {
break
}
this.window.Advance()
this.nextNumber++
this.updated = true
nBytes := copy(b[total:], seg.Data.Value)
total += nBytes
if nBytes < seg.Data.Len() {
seg.Data.SliceFrom(nBytes)
this.leftOver = seg.Data
seg.Data = nil
seg.Release()
break
}
seg.Release()
}
return total
} }
func (this *ReceivingWorker) Flush(current uint32) { func (this *ReceivingWorker) Flush(current uint32) {
@ -261,10 +216,6 @@ func (this *ReceivingWorker) Write(seg Segment) {
} }
func (this *ReceivingWorker) CloseRead() { func (this *ReceivingWorker) CloseRead() {
this.Lock()
defer this.Unlock()
this.queue.Close()
} }
func (this *ReceivingWorker) PingNecessary() bool { func (this *ReceivingWorker) PingNecessary() bool {

View File

@ -3,7 +3,6 @@ package kcp_test
import ( import (
"testing" "testing"
"v2ray.com/core/common/alloc"
"v2ray.com/core/testing/assert" "v2ray.com/core/testing/assert"
. "v2ray.com/core/transport/internet/kcp" . "v2ray.com/core/transport/internet/kcp"
) )
@ -35,22 +34,3 @@ func TestRecivingWindow(t *testing.T) {
assert.Pointer(window.Remove(1)).Equals(seg2) assert.Pointer(window.Remove(1)).Equals(seg2)
assert.Pointer(window.Remove(2)).Equals(seg3) assert.Pointer(window.Remove(2)).Equals(seg3)
} }
func TestRecivingQueue(t *testing.T) {
assert := assert.On(t)
queue := NewReceivingQueue(2)
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)
nBytes := queue.Read(b)
assert.Int(nBytes).Equals(7)
assert.String(string(b[:nBytes])).Equals("abcdefg")
queue.Put(alloc.NewLocalBuffer(512).Clear().AppendString("1"))
queue.Close()
nBytes = queue.Read(b)
assert.Int(nBytes).Equals(0)
}