github.com/nicocha30/gvisor-ligolo@v0.0.0-20230726075806-989fa2c0a413/pkg/tcpip/link/sharedmem/pipe/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 pipe
    16  
    17  // Tx is the transmit side of the shared memory ring buffer.
    18  type Tx struct {
    19  	p              pipe
    20  	maxPayloadSize uint64
    21  
    22  	head uint64
    23  	tail uint64
    24  	next uint64
    25  
    26  	tailHeader uint64
    27  }
    28  
    29  // Init initializes the transmit end of the pipe. In the initial state, the next
    30  // slot to be written is the very first one, and the transmitter has the whole
    31  // ring buffer available to it.
    32  func (t *Tx) Init(b []byte) {
    33  	t.p.init(b)
    34  	// maxPayloadSize excludes the header of the payload, and the header
    35  	// of the wrapping message.
    36  	t.maxPayloadSize = uint64(len(t.p.buffer)) - 2*sizeOfSlotHeader
    37  	t.tail = 0xfffffffe * jump
    38  	t.next = t.tail
    39  	t.head = t.tail + jump
    40  	t.p.write(t.tail, slotFree)
    41  }
    42  
    43  // Capacity determines how many records of the given size can be written to the
    44  // pipe before it fills up.
    45  func (t *Tx) Capacity(recordSize uint64) uint64 {
    46  	available := uint64(len(t.p.buffer)) - sizeOfSlotHeader
    47  	entryLen := payloadToSlotSize(recordSize)
    48  	return available / entryLen
    49  }
    50  
    51  // Push reserves "payloadSize" bytes for transmission in the pipe. The caller
    52  // populates the returned slice with the data to be transferred and enventually
    53  // calls Flush() to make the data visible to the reader, or Abort() to make the
    54  // pipe forget all Push() calls since the last Flush().
    55  //
    56  // The returned slice is available until Flush() or Abort() is next called.
    57  // After that, it must not be touched.
    58  func (t *Tx) Push(payloadSize uint64) []byte {
    59  	// Fail request if we know we will never have enough room.
    60  	if payloadSize > t.maxPayloadSize {
    61  		return nil
    62  	}
    63  
    64  	// True if TxPipe currently has a pushed message, i.e., it is not
    65  	// Flush()'ed.
    66  	messageAhead := t.next != t.tail
    67  	totalLen := payloadToSlotSize(payloadSize)
    68  	newNext := t.next + totalLen
    69  	nextWrap := (t.next & revolutionMask) | uint64(len(t.p.buffer))
    70  	if int64(newNext-nextWrap) >= 0 {
    71  		// The new buffer would overflow the pipe, so we push a wrapping
    72  		// slot, then try to add the actual slot to the front of the
    73  		// pipe.
    74  		newNext = (newNext & revolutionMask) + jump
    75  		if !t.reclaim(newNext) {
    76  			return nil
    77  		}
    78  		wrappingPayloadSize := slotToPayloadSize(newNext - t.next)
    79  		oldNext := t.next
    80  		t.next = newNext
    81  		if messageAhead {
    82  			t.p.write(oldNext, wrappingPayloadSize)
    83  		} else {
    84  			t.tailHeader = wrappingPayloadSize
    85  			t.Flush()
    86  		}
    87  		return t.Push(payloadSize)
    88  	}
    89  
    90  	// Check that we have enough room for the buffer.
    91  	if !t.reclaim(newNext) {
    92  		return nil
    93  	}
    94  
    95  	if messageAhead {
    96  		t.p.write(t.next, payloadSize)
    97  	} else {
    98  		t.tailHeader = payloadSize
    99  	}
   100  
   101  	// Grab the buffer before updating t.next.
   102  	b := t.p.data(t.next, payloadSize)
   103  	t.next = newNext
   104  
   105  	return b
   106  }
   107  
   108  // reclaim attempts to advance the head until at least newNext. If the head is
   109  // already at or beyond newNext, nothing happens and true is returned; otherwise
   110  // it tries to reclaim slots that have already been consumed by the receive end
   111  // of the pipe (they will be marked as free) and returns a boolean indicating
   112  // whether it was successful in reclaiming enough slots.
   113  func (t *Tx) reclaim(newNext uint64) bool {
   114  	for int64(newNext-t.head) > 0 {
   115  		// Can't reclaim if slot is not free.
   116  		header := t.p.readAtomic(t.head)
   117  		if header&slotFree == 0 {
   118  			return false
   119  		}
   120  
   121  		payloadSize := header & slotSizeMask
   122  		newHead := t.head + payloadToSlotSize(payloadSize)
   123  
   124  		// Check newHead is within bounds and valid.
   125  		if int64(newHead-t.tail) > int64(jump) || newHead&offsetMask >= uint64(len(t.p.buffer)) {
   126  			return false
   127  		}
   128  
   129  		t.head = newHead
   130  	}
   131  
   132  	return true
   133  }
   134  
   135  // Abort causes all Push() calls since the last Flush() to be forgotten and
   136  // therefore they will not be made visible to the receiver.
   137  func (t *Tx) Abort() {
   138  	t.next = t.tail
   139  }
   140  
   141  // Flush causes all buffers pushed since the last Flush() [or Abort(), whichever
   142  // is the most recent] to be made visible to the receiver.
   143  func (t *Tx) Flush() {
   144  	if t.next == t.tail {
   145  		// Nothing to do if there are no pushed buffers.
   146  		return
   147  	}
   148  
   149  	if t.next != t.head {
   150  		// The receiver will spin in t.next, so we must make sure that
   151  		// the slotFree bit is set.
   152  		t.p.write(t.next, slotFree)
   153  	}
   154  
   155  	t.p.writeAtomic(t.tail, t.tailHeader)
   156  	t.tail = t.next
   157  }
   158  
   159  // Bytes returns the byte slice on which the pipe operates.
   160  func (t *Tx) Bytes() []byte {
   161  	return t.p.buffer
   162  }