Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 1 | // Copyright 2009 The Go Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
Joel Sing | 9ca57a7 | 2011-12-21 21:44:47 +1100 | [diff] [blame] | 5 | // +build darwin freebsd linux netbsd openbsd |
Russ Cox | 2715956 | 2011-09-15 16:48:57 -0400 | [diff] [blame] | 6 | |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 7 | package net |
| 8 | |
| 9 | import ( |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 10 | "io" |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 11 | "os" |
| 12 | "sync" |
| 13 | "syscall" |
Russ Cox | 00f9f0c | 2010-03-30 10:34:57 -0700 | [diff] [blame] | 14 | "time" |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 15 | ) |
| 16 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 17 | // Network file descriptor. |
| 18 | type netFD struct { |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 19 | // locking/lifetime of sysfd |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 20 | sysmu sync.Mutex |
| 21 | sysref int |
| 22 | closing bool |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 23 | |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 24 | // immutable until Close |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 25 | sysfd int |
| 26 | family int |
Mikio Hara | 743c2d0 | 2012-01-20 07:31:13 +0900 | [diff] [blame^] | 27 | sotype int |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 28 | sysfile *os.File |
Russ Cox | caa149f | 2010-04-06 16:50:27 -0700 | [diff] [blame] | 29 | cr chan bool |
| 30 | cw chan bool |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 31 | net string |
| 32 | laddr Addr |
| 33 | raddr Addr |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 34 | |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 35 | // owned by client |
Brad Fitzpatrick | b71883e | 2012-01-18 16:24:06 -0800 | [diff] [blame] | 36 | rdeadline int64 |
| 37 | rio sync.Mutex |
| 38 | wdeadline int64 |
| 39 | wio sync.Mutex |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 40 | |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 41 | // owned by fd wait server |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 42 | ncr, ncw int |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 43 | } |
| 44 | |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 45 | type InvalidConnError struct{} |
| 46 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 47 | func (e *InvalidConnError) Error() string { return "invalid net.Conn" } |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 48 | func (e *InvalidConnError) Temporary() bool { return false } |
| 49 | func (e *InvalidConnError) Timeout() bool { return false } |
| 50 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 51 | // A pollServer helps FDs determine when to retry a non-blocking |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 52 | // read or write after they get EAGAIN. When an FD needs to wait, |
| 53 | // send the fd on s.cr (for a read) or s.cw (for a write) to pass the |
| 54 | // request to the poll server. Then receive on fd.cr/fd.cw. |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 55 | // When the pollServer finds that i/o on FD should be possible |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 56 | // again, it will send fd on fd.cr/fd.cw to wake any waiting processes. |
| 57 | // This protocol is implemented as s.WaitRead() and s.WaitWrite(). |
| 58 | // |
| 59 | // There is one subtlety: when sending on s.cr/s.cw, the |
| 60 | // poll server is probably in a system call, waiting for an fd |
| 61 | // to become ready. It's not looking at the request channels. |
| 62 | // To resolve this, the poll server waits not just on the FDs it has |
| 63 | // been given but also its own pipe. After sending on the |
| 64 | // buffered channel s.cr/s.cw, WaitRead/WaitWrite writes a |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 65 | // byte to the pipe, causing the pollServer's poll system call to |
| 66 | // return. In response to the pipe being readable, the pollServer |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 67 | // re-polls its request channels. |
| 68 | // |
| 69 | // Note that the ordering is "send request" and then "wake up server". |
| 70 | // If the operations were reversed, there would be a race: the poll |
| 71 | // server might wake up and look at the request channel, see that it |
| 72 | // was empty, and go back to sleep, all before the requester managed |
| 73 | // to send the request. Because the send must complete before the wakeup, |
| 74 | // the request channel must be buffered. A buffer of size 1 is sufficient |
| 75 | // for any request load. If many processes are trying to submit requests, |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 76 | // one will succeed, the pollServer will read the request, and then the |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 77 | // channel will be empty for the next process's request. A larger buffer |
| 78 | // might help batch requests. |
Adam Langley | ef8f483 | 2009-11-18 13:18:34 -0800 | [diff] [blame] | 79 | // |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 80 | // To avoid races in closing, all fd operations are locked and |
| 81 | // refcounted. when netFD.Close() is called, it calls syscall.Shutdown |
| 82 | // and sets a closing flag. Only when the last reference is removed |
| 83 | // will the fd be closed. |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 84 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 85 | type pollServer struct { |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 86 | cr, cw chan *netFD // buffered >= 1 |
| 87 | pr, pw *os.File |
| 88 | poll *pollster // low-level OS hooks |
| 89 | sync.Mutex // controls pending and deadline |
| 90 | pending map[int]*netFD |
| 91 | deadline int64 // next deadline (nsec since 1970) |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 92 | } |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 93 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 94 | func (s *pollServer) AddFD(fd *netFD, mode int) { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 95 | intfd := fd.sysfd |
Russ Cox | 37a5374 | 2009-04-15 19:01:48 -0700 | [diff] [blame] | 96 | if intfd < 0 { |
| 97 | // fd closed underfoot |
| 98 | if mode == 'r' { |
Russ Cox | caa149f | 2010-04-06 16:50:27 -0700 | [diff] [blame] | 99 | fd.cr <- true |
Russ Cox | 37a5374 | 2009-04-15 19:01:48 -0700 | [diff] [blame] | 100 | } else { |
Russ Cox | caa149f | 2010-04-06 16:50:27 -0700 | [diff] [blame] | 101 | fd.cw <- true |
Russ Cox | 37a5374 | 2009-04-15 19:01:48 -0700 | [diff] [blame] | 102 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 103 | return |
Russ Cox | 37a5374 | 2009-04-15 19:01:48 -0700 | [diff] [blame] | 104 | } |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 105 | |
| 106 | s.Lock() |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 107 | |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 108 | var t int64 |
| 109 | key := intfd << 1 |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 110 | if mode == 'r' { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 111 | fd.ncr++ |
| 112 | t = fd.rdeadline |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 113 | } else { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 114 | fd.ncw++ |
| 115 | key++ |
| 116 | t = fd.wdeadline |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 117 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 118 | s.pending[key] = fd |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 119 | doWakeup := false |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 120 | if t > 0 && (s.deadline == 0 || t < s.deadline) { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 121 | s.deadline = t |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 122 | doWakeup = true |
| 123 | } |
| 124 | |
Ian Lance Taylor | 0caa0c0 | 2011-03-28 12:39:09 -0700 | [diff] [blame] | 125 | wake, err := s.poll.AddFD(intfd, mode, false) |
| 126 | if err != nil { |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 127 | panic("pollServer AddFD " + err.Error()) |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 128 | } |
Ian Lance Taylor | 0caa0c0 | 2011-03-28 12:39:09 -0700 | [diff] [blame] | 129 | if wake { |
| 130 | doWakeup = true |
| 131 | } |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 132 | |
| 133 | s.Unlock() |
| 134 | |
| 135 | if doWakeup { |
| 136 | s.Wakeup() |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 137 | } |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 138 | } |
| 139 | |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 140 | var wakeupbuf [1]byte |
| 141 | |
| 142 | func (s *pollServer) Wakeup() { s.pw.Write(wakeupbuf[0:]) } |
| 143 | |
Russ Cox | 9e0fec9 | 2009-06-01 22:14:39 -0700 | [diff] [blame] | 144 | func (s *pollServer) LookupFD(fd int, mode int) *netFD { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 145 | key := fd << 1 |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 146 | if mode == 'w' { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 147 | key++ |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 148 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 149 | netfd, ok := s.pending[key] |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 150 | if !ok { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 151 | return nil |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 152 | } |
Russ Cox | 313c822 | 2011-10-18 09:56:34 -0400 | [diff] [blame] | 153 | delete(s.pending, key) |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 154 | return netfd |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 155 | } |
| 156 | |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 157 | func (s *pollServer) WakeFD(fd *netFD, mode int) { |
| 158 | if mode == 'r' { |
| 159 | for fd.ncr > 0 { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 160 | fd.ncr-- |
Russ Cox | caa149f | 2010-04-06 16:50:27 -0700 | [diff] [blame] | 161 | fd.cr <- true |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 162 | } |
| 163 | } else { |
| 164 | for fd.ncw > 0 { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 165 | fd.ncw-- |
Russ Cox | caa149f | 2010-04-06 16:50:27 -0700 | [diff] [blame] | 166 | fd.cw <- true |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 167 | } |
| 168 | } |
| 169 | } |
| 170 | |
| 171 | func (s *pollServer) Now() int64 { |
Russ Cox | 03823b8 | 2011-11-30 12:01:46 -0500 | [diff] [blame] | 172 | return time.Now().UnixNano() |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 173 | } |
| 174 | |
| 175 | func (s *pollServer) CheckDeadlines() { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 176 | now := s.Now() |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 177 | // TODO(rsc): This will need to be handled more efficiently, |
| 178 | // probably with a heap indexed by wakeup time. |
| 179 | |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 180 | var next_deadline int64 |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 181 | for key, fd := range s.pending { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 182 | var t int64 |
| 183 | var mode int |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 184 | if key&1 == 0 { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 185 | mode = 'r' |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 186 | } else { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 187 | mode = 'w' |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 188 | } |
| 189 | if mode == 'r' { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 190 | t = fd.rdeadline |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 191 | } else { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 192 | t = fd.wdeadline |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 193 | } |
| 194 | if t > 0 { |
| 195 | if t <= now { |
Russ Cox | 313c822 | 2011-10-18 09:56:34 -0400 | [diff] [blame] | 196 | delete(s.pending, key) |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 197 | if mode == 'r' { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 198 | s.poll.DelFD(fd.sysfd, mode) |
| 199 | fd.rdeadline = -1 |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 200 | } else { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 201 | s.poll.DelFD(fd.sysfd, mode) |
| 202 | fd.wdeadline = -1 |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 203 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 204 | s.WakeFD(fd, mode) |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 205 | } else if next_deadline == 0 || t < next_deadline { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 206 | next_deadline = t |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 207 | } |
| 208 | } |
| 209 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 210 | s.deadline = next_deadline |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 211 | } |
| 212 | |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 213 | func (s *pollServer) Run() { |
| 214 | var scratch [100]byte |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 215 | s.Lock() |
| 216 | defer s.Unlock() |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 217 | for { |
| 218 | var t = s.deadline |
| 219 | if t > 0 { |
| 220 | t = t - s.Now() |
| 221 | if t <= 0 { |
| 222 | s.CheckDeadlines() |
| 223 | continue |
| 224 | } |
| 225 | } |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 226 | fd, mode, err := s.poll.WaitFD(s, t) |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 227 | if err != nil { |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 228 | print("pollServer WaitFD: ", err.Error(), "\n") |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 229 | return |
| 230 | } |
| 231 | if fd < 0 { |
| 232 | // Timeout happened. |
| 233 | s.CheckDeadlines() |
| 234 | continue |
| 235 | } |
| 236 | if fd == s.pr.Fd() { |
Ian Lance Taylor | c01238a | 2011-03-11 18:01:28 -0800 | [diff] [blame] | 237 | // Drain our wakeup pipe (we could loop here, |
| 238 | // but it's unlikely that there are more than |
| 239 | // len(scratch) wakeup calls). |
| 240 | s.pr.Read(scratch[0:]) |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 241 | s.CheckDeadlines() |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 242 | } else { |
| 243 | netfd := s.LookupFD(fd, mode) |
| 244 | if netfd == nil { |
Vinu Rajashekhar | a9a24d4 | 2010-07-10 14:40:48 -0700 | [diff] [blame] | 245 | print("pollServer: unexpected wakeup for fd=", fd, " mode=", string(mode), "\n") |
Vinu Rajashekhar | 00ad47f | 2010-06-29 12:04:04 -0700 | [diff] [blame] | 246 | continue |
| 247 | } |
| 248 | s.WakeFD(netfd, mode) |
| 249 | } |
| 250 | } |
| 251 | } |
| 252 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 253 | func (s *pollServer) WaitRead(fd *netFD) { |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 254 | s.AddFD(fd, 'r') |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 255 | <-fd.cr |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 256 | } |
| 257 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 258 | func (s *pollServer) WaitWrite(fd *netFD) { |
Ian Lance Taylor | 42bc7fc | 2011-03-17 13:42:40 -0700 | [diff] [blame] | 259 | s.AddFD(fd, 'w') |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 260 | <-fd.cw |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 261 | } |
| 262 | |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 263 | // Network FD methods. |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 264 | // All the network FDs use a single pollServer. |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 265 | |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 266 | var pollserver *pollServer |
Rob Pike | c78be46 | 2010-08-06 06:14:41 +1000 | [diff] [blame] | 267 | var onceStartServer sync.Once |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 268 | |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 269 | func startServer() { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 270 | p, err := newPollServer() |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 271 | if err != nil { |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 272 | print("Start pollServer: ", err.Error(), "\n") |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 273 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 274 | pollserver = p |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 275 | } |
| 276 | |
Mikio Hara | 743c2d0 | 2012-01-20 07:31:13 +0900 | [diff] [blame^] | 277 | func newFD(fd, family, sotype int, net string) (f *netFD, err error) { |
Rob Pike | c78be46 | 2010-08-06 06:14:41 +1000 | [diff] [blame] | 278 | onceStartServer.Do(startServer) |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 279 | if e := syscall.SetNonblock(fd, true); e != nil { |
| 280 | return nil, e |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 281 | } |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 282 | f = &netFD{ |
Robert Griesemer | f44fa9b | 2010-03-02 13:46:51 -0800 | [diff] [blame] | 283 | sysfd: fd, |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 284 | family: family, |
Mikio Hara | 743c2d0 | 2012-01-20 07:31:13 +0900 | [diff] [blame^] | 285 | sotype: sotype, |
Robert Griesemer | f44fa9b | 2010-03-02 13:46:51 -0800 | [diff] [blame] | 286 | net: net, |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 287 | } |
Alexey Borzenkov | 2f45f72 | 2011-03-28 23:40:01 -0400 | [diff] [blame] | 288 | f.cr = make(chan bool, 1) |
| 289 | f.cw = make(chan bool, 1) |
| 290 | return f, nil |
| 291 | } |
| 292 | |
| 293 | func (fd *netFD) setAddr(laddr, raddr Addr) { |
| 294 | fd.laddr = laddr |
| 295 | fd.raddr = raddr |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 296 | var ls, rs string |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 297 | if laddr != nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 298 | ls = laddr.String() |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 299 | } |
| 300 | if raddr != nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 301 | rs = raddr.String() |
Russ Cox | c83b838 | 2009-11-02 18:37:30 -0800 | [diff] [blame] | 302 | } |
Alexey Borzenkov | 2f45f72 | 2011-03-28 23:40:01 -0400 | [diff] [blame] | 303 | fd.sysfile = os.NewFile(fd.sysfd, fd.net+":"+ls+"->"+rs) |
| 304 | } |
| 305 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 306 | func (fd *netFD) connect(ra syscall.Sockaddr) (err error) { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 307 | err = syscall.Connect(fd.sysfd, ra) |
| 308 | if err == syscall.EINPROGRESS { |
Alexey Borzenkov | 0793176 | 2011-03-29 14:23:42 -0400 | [diff] [blame] | 309 | pollserver.WaitWrite(fd) |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 310 | var e int |
| 311 | e, err = syscall.GetsockoptInt(fd.sysfd, syscall.SOL_SOCKET, syscall.SO_ERROR) |
| 312 | if err != nil { |
| 313 | return os.NewSyscallError("getsockopt", err) |
| 314 | } |
| 315 | if e != 0 { |
| 316 | err = syscall.Errno(e) |
Alexey Borzenkov | 2f45f72 | 2011-03-28 23:40:01 -0400 | [diff] [blame] | 317 | } |
| 318 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 319 | return err |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 320 | } |
| 321 | |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 322 | // Add a reference to this fd. |
| 323 | func (fd *netFD) incref() { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 324 | fd.sysmu.Lock() |
| 325 | fd.sysref++ |
| 326 | fd.sysmu.Unlock() |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 327 | } |
| 328 | |
| 329 | // Remove a reference to this FD and close if we've been asked to do so (and |
| 330 | // there are no references left. |
| 331 | func (fd *netFD) decref() { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 332 | fd.sysmu.Lock() |
| 333 | fd.sysref-- |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 334 | if fd.closing && fd.sysref == 0 && fd.sysfd >= 0 { |
| 335 | // In case the user has set linger, switch to blocking mode so |
| 336 | // the close blocks. As long as this doesn't happen often, we |
| 337 | // can handle the extra OS processes. Otherwise we'll need to |
| 338 | // use the pollserver for Close too. Sigh. |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 339 | syscall.SetNonblock(fd.sysfd, false) |
| 340 | fd.sysfile.Close() |
| 341 | fd.sysfile = nil |
| 342 | fd.sysfd = -1 |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 343 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 344 | fd.sysmu.Unlock() |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 345 | } |
| 346 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 347 | func (fd *netFD) Close() error { |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 348 | if fd == nil || fd.sysfile == nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 349 | return os.EINVAL |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 350 | } |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 351 | |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 352 | fd.incref() |
| 353 | syscall.Shutdown(fd.sysfd, syscall.SHUT_RDWR) |
| 354 | fd.closing = true |
| 355 | fd.decref() |
| 356 | return nil |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 357 | } |
| 358 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 359 | func (fd *netFD) shutdown(how int) error { |
Brad Fitzpatrick | 394842e | 2011-09-28 08:12:38 -0700 | [diff] [blame] | 360 | if fd == nil || fd.sysfile == nil { |
| 361 | return os.EINVAL |
| 362 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 363 | err := syscall.Shutdown(fd.sysfd, how) |
| 364 | if err != nil { |
| 365 | return &OpError{"shutdown", fd.net, fd.laddr, err} |
Albert Strasheim | 791b2a4 | 2011-10-12 13:45:25 -0400 | [diff] [blame] | 366 | } |
Brad Fitzpatrick | 394842e | 2011-09-28 08:12:38 -0700 | [diff] [blame] | 367 | return nil |
| 368 | } |
| 369 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 370 | func (fd *netFD) CloseRead() error { |
Albert Strasheim | 791b2a4 | 2011-10-12 13:45:25 -0400 | [diff] [blame] | 371 | return fd.shutdown(syscall.SHUT_RD) |
| 372 | } |
| 373 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 374 | func (fd *netFD) CloseWrite() error { |
Albert Strasheim | 791b2a4 | 2011-10-12 13:45:25 -0400 | [diff] [blame] | 375 | return fd.shutdown(syscall.SHUT_WR) |
Brad Fitzpatrick | 394842e | 2011-09-28 08:12:38 -0700 | [diff] [blame] | 376 | } |
| 377 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 378 | func (fd *netFD) Read(p []byte) (n int, err error) { |
Michael Hoisie | 9192ec2 | 2010-05-20 17:13:50 -0700 | [diff] [blame] | 379 | if fd == nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 380 | return 0, os.EINVAL |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 381 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 382 | fd.rio.Lock() |
| 383 | defer fd.rio.Unlock() |
| 384 | fd.incref() |
| 385 | defer fd.decref() |
Michael Hoisie | 9192ec2 | 2010-05-20 17:13:50 -0700 | [diff] [blame] | 386 | if fd.sysfile == nil { |
| 387 | return 0, os.EINVAL |
| 388 | } |
Russ Cox | a0bcaf4 | 2009-06-25 20:24:55 -0700 | [diff] [blame] | 389 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 390 | n, err = syscall.Read(fd.sysfile.Fd(), p) |
| 391 | if err == syscall.EAGAIN { |
| 392 | if fd.rdeadline >= 0 { |
| 393 | pollserver.WaitRead(fd) |
| 394 | continue |
| 395 | } |
| 396 | err = errTimeout |
Russ Cox | a0bcaf4 | 2009-06-25 20:24:55 -0700 | [diff] [blame] | 397 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 398 | if err != nil { |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 399 | n = 0 |
Mikio Hara | 743c2d0 | 2012-01-20 07:31:13 +0900 | [diff] [blame^] | 400 | } else if n == 0 && err == nil && fd.sotype != syscall.SOCK_DGRAM { |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 401 | err = io.EOF |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 402 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 403 | break |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 404 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 405 | if err != nil && err != io.EOF { |
| 406 | err = &OpError{"read", fd.net, fd.raddr, err} |
Russ Cox | e89441b | 2009-12-02 15:17:49 -0800 | [diff] [blame] | 407 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 408 | return |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 409 | } |
| 410 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 411 | func (fd *netFD) ReadFrom(p []byte) (n int, sa syscall.Sockaddr, err error) { |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 412 | if fd == nil || fd.sysfile == nil { |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 413 | return 0, nil, os.EINVAL |
| 414 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 415 | fd.rio.Lock() |
| 416 | defer fd.rio.Unlock() |
| 417 | fd.incref() |
| 418 | defer fd.decref() |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 419 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 420 | n, sa, err = syscall.Recvfrom(fd.sysfd, p, 0) |
| 421 | if err == syscall.EAGAIN { |
| 422 | if fd.rdeadline >= 0 { |
| 423 | pollserver.WaitRead(fd) |
| 424 | continue |
| 425 | } |
| 426 | err = errTimeout |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 427 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 428 | if err != nil { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 429 | n = 0 |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 430 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 431 | break |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 432 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 433 | if err != nil { |
| 434 | err = &OpError{"read", fd.net, fd.laddr, err} |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 435 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 436 | return |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 437 | } |
| 438 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 439 | func (fd *netFD) ReadMsg(p []byte, oob []byte) (n, oobn, flags int, sa syscall.Sockaddr, err error) { |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 440 | if fd == nil || fd.sysfile == nil { |
| 441 | return 0, 0, 0, nil, os.EINVAL |
| 442 | } |
| 443 | fd.rio.Lock() |
| 444 | defer fd.rio.Unlock() |
| 445 | fd.incref() |
| 446 | defer fd.decref() |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 447 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 448 | n, oobn, flags, sa, err = syscall.Recvmsg(fd.sysfd, p, oob, 0) |
| 449 | if err == syscall.EAGAIN { |
| 450 | if fd.rdeadline >= 0 { |
| 451 | pollserver.WaitRead(fd) |
| 452 | continue |
| 453 | } |
| 454 | err = errTimeout |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 455 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 456 | if err == nil && n == 0 { |
| 457 | err = io.EOF |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 458 | } |
| 459 | break |
| 460 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 461 | if err != nil && err != io.EOF { |
| 462 | err = &OpError{"read", fd.net, fd.laddr, err} |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 463 | return |
| 464 | } |
| 465 | return |
| 466 | } |
| 467 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 468 | func (fd *netFD) Write(p []byte) (n int, err error) { |
Michael Hoisie | 9192ec2 | 2010-05-20 17:13:50 -0700 | [diff] [blame] | 469 | if fd == nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 470 | return 0, os.EINVAL |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 471 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 472 | fd.wio.Lock() |
| 473 | defer fd.wio.Unlock() |
| 474 | fd.incref() |
| 475 | defer fd.decref() |
Michael Hoisie | 9192ec2 | 2010-05-20 17:13:50 -0700 | [diff] [blame] | 476 | if fd.sysfile == nil { |
| 477 | return 0, os.EINVAL |
| 478 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 479 | nn := 0 |
Michael Hoisie | 9192ec2 | 2010-05-20 17:13:50 -0700 | [diff] [blame] | 480 | |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 481 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 482 | var n int |
| 483 | n, err = syscall.Write(fd.sysfile.Fd(), p[nn:]) |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 484 | if n > 0 { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 485 | nn += n |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 486 | } |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 487 | if nn == len(p) { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 488 | break |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 489 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 490 | if err == syscall.EAGAIN { |
| 491 | if fd.wdeadline >= 0 { |
| 492 | pollserver.WaitWrite(fd) |
| 493 | continue |
| 494 | } |
| 495 | err = errTimeout |
Russ Cox | 1e37e8a | 2009-03-06 17:51:31 -0800 | [diff] [blame] | 496 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 497 | if err != nil { |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 498 | n = 0 |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 499 | break |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 500 | } |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 501 | if n == 0 { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 502 | err = io.ErrUnexpectedEOF |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 503 | break |
| 504 | } |
| 505 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 506 | if err != nil { |
| 507 | err = &OpError{"write", fd.net, fd.raddr, err} |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 508 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 509 | return nn, err |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 510 | } |
| 511 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 512 | func (fd *netFD) WriteTo(p []byte, sa syscall.Sockaddr) (n int, err error) { |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 513 | if fd == nil || fd.sysfile == nil { |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 514 | return 0, os.EINVAL |
| 515 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 516 | fd.wio.Lock() |
| 517 | defer fd.wio.Unlock() |
| 518 | fd.incref() |
| 519 | defer fd.decref() |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 520 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 521 | err = syscall.Sendto(fd.sysfd, p, 0, sa) |
| 522 | if err == syscall.EAGAIN { |
| 523 | if fd.wdeadline >= 0 { |
| 524 | pollserver.WaitWrite(fd) |
| 525 | continue |
| 526 | } |
| 527 | err = errTimeout |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 528 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 529 | break |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 530 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 531 | if err == nil { |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 532 | n = len(p) |
Russ Cox | 47a0533 | 2010-04-26 22:15:25 -0700 | [diff] [blame] | 533 | } else { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 534 | err = &OpError{"write", fd.net, fd.raddr, err} |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 535 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 536 | return |
Russ Cox | 6e788e0 | 2009-11-17 08:39:17 -0800 | [diff] [blame] | 537 | } |
| 538 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 539 | func (fd *netFD) WriteMsg(p []byte, oob []byte, sa syscall.Sockaddr) (n int, oobn int, err error) { |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 540 | if fd == nil || fd.sysfile == nil { |
| 541 | return 0, 0, os.EINVAL |
| 542 | } |
| 543 | fd.wio.Lock() |
| 544 | defer fd.wio.Unlock() |
| 545 | fd.incref() |
| 546 | defer fd.decref() |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 547 | for { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 548 | err = syscall.Sendmsg(fd.sysfd, p, oob, sa, 0) |
| 549 | if err == syscall.EAGAIN { |
| 550 | if fd.wdeadline >= 0 { |
| 551 | pollserver.WaitWrite(fd) |
| 552 | continue |
| 553 | } |
| 554 | err = errTimeout |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 555 | } |
| 556 | break |
| 557 | } |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 558 | if err == nil { |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 559 | n = len(p) |
| 560 | oobn = len(oob) |
| 561 | } else { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 562 | err = &OpError{"write", fd.net, fd.raddr, err} |
Albert Strasheim | cf6c212 | 2010-12-07 13:40:14 -0500 | [diff] [blame] | 563 | } |
| 564 | return |
| 565 | } |
| 566 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 567 | func (fd *netFD) accept(toAddr func(syscall.Sockaddr) Addr) (nfd *netFD, err error) { |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 568 | if fd == nil || fd.sysfile == nil { |
Robert Griesemer | 40621d5 | 2009-11-09 12:07:39 -0800 | [diff] [blame] | 569 | return nil, os.EINVAL |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 570 | } |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 571 | |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 572 | fd.incref() |
| 573 | defer fd.decref() |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 574 | |
Russ Cox | 91ceda5 | 2009-02-15 19:35:52 -0800 | [diff] [blame] | 575 | // See ../syscall/exec.go for description of ForkLock. |
| 576 | // It is okay to hold the lock across syscall.Accept |
Devon H. O'Dell | eb16346 | 2009-12-01 23:28:57 -0800 | [diff] [blame] | 577 | // because we have put fd.sysfd into non-blocking mode. |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 578 | var s int |
Mikio Hara | c20ced9 | 2011-08-16 16:53:09 -0400 | [diff] [blame] | 579 | var rsa syscall.Sockaddr |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 580 | for { |
Michael Hoisie | ed1cbca | 2011-01-04 11:55:13 -0500 | [diff] [blame] | 581 | if fd.closing { |
Michael Hoisie | ed1cbca | 2011-01-04 11:55:13 -0500 | [diff] [blame] | 582 | return nil, os.EINVAL |
| 583 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 584 | syscall.ForkLock.RLock() |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 585 | s, rsa, err = syscall.Accept(fd.sysfd) |
| 586 | if err != nil { |
| 587 | syscall.ForkLock.RUnlock() |
| 588 | if err == syscall.EAGAIN { |
| 589 | if fd.rdeadline >= 0 { |
| 590 | pollserver.WaitRead(fd) |
| 591 | continue |
| 592 | } |
| 593 | err = errTimeout |
| 594 | } |
| 595 | return nil, &OpError{"accept", fd.net, fd.laddr, err} |
| 596 | } |
| 597 | break |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 598 | } |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 599 | syscall.CloseOnExec(s) |
| 600 | syscall.ForkLock.RUnlock() |
Russ Cox | d8921c5 | 2009-02-15 14:18:39 -0800 | [diff] [blame] | 601 | |
Mikio Hara | 743c2d0 | 2012-01-20 07:31:13 +0900 | [diff] [blame^] | 602 | if nfd, err = newFD(s, fd.family, fd.sotype, fd.net); err != nil { |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 603 | syscall.Close(s) |
| 604 | return nil, err |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 605 | } |
Mikio Hara | c20ced9 | 2011-08-16 16:53:09 -0400 | [diff] [blame] | 606 | lsa, _ := syscall.Getsockname(nfd.sysfd) |
| 607 | nfd.setAddr(toAddr(lsa), toAddr(rsa)) |
Robert Griesemer | a3d1045 | 2009-12-15 15:35:38 -0800 | [diff] [blame] | 608 | return nfd, nil |
Russ Cox | e4a61c6 | 2008-09-29 13:37:00 -0700 | [diff] [blame] | 609 | } |
Keith Rarick | a144e3e | 2010-11-05 14:02:03 -0400 | [diff] [blame] | 610 | |
Russ Cox | eb69292 | 2011-11-01 22:05:34 -0400 | [diff] [blame] | 611 | func (fd *netFD) dup() (f *os.File, err error) { |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 612 | ns, err := syscall.Dup(fd.sysfd) |
| 613 | if err != nil { |
| 614 | return nil, &OpError{"dup", fd.net, fd.laddr, err} |
Keith Rarick | a144e3e | 2010-11-05 14:02:03 -0400 | [diff] [blame] | 615 | } |
| 616 | |
| 617 | // We want blocking mode for the new fd, hence the double negative. |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 618 | if err = syscall.SetNonblock(ns, false); err != nil { |
| 619 | return nil, &OpError{"setnonblock", fd.net, fd.laddr, err} |
Keith Rarick | a144e3e | 2010-11-05 14:02:03 -0400 | [diff] [blame] | 620 | } |
| 621 | |
| 622 | return os.NewFile(ns, fd.sysfile.Name()), nil |
| 623 | } |
Alex Brainman | 3a052b5 | 2011-01-12 15:55:17 +1100 | [diff] [blame] | 624 | |
Russ Cox | c017a82 | 2011-11-13 22:44:52 -0500 | [diff] [blame] | 625 | func closesocket(s int) error { |
Alex Brainman | 3a052b5 | 2011-01-12 15:55:17 +1100 | [diff] [blame] | 626 | return syscall.Close(s) |
| 627 | } |