github.com/google/netstack@v0.0.0-20191123085552-55fcc16cd0eb/tcpip/link/sharedmem/queue/rx.go (about) 1 // Copyright 2018 The gVisor Authors. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 // Package queue provides the implementation of transmit and receive queues 16 // based on shared memory ring buffers. 17 package queue 18 19 import ( 20 "encoding/binary" 21 "sync/atomic" 22 23 "github.com/google/netstack/tcpip/link/sharedmem/pipe" 24 "log" 25 ) 26 27 const ( 28 // Offsets within a posted buffer. 29 postedOffset = 0 30 postedSize = 8 31 postedRemainingInGroup = 12 32 postedUserData = 16 33 postedID = 24 34 35 sizeOfPostedBuffer = 32 36 37 // Offsets within a received packet header. 38 consumedPacketSize = 0 39 consumedPacketReserved = 4 40 41 sizeOfConsumedPacketHeader = 8 42 43 // Offsets within a consumed buffer. 44 consumedOffset = 0 45 consumedSize = 8 46 consumedUserData = 12 47 consumedID = 20 48 49 sizeOfConsumedBuffer = 28 50 51 // The following are the allowed states of the shared data area. 52 eventFDUninitialized = 0 53 eventFDDisabled = 1 54 eventFDEnabled = 2 55 ) 56 57 // RxBuffer is the descriptor of a receive buffer. 58 type RxBuffer struct { 59 Offset uint64 60 Size uint32 61 ID uint64 62 UserData uint64 63 } 64 65 // Rx is a receive queue. It is implemented with one tx and one rx pipe: the tx 66 // pipe is used to "post" buffers, while the rx pipe is used to receive packets 67 // whose contents have been written to previously posted buffers. 68 // 69 // This struct is thread-compatible. 70 type Rx struct { 71 tx pipe.Tx 72 rx pipe.Rx 73 sharedEventFDState *uint32 74 } 75 76 // Init initializes the receive queue with the given pipes, and shared state 77 // pointer -- the latter is used to enable/disable eventfd notifications. 78 func (r *Rx) Init(tx, rx []byte, sharedEventFDState *uint32) { 79 r.sharedEventFDState = sharedEventFDState 80 r.tx.Init(tx) 81 r.rx.Init(rx) 82 } 83 84 // EnableNotification updates the shared state such that the peer will notify 85 // the eventfd when there are packets to be dequeued. 86 func (r *Rx) EnableNotification() { 87 atomic.StoreUint32(r.sharedEventFDState, eventFDEnabled) 88 } 89 90 // DisableNotification updates the shared state such that the peer will not 91 // notify the eventfd. 92 func (r *Rx) DisableNotification() { 93 atomic.StoreUint32(r.sharedEventFDState, eventFDDisabled) 94 } 95 96 // PostedBuffersLimit returns the maximum number of buffers that can be posted 97 // before the tx queue fills up. 98 func (r *Rx) PostedBuffersLimit() uint64 { 99 return r.tx.Capacity(sizeOfPostedBuffer) 100 } 101 102 // PostBuffers makes the given buffers available for receiving data from the 103 // peer. Once they are posted, the peer is free to write to them and will 104 // eventually post them back for consumption. 105 func (r *Rx) PostBuffers(buffers []RxBuffer) bool { 106 for i := range buffers { 107 b := r.tx.Push(sizeOfPostedBuffer) 108 if b == nil { 109 r.tx.Abort() 110 return false 111 } 112 113 pb := &buffers[i] 114 binary.LittleEndian.PutUint64(b[postedOffset:], pb.Offset) 115 binary.LittleEndian.PutUint32(b[postedSize:], pb.Size) 116 binary.LittleEndian.PutUint32(b[postedRemainingInGroup:], 0) 117 binary.LittleEndian.PutUint64(b[postedUserData:], pb.UserData) 118 binary.LittleEndian.PutUint64(b[postedID:], pb.ID) 119 } 120 121 r.tx.Flush() 122 123 return true 124 } 125 126 // Dequeue receives buffers that have been previously posted by PostBuffers() 127 // and that have been filled by the peer and posted back. 128 // 129 // This is similar to append() in that new buffers are appended to "bufs", with 130 // reallocation only if "bufs" doesn't have enough capacity. 131 func (r *Rx) Dequeue(bufs []RxBuffer) ([]RxBuffer, uint32) { 132 for { 133 outBufs := bufs 134 135 // Pull the next descriptor from the rx pipe. 136 b := r.rx.Pull() 137 if b == nil { 138 return bufs, 0 139 } 140 141 if len(b) < sizeOfConsumedPacketHeader { 142 log.Printf("Ignoring packet header: size (%v) is less than header size (%v)", len(b), sizeOfConsumedPacketHeader) 143 r.rx.Flush() 144 continue 145 } 146 147 totalDataSize := binary.LittleEndian.Uint32(b[consumedPacketSize:]) 148 149 // Calculate the number of buffer descriptors and copy them 150 // over to the output. 151 count := (len(b) - sizeOfConsumedPacketHeader) / sizeOfConsumedBuffer 152 offset := sizeOfConsumedPacketHeader 153 buffersSize := uint32(0) 154 for i := count; i > 0; i-- { 155 s := binary.LittleEndian.Uint32(b[offset+consumedSize:]) 156 buffersSize += s 157 if buffersSize < s { 158 // The buffer size overflows an unsigned 32-bit 159 // integer, so break out and force it to be 160 // ignored. 161 totalDataSize = 1 162 buffersSize = 0 163 break 164 } 165 166 outBufs = append(outBufs, RxBuffer{ 167 Offset: binary.LittleEndian.Uint64(b[offset+consumedOffset:]), 168 Size: s, 169 ID: binary.LittleEndian.Uint64(b[offset+consumedID:]), 170 }) 171 172 offset += sizeOfConsumedBuffer 173 } 174 175 r.rx.Flush() 176 177 if buffersSize < totalDataSize { 178 // The descriptor is corrupted, ignore it. 179 log.Printf("Ignoring packet: actual data size (%v) less than expected size (%v)", buffersSize, totalDataSize) 180 continue 181 } 182 183 return outBufs, totalDataSize 184 } 185 } 186 187 // Bytes returns the byte slices on which the queue operates. 188 func (r *Rx) Bytes() (tx, rx []byte) { 189 return r.tx.Bytes(), r.rx.Bytes() 190 } 191 192 // DecodeRxBufferHeader decodes the header of a buffer posted on an rx queue. 193 func DecodeRxBufferHeader(b []byte) RxBuffer { 194 return RxBuffer{ 195 Offset: binary.LittleEndian.Uint64(b[postedOffset:]), 196 Size: binary.LittleEndian.Uint32(b[postedSize:]), 197 ID: binary.LittleEndian.Uint64(b[postedID:]), 198 UserData: binary.LittleEndian.Uint64(b[postedUserData:]), 199 } 200 } 201 202 // RxCompletionSize returns the number of bytes needed to encode an rx 203 // completion containing "count" buffers. 204 func RxCompletionSize(count int) uint64 { 205 return sizeOfConsumedPacketHeader + uint64(count)*sizeOfConsumedBuffer 206 } 207 208 // EncodeRxCompletion encodes an rx completion header. 209 func EncodeRxCompletion(b []byte, size, reserved uint32) { 210 binary.LittleEndian.PutUint32(b[consumedPacketSize:], size) 211 binary.LittleEndian.PutUint32(b[consumedPacketReserved:], reserved) 212 } 213 214 // EncodeRxCompletionBuffer encodes the i-th rx completion buffer header. 215 func EncodeRxCompletionBuffer(b []byte, i int, rxb RxBuffer) { 216 b = b[RxCompletionSize(i):] 217 binary.LittleEndian.PutUint64(b[consumedOffset:], rxb.Offset) 218 binary.LittleEndian.PutUint32(b[consumedSize:], rxb.Size) 219 binary.LittleEndian.PutUint64(b[consumedUserData:], rxb.UserData) 220 binary.LittleEndian.PutUint64(b[consumedID:], rxb.ID) 221 }