mirror of
https://github.com/v2fly/v2ray-core.git
synced 2025-01-02 23:47:07 -05:00
simplify receiving worker
This commit is contained in:
parent
5d20e3f70b
commit
2839ce7a88
@ -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 {
|
||||||
|
@ -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)
|
|
||||||
}
|
|
||||||
|
Loading…
Reference in New Issue
Block a user