github.com/google/netstack@v0.0.0-20191123085552-55fcc16cd0eb/tcpip/link/sharedmem/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 // +build linux 16 17 package sharedmem 18 19 import ( 20 "sync/atomic" 21 "syscall" 22 23 "github.com/google/netstack/tcpip/link/rawfile" 24 "github.com/google/netstack/tcpip/link/sharedmem/queue" 25 ) 26 27 // rx holds all state associated with an rx queue. 28 type rx struct { 29 data []byte 30 sharedData []byte 31 q queue.Rx 32 eventFD int 33 } 34 35 // init initializes all state needed by the rx queue based on the information 36 // provided. 37 // 38 // The caller always retains ownership of all file descriptors passed in. The 39 // queue implementation will duplicate any that it may need in the future. 40 func (r *rx) init(mtu uint32, c *QueueConfig) error { 41 // Map in all buffers. 42 txPipe, err := getBuffer(c.TxPipeFD) 43 if err != nil { 44 return err 45 } 46 47 rxPipe, err := getBuffer(c.RxPipeFD) 48 if err != nil { 49 syscall.Munmap(txPipe) 50 return err 51 } 52 53 data, err := getBuffer(c.DataFD) 54 if err != nil { 55 syscall.Munmap(txPipe) 56 syscall.Munmap(rxPipe) 57 return err 58 } 59 60 sharedData, err := getBuffer(c.SharedDataFD) 61 if err != nil { 62 syscall.Munmap(txPipe) 63 syscall.Munmap(rxPipe) 64 syscall.Munmap(data) 65 return err 66 } 67 68 // Duplicate the eventFD so that caller can close it but we can still 69 // use it. 70 efd, err := syscall.Dup(c.EventFD) 71 if err != nil { 72 syscall.Munmap(txPipe) 73 syscall.Munmap(rxPipe) 74 syscall.Munmap(data) 75 syscall.Munmap(sharedData) 76 return err 77 } 78 79 // Set the eventfd as non-blocking. 80 if err := syscall.SetNonblock(efd, true); err != nil { 81 syscall.Munmap(txPipe) 82 syscall.Munmap(rxPipe) 83 syscall.Munmap(data) 84 syscall.Munmap(sharedData) 85 syscall.Close(efd) 86 return err 87 } 88 89 // Initialize state based on buffers. 90 r.q.Init(txPipe, rxPipe, sharedDataPointer(sharedData)) 91 r.data = data 92 r.eventFD = efd 93 r.sharedData = sharedData 94 95 return nil 96 } 97 98 // cleanup releases all resources allocated during init(). It must only be 99 // called if init() has previously succeeded. 100 func (r *rx) cleanup() { 101 a, b := r.q.Bytes() 102 syscall.Munmap(a) 103 syscall.Munmap(b) 104 105 syscall.Munmap(r.data) 106 syscall.Munmap(r.sharedData) 107 syscall.Close(r.eventFD) 108 } 109 110 // postAndReceive posts the provided buffers (if any), and then tries to read 111 // from the receive queue. 112 // 113 // Capacity permitting, it reuses the posted buffer slice to store the buffers 114 // that were read as well. 115 // 116 // This function will block if there aren't any available packets. 117 func (r *rx) postAndReceive(b []queue.RxBuffer, stopRequested *uint32) ([]queue.RxBuffer, uint32) { 118 // Post the buffers first. If we cannot post, sleep until we can. We 119 // never post more than will fit concurrently, so it's safe to wait 120 // until enough room is available. 121 if len(b) != 0 && !r.q.PostBuffers(b) { 122 r.q.EnableNotification() 123 for !r.q.PostBuffers(b) { 124 var tmp [8]byte 125 rawfile.BlockingRead(r.eventFD, tmp[:]) 126 if atomic.LoadUint32(stopRequested) != 0 { 127 r.q.DisableNotification() 128 return nil, 0 129 } 130 } 131 r.q.DisableNotification() 132 } 133 134 // Read the next set of descriptors. 135 b, n := r.q.Dequeue(b[:0]) 136 if len(b) != 0 { 137 return b, n 138 } 139 140 // Data isn't immediately available. Enable eventfd notifications. 141 r.q.EnableNotification() 142 for { 143 b, n = r.q.Dequeue(b) 144 if len(b) != 0 { 145 break 146 } 147 148 // Wait for notification. 149 var tmp [8]byte 150 rawfile.BlockingRead(r.eventFD, tmp[:]) 151 if atomic.LoadUint32(stopRequested) != 0 { 152 r.q.DisableNotification() 153 return nil, 0 154 } 155 } 156 r.q.DisableNotification() 157 158 return b, n 159 }