github.com/google/netstack@v0.0.0-20191123085552-55fcc16cd0eb/tcpip/link/sharedmem/queue/tx.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
    16  
    17  import (
    18  	"encoding/binary"
    19  
    20  	"github.com/google/netstack/tcpip/link/sharedmem/pipe"
    21  	"log"
    22  )
    23  
    24  const (
    25  	// Offsets within a packet header.
    26  	packetID       = 0
    27  	packetSize     = 8
    28  	packetReserved = 12
    29  
    30  	sizeOfPacketHeader = 16
    31  
    32  	// Offsets with a buffer descriptor
    33  	bufferOffset = 0
    34  	bufferSize   = 8
    35  
    36  	sizeOfBufferDescriptor = 12
    37  )
    38  
    39  // TxBuffer is the descriptor of a transmit buffer.
    40  type TxBuffer struct {
    41  	Next   *TxBuffer
    42  	Offset uint64
    43  	Size   uint32
    44  }
    45  
    46  // Tx is a transmit queue. It is implemented with one tx and one rx pipe: the
    47  // tx pipe is used to request the transmission of packets, while the rx pipe
    48  // is used to receive which transmissions have completed.
    49  //
    50  // This struct is thread-compatible.
    51  type Tx struct {
    52  	tx pipe.Tx
    53  	rx pipe.Rx
    54  }
    55  
    56  // Init initializes the transmit queue with the given pipes.
    57  func (t *Tx) Init(tx, rx []byte) {
    58  	t.tx.Init(tx)
    59  	t.rx.Init(rx)
    60  }
    61  
    62  // Enqueue queues the given linked list of buffers for transmission as one
    63  // packet. While it is queued, the caller must not modify them.
    64  func (t *Tx) Enqueue(id uint64, totalDataLen, bufferCount uint32, buffer *TxBuffer) bool {
    65  	// Reserve room in the tx pipe.
    66  	totalLen := sizeOfPacketHeader + uint64(bufferCount)*sizeOfBufferDescriptor
    67  
    68  	b := t.tx.Push(totalLen)
    69  	if b == nil {
    70  		return false
    71  	}
    72  
    73  	// Initialize the packet and buffer descriptors.
    74  	binary.LittleEndian.PutUint64(b[packetID:], id)
    75  	binary.LittleEndian.PutUint32(b[packetSize:], totalDataLen)
    76  	binary.LittleEndian.PutUint32(b[packetReserved:], 0)
    77  
    78  	offset := sizeOfPacketHeader
    79  	for i := bufferCount; i != 0; i-- {
    80  		binary.LittleEndian.PutUint64(b[offset+bufferOffset:], buffer.Offset)
    81  		binary.LittleEndian.PutUint32(b[offset+bufferSize:], buffer.Size)
    82  		offset += sizeOfBufferDescriptor
    83  		buffer = buffer.Next
    84  	}
    85  
    86  	t.tx.Flush()
    87  
    88  	return true
    89  }
    90  
    91  // CompletedPacket returns the id of the last completed transmission. The
    92  // returned id, if any, refers to a value passed on a previous call to
    93  // Enqueue().
    94  func (t *Tx) CompletedPacket() (id uint64, ok bool) {
    95  	for {
    96  		b := t.rx.Pull()
    97  		if b == nil {
    98  			return 0, false
    99  		}
   100  
   101  		if len(b) != 8 {
   102  			t.rx.Flush()
   103  			log.Printf("Ignoring completed packet: size (%v) is less than expected (%v)", len(b), 8)
   104  			continue
   105  		}
   106  
   107  		v := binary.LittleEndian.Uint64(b)
   108  
   109  		t.rx.Flush()
   110  
   111  		return v, true
   112  	}
   113  }
   114  
   115  // Bytes returns the byte slices on which the queue operates.
   116  func (t *Tx) Bytes() (tx, rx []byte) {
   117  	return t.tx.Bytes(), t.rx.Bytes()
   118  }
   119  
   120  // TxPacketInfo holds information about a packet sent on a tx queue.
   121  type TxPacketInfo struct {
   122  	ID          uint64
   123  	Size        uint32
   124  	Reserved    uint32
   125  	BufferCount int
   126  }
   127  
   128  // DecodeTxPacketHeader decodes the header of a packet sent over a tx queue.
   129  func DecodeTxPacketHeader(b []byte) TxPacketInfo {
   130  	return TxPacketInfo{
   131  		ID:          binary.LittleEndian.Uint64(b[packetID:]),
   132  		Size:        binary.LittleEndian.Uint32(b[packetSize:]),
   133  		Reserved:    binary.LittleEndian.Uint32(b[packetReserved:]),
   134  		BufferCount: (len(b) - sizeOfPacketHeader) / sizeOfBufferDescriptor,
   135  	}
   136  }
   137  
   138  // DecodeTxBufferHeader decodes the header of the i-th buffer of a packet sent
   139  // over a tx queue.
   140  func DecodeTxBufferHeader(b []byte, i int) TxBuffer {
   141  	b = b[sizeOfPacketHeader+i*sizeOfBufferDescriptor:]
   142  	return TxBuffer{
   143  		Offset: binary.LittleEndian.Uint64(b[bufferOffset:]),
   144  		Size:   binary.LittleEndian.Uint32(b[bufferSize:]),
   145  	}
   146  }
   147  
   148  // EncodeTxCompletion encodes a tx completion header.
   149  func EncodeTxCompletion(b []byte, id uint64) {
   150  	binary.LittleEndian.PutUint64(b, id)
   151  }