v2ray-core/common/buf/multi_buffer.go

139 lines
2.6 KiB
Go
Raw Normal View History

2017-04-15 19:07:23 +00:00
package buf
2017-04-19 19:27:21 +00:00
import "net"
2017-04-15 19:07:23 +00:00
type MultiBufferWriter interface {
2017-04-23 11:41:52 +00:00
WriteMultiBuffer(MultiBuffer) error
2017-04-15 19:07:23 +00:00
}
2017-04-15 20:22:29 +00:00
type MultiBufferReader interface {
ReadMultiBuffer() (MultiBuffer, error)
}
2017-04-23 17:16:56 +00:00
// MultiBuffer is a list of Buffers. The order of Buffer matters.
2017-04-15 19:07:23 +00:00
type MultiBuffer []*Buffer
2017-04-23 17:16:56 +00:00
// NewMultiBuffer creates a new MultiBuffer instance.
2017-04-15 19:07:23 +00:00
func NewMultiBuffer() MultiBuffer {
2017-04-16 20:33:20 +00:00
return MultiBuffer(make([]*Buffer, 0, 128))
2017-04-15 19:07:23 +00:00
}
2017-04-23 17:16:56 +00:00
// NewMultiBufferValue wraps a list of Buffers into MultiBuffer.
2017-04-15 19:07:23 +00:00
func NewMultiBufferValue(b ...*Buffer) MultiBuffer {
return MultiBuffer(b)
}
2017-04-23 17:16:56 +00:00
func (mb *MultiBuffer) Append(buf *Buffer) {
*mb = append(*mb, buf)
2017-04-15 19:07:23 +00:00
}
2017-04-23 17:16:56 +00:00
func (mb *MultiBuffer) AppendMulti(buf MultiBuffer) {
*mb = append(*mb, buf...)
2017-04-15 19:07:23 +00:00
}
2017-04-24 23:56:08 +00:00
func (mb MultiBuffer) Copy(b []byte) int {
total := 0
for _, bb := range mb {
nBytes := copy(b[total:], bb.Bytes())
total += nBytes
if nBytes < bb.Len() {
break
}
}
return total
}
2017-04-15 19:07:23 +00:00
func (mb *MultiBuffer) Read(b []byte) (int, error) {
endIndex := len(*mb)
totalBytes := 0
for i, bb := range *mb {
2017-04-19 19:27:21 +00:00
nBytes, _ := bb.Read(b)
2017-04-15 19:07:23 +00:00
totalBytes += nBytes
b = b[nBytes:]
if bb.IsEmpty() {
bb.Release()
} else {
endIndex = i
break
}
}
*mb = (*mb)[endIndex:]
return totalBytes, nil
}
2017-05-01 22:28:16 +00:00
func (mb *MultiBuffer) Write(b []byte) {
n := len(*mb)
if n > 0 && !(*mb)[n-1].IsFull() {
nBytes, _ := (*mb)[n-1].Write(b)
b = b[nBytes:]
}
for len(b) > 0 {
bb := New()
nBytes, _ := bb.Write(b)
b = b[nBytes:]
mb.Append(bb)
}
}
2017-04-23 17:16:56 +00:00
// Len returns the total number of bytes in the MultiBuffer.
2017-04-15 19:07:23 +00:00
func (mb MultiBuffer) Len() int {
size := 0
for _, b := range mb {
size += b.Len()
}
return size
}
2017-04-23 17:16:56 +00:00
// IsEmpty return true if the MultiBuffer has no content.
2017-04-15 19:07:23 +00:00
func (mb MultiBuffer) IsEmpty() bool {
for _, b := range mb {
if !b.IsEmpty() {
return false
}
}
return true
}
2017-04-23 17:16:56 +00:00
// Release releases all Buffers in the MultiBuffer.
2017-04-15 19:07:23 +00:00
func (mb MultiBuffer) Release() {
2017-04-16 11:12:58 +00:00
for i, b := range mb {
2017-04-15 19:07:23 +00:00
b.Release()
2017-04-16 11:12:58 +00:00
mb[i] = nil
2017-04-15 19:07:23 +00:00
}
}
2017-04-16 20:30:29 +00:00
2017-04-23 17:16:56 +00:00
// ToNetBuffers converts this MultiBuffer to net.Buffers. The return net.Buffers points to the same content of the MultiBuffer.
2017-04-16 20:30:29 +00:00
func (mb MultiBuffer) ToNetBuffers() net.Buffers {
bs := make([][]byte, len(mb))
for i, b := range mb {
bs[i] = b.Bytes()
}
return bs
}
func (mb *MultiBuffer) SliceBySize(size int) MultiBuffer {
slice := NewMultiBuffer()
sliceSize := 0
endIndex := len(*mb)
for i, b := range *mb {
if b.Len()+sliceSize > size {
endIndex = i
break
}
2017-04-19 19:27:21 +00:00
sliceSize += b.Len()
slice.Append(b)
}
*mb = (*mb)[endIndex:]
return slice
}
2017-05-01 22:28:16 +00:00
func (mb *MultiBuffer) SplitFirst() *Buffer {
if len(*mb) == 0 {
return nil
}
b := (*mb)[0]
*mb = (*mb)[1:]
return b
}