foolio 0.0.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (261) hide show
  1. data/.gitignore +22 -0
  2. data/Gemfile +4 -0
  3. data/LICENSE +22 -0
  4. data/README.md +29 -0
  5. data/Rakefile +21 -0
  6. data/examples/timer.rb +20 -0
  7. data/ext/foolio/extconf.rb +34 -0
  8. data/ext/foolio/foolio_ext.c +921 -0
  9. data/ext/foolio/gen.rb +50 -0
  10. data/ext/foolio/make_table.rb +12 -0
  11. data/ext/foolio/templ +243 -0
  12. data/ext/libuv/.gitignore +33 -0
  13. data/ext/libuv/.mailmap +13 -0
  14. data/ext/libuv/.travis.yml +9 -0
  15. data/ext/libuv/AUTHORS +61 -0
  16. data/ext/libuv/LICENSE +44 -0
  17. data/ext/libuv/Makefile +71 -0
  18. data/ext/libuv/README.md +90 -0
  19. data/ext/libuv/common.gypi +178 -0
  20. data/ext/libuv/gyp_uv +73 -0
  21. data/ext/libuv/include/uv-private/eio.h +403 -0
  22. data/ext/libuv/include/uv-private/ev.h +838 -0
  23. data/ext/libuv/include/uv-private/ngx-queue.h +108 -0
  24. data/ext/libuv/include/uv-private/tree.h +768 -0
  25. data/ext/libuv/include/uv-private/uv-unix.h +324 -0
  26. data/ext/libuv/include/uv-private/uv-win.h +517 -0
  27. data/ext/libuv/include/uv.h +1838 -0
  28. data/ext/libuv/src/fs-poll.c +235 -0
  29. data/ext/libuv/src/inet.c +293 -0
  30. data/ext/libuv/src/unix/async.c +148 -0
  31. data/ext/libuv/src/unix/core.c +696 -0
  32. data/ext/libuv/src/unix/cygwin.c +83 -0
  33. data/ext/libuv/src/unix/darwin.c +342 -0
  34. data/ext/libuv/src/unix/dl.c +83 -0
  35. data/ext/libuv/src/unix/eio/Changes +63 -0
  36. data/ext/libuv/src/unix/eio/LICENSE +36 -0
  37. data/ext/libuv/src/unix/eio/Makefile.am +15 -0
  38. data/ext/libuv/src/unix/eio/aclocal.m4 +8957 -0
  39. data/ext/libuv/src/unix/eio/autogen.sh +3 -0
  40. data/ext/libuv/src/unix/eio/config.h.in +86 -0
  41. data/ext/libuv/src/unix/eio/config_cygwin.h +80 -0
  42. data/ext/libuv/src/unix/eio/config_darwin.h +141 -0
  43. data/ext/libuv/src/unix/eio/config_freebsd.h +81 -0
  44. data/ext/libuv/src/unix/eio/config_linux.h +94 -0
  45. data/ext/libuv/src/unix/eio/config_netbsd.h +81 -0
  46. data/ext/libuv/src/unix/eio/config_openbsd.h +137 -0
  47. data/ext/libuv/src/unix/eio/config_sunos.h +84 -0
  48. data/ext/libuv/src/unix/eio/configure.ac +22 -0
  49. data/ext/libuv/src/unix/eio/demo.c +194 -0
  50. data/ext/libuv/src/unix/eio/ecb.h +370 -0
  51. data/ext/libuv/src/unix/eio/eio.3 +3428 -0
  52. data/ext/libuv/src/unix/eio/eio.c +2593 -0
  53. data/ext/libuv/src/unix/eio/eio.pod +969 -0
  54. data/ext/libuv/src/unix/eio/libeio.m4 +195 -0
  55. data/ext/libuv/src/unix/eio/xthread.h +164 -0
  56. data/ext/libuv/src/unix/error.c +105 -0
  57. data/ext/libuv/src/unix/ev/Changes +388 -0
  58. data/ext/libuv/src/unix/ev/LICENSE +36 -0
  59. data/ext/libuv/src/unix/ev/Makefile.am +18 -0
  60. data/ext/libuv/src/unix/ev/Makefile.in +771 -0
  61. data/ext/libuv/src/unix/ev/README +58 -0
  62. data/ext/libuv/src/unix/ev/aclocal.m4 +8957 -0
  63. data/ext/libuv/src/unix/ev/autogen.sh +6 -0
  64. data/ext/libuv/src/unix/ev/config.guess +1526 -0
  65. data/ext/libuv/src/unix/ev/config.h.in +125 -0
  66. data/ext/libuv/src/unix/ev/config.sub +1658 -0
  67. data/ext/libuv/src/unix/ev/config_cygwin.h +123 -0
  68. data/ext/libuv/src/unix/ev/config_darwin.h +122 -0
  69. data/ext/libuv/src/unix/ev/config_freebsd.h +120 -0
  70. data/ext/libuv/src/unix/ev/config_linux.h +141 -0
  71. data/ext/libuv/src/unix/ev/config_netbsd.h +120 -0
  72. data/ext/libuv/src/unix/ev/config_openbsd.h +126 -0
  73. data/ext/libuv/src/unix/ev/config_sunos.h +122 -0
  74. data/ext/libuv/src/unix/ev/configure +13037 -0
  75. data/ext/libuv/src/unix/ev/configure.ac +18 -0
  76. data/ext/libuv/src/unix/ev/depcomp +630 -0
  77. data/ext/libuv/src/unix/ev/ev++.h +816 -0
  78. data/ext/libuv/src/unix/ev/ev.3 +5311 -0
  79. data/ext/libuv/src/unix/ev/ev.c +3925 -0
  80. data/ext/libuv/src/unix/ev/ev.pod +5243 -0
  81. data/ext/libuv/src/unix/ev/ev_epoll.c +266 -0
  82. data/ext/libuv/src/unix/ev/ev_kqueue.c +235 -0
  83. data/ext/libuv/src/unix/ev/ev_poll.c +148 -0
  84. data/ext/libuv/src/unix/ev/ev_port.c +179 -0
  85. data/ext/libuv/src/unix/ev/ev_select.c +310 -0
  86. data/ext/libuv/src/unix/ev/ev_vars.h +203 -0
  87. data/ext/libuv/src/unix/ev/ev_win32.c +153 -0
  88. data/ext/libuv/src/unix/ev/ev_wrap.h +196 -0
  89. data/ext/libuv/src/unix/ev/event.c +402 -0
  90. data/ext/libuv/src/unix/ev/event.h +170 -0
  91. data/ext/libuv/src/unix/ev/install-sh +294 -0
  92. data/ext/libuv/src/unix/ev/libev.m4 +39 -0
  93. data/ext/libuv/src/unix/ev/ltmain.sh +8413 -0
  94. data/ext/libuv/src/unix/ev/missing +336 -0
  95. data/ext/libuv/src/unix/ev/mkinstalldirs +111 -0
  96. data/ext/libuv/src/unix/freebsd.c +326 -0
  97. data/ext/libuv/src/unix/fs.c +739 -0
  98. data/ext/libuv/src/unix/internal.h +188 -0
  99. data/ext/libuv/src/unix/kqueue.c +120 -0
  100. data/ext/libuv/src/unix/linux/inotify.c +239 -0
  101. data/ext/libuv/src/unix/linux/linux-core.c +557 -0
  102. data/ext/libuv/src/unix/linux/syscalls.c +388 -0
  103. data/ext/libuv/src/unix/linux/syscalls.h +124 -0
  104. data/ext/libuv/src/unix/loop-watcher.c +62 -0
  105. data/ext/libuv/src/unix/loop.c +94 -0
  106. data/ext/libuv/src/unix/netbsd.c +108 -0
  107. data/ext/libuv/src/unix/openbsd.c +295 -0
  108. data/ext/libuv/src/unix/pipe.c +259 -0
  109. data/ext/libuv/src/unix/poll.c +114 -0
  110. data/ext/libuv/src/unix/process.c +495 -0
  111. data/ext/libuv/src/unix/signal.c +269 -0
  112. data/ext/libuv/src/unix/stream.c +990 -0
  113. data/ext/libuv/src/unix/sunos.c +481 -0
  114. data/ext/libuv/src/unix/tcp.c +393 -0
  115. data/ext/libuv/src/unix/thread.c +251 -0
  116. data/ext/libuv/src/unix/timer.c +136 -0
  117. data/ext/libuv/src/unix/tty.c +145 -0
  118. data/ext/libuv/src/unix/udp.c +659 -0
  119. data/ext/libuv/src/unix/uv-eio.c +107 -0
  120. data/ext/libuv/src/unix/uv-eio.h +13 -0
  121. data/ext/libuv/src/uv-common.c +380 -0
  122. data/ext/libuv/src/uv-common.h +170 -0
  123. data/ext/libuv/src/win/async.c +100 -0
  124. data/ext/libuv/src/win/atomicops-inl.h +56 -0
  125. data/ext/libuv/src/win/core.c +278 -0
  126. data/ext/libuv/src/win/dl.c +86 -0
  127. data/ext/libuv/src/win/error.c +155 -0
  128. data/ext/libuv/src/win/fs-event.c +510 -0
  129. data/ext/libuv/src/win/fs.c +1948 -0
  130. data/ext/libuv/src/win/getaddrinfo.c +365 -0
  131. data/ext/libuv/src/win/handle-inl.h +149 -0
  132. data/ext/libuv/src/win/handle.c +154 -0
  133. data/ext/libuv/src/win/internal.h +343 -0
  134. data/ext/libuv/src/win/loop-watcher.c +122 -0
  135. data/ext/libuv/src/win/pipe.c +1672 -0
  136. data/ext/libuv/src/win/poll.c +616 -0
  137. data/ext/libuv/src/win/process-stdio.c +500 -0
  138. data/ext/libuv/src/win/process.c +1013 -0
  139. data/ext/libuv/src/win/req-inl.h +220 -0
  140. data/ext/libuv/src/win/req.c +25 -0
  141. data/ext/libuv/src/win/signal.c +57 -0
  142. data/ext/libuv/src/win/stream-inl.h +67 -0
  143. data/ext/libuv/src/win/stream.c +167 -0
  144. data/ext/libuv/src/win/tcp.c +1394 -0
  145. data/ext/libuv/src/win/thread.c +372 -0
  146. data/ext/libuv/src/win/threadpool.c +74 -0
  147. data/ext/libuv/src/win/timer.c +224 -0
  148. data/ext/libuv/src/win/tty.c +1799 -0
  149. data/ext/libuv/src/win/udp.c +716 -0
  150. data/ext/libuv/src/win/util.c +864 -0
  151. data/ext/libuv/src/win/winapi.c +132 -0
  152. data/ext/libuv/src/win/winapi.h +4452 -0
  153. data/ext/libuv/src/win/winsock.c +557 -0
  154. data/ext/libuv/src/win/winsock.h +171 -0
  155. data/ext/libuv/test/benchmark-async-pummel.c +97 -0
  156. data/ext/libuv/test/benchmark-async.c +137 -0
  157. data/ext/libuv/test/benchmark-fs-stat.c +135 -0
  158. data/ext/libuv/test/benchmark-getaddrinfo.c +94 -0
  159. data/ext/libuv/test/benchmark-list.h +127 -0
  160. data/ext/libuv/test/benchmark-loop-count.c +88 -0
  161. data/ext/libuv/test/benchmark-million-timers.c +65 -0
  162. data/ext/libuv/test/benchmark-ping-pongs.c +213 -0
  163. data/ext/libuv/test/benchmark-pound.c +324 -0
  164. data/ext/libuv/test/benchmark-pump.c +462 -0
  165. data/ext/libuv/test/benchmark-sizes.c +44 -0
  166. data/ext/libuv/test/benchmark-spawn.c +162 -0
  167. data/ext/libuv/test/benchmark-tcp-write-batch.c +140 -0
  168. data/ext/libuv/test/benchmark-thread.c +64 -0
  169. data/ext/libuv/test/benchmark-udp-packet-storm.c +247 -0
  170. data/ext/libuv/test/blackhole-server.c +118 -0
  171. data/ext/libuv/test/dns-server.c +321 -0
  172. data/ext/libuv/test/echo-server.c +378 -0
  173. data/ext/libuv/test/fixtures/empty_file +0 -0
  174. data/ext/libuv/test/fixtures/load_error.node +1 -0
  175. data/ext/libuv/test/run-benchmarks.c +64 -0
  176. data/ext/libuv/test/run-tests.c +138 -0
  177. data/ext/libuv/test/runner-unix.c +295 -0
  178. data/ext/libuv/test/runner-unix.h +36 -0
  179. data/ext/libuv/test/runner-win.c +285 -0
  180. data/ext/libuv/test/runner-win.h +42 -0
  181. data/ext/libuv/test/runner.c +355 -0
  182. data/ext/libuv/test/runner.h +159 -0
  183. data/ext/libuv/test/task.h +112 -0
  184. data/ext/libuv/test/test-async.c +118 -0
  185. data/ext/libuv/test/test-callback-order.c +76 -0
  186. data/ext/libuv/test/test-callback-stack.c +203 -0
  187. data/ext/libuv/test/test-connection-fail.c +148 -0
  188. data/ext/libuv/test/test-cwd-and-chdir.c +64 -0
  189. data/ext/libuv/test/test-delayed-accept.c +188 -0
  190. data/ext/libuv/test/test-dlerror.c +58 -0
  191. data/ext/libuv/test/test-error.c +59 -0
  192. data/ext/libuv/test/test-fail-always.c +29 -0
  193. data/ext/libuv/test/test-fs-event.c +474 -0
  194. data/ext/libuv/test/test-fs-poll.c +146 -0
  195. data/ext/libuv/test/test-fs.c +1843 -0
  196. data/ext/libuv/test/test-get-currentexe.c +63 -0
  197. data/ext/libuv/test/test-get-loadavg.c +36 -0
  198. data/ext/libuv/test/test-get-memory.c +38 -0
  199. data/ext/libuv/test/test-getaddrinfo.c +122 -0
  200. data/ext/libuv/test/test-getsockname.c +342 -0
  201. data/ext/libuv/test/test-hrtime.c +54 -0
  202. data/ext/libuv/test/test-idle.c +81 -0
  203. data/ext/libuv/test/test-ipc-send-recv.c +209 -0
  204. data/ext/libuv/test/test-ipc.c +620 -0
  205. data/ext/libuv/test/test-list.h +427 -0
  206. data/ext/libuv/test/test-loop-handles.c +336 -0
  207. data/ext/libuv/test/test-multiple-listen.c +102 -0
  208. data/ext/libuv/test/test-mutexes.c +63 -0
  209. data/ext/libuv/test/test-pass-always.c +28 -0
  210. data/ext/libuv/test/test-ping-pong.c +253 -0
  211. data/ext/libuv/test/test-pipe-bind-error.c +140 -0
  212. data/ext/libuv/test/test-pipe-connect-error.c +96 -0
  213. data/ext/libuv/test/test-platform-output.c +87 -0
  214. data/ext/libuv/test/test-poll-close.c +72 -0
  215. data/ext/libuv/test/test-poll.c +573 -0
  216. data/ext/libuv/test/test-process-title.c +49 -0
  217. data/ext/libuv/test/test-ref.c +338 -0
  218. data/ext/libuv/test/test-run-once.c +48 -0
  219. data/ext/libuv/test/test-semaphore.c +111 -0
  220. data/ext/libuv/test/test-shutdown-close.c +103 -0
  221. data/ext/libuv/test/test-shutdown-eof.c +183 -0
  222. data/ext/libuv/test/test-signal.c +162 -0
  223. data/ext/libuv/test/test-spawn.c +863 -0
  224. data/ext/libuv/test/test-stdio-over-pipes.c +246 -0
  225. data/ext/libuv/test/test-tcp-bind-error.c +191 -0
  226. data/ext/libuv/test/test-tcp-bind6-error.c +154 -0
  227. data/ext/libuv/test/test-tcp-close-while-connecting.c +80 -0
  228. data/ext/libuv/test/test-tcp-close.c +129 -0
  229. data/ext/libuv/test/test-tcp-connect-error-after-write.c +95 -0
  230. data/ext/libuv/test/test-tcp-connect-error.c +70 -0
  231. data/ext/libuv/test/test-tcp-connect-timeout.c +85 -0
  232. data/ext/libuv/test/test-tcp-connect6-error.c +68 -0
  233. data/ext/libuv/test/test-tcp-flags.c +51 -0
  234. data/ext/libuv/test/test-tcp-shutdown-after-write.c +131 -0
  235. data/ext/libuv/test/test-tcp-unexpected-read.c +113 -0
  236. data/ext/libuv/test/test-tcp-write-error.c +168 -0
  237. data/ext/libuv/test/test-tcp-write-to-half-open-connection.c +135 -0
  238. data/ext/libuv/test/test-tcp-writealot.c +170 -0
  239. data/ext/libuv/test/test-thread.c +183 -0
  240. data/ext/libuv/test/test-threadpool.c +57 -0
  241. data/ext/libuv/test/test-timer-again.c +141 -0
  242. data/ext/libuv/test/test-timer.c +152 -0
  243. data/ext/libuv/test/test-tty.c +110 -0
  244. data/ext/libuv/test/test-udp-dgram-too-big.c +86 -0
  245. data/ext/libuv/test/test-udp-ipv6.c +156 -0
  246. data/ext/libuv/test/test-udp-multicast-join.c +139 -0
  247. data/ext/libuv/test/test-udp-multicast-ttl.c +86 -0
  248. data/ext/libuv/test/test-udp-options.c +86 -0
  249. data/ext/libuv/test/test-udp-send-and-recv.c +208 -0
  250. data/ext/libuv/test/test-util.c +97 -0
  251. data/ext/libuv/test/test-walk-handles.c +77 -0
  252. data/ext/libuv/uv.gyp +375 -0
  253. data/ext/libuv/vcbuild.bat +105 -0
  254. data/foolio.gemspec +18 -0
  255. data/lib/foolio.rb +9 -0
  256. data/lib/foolio/handle.rb +27 -0
  257. data/lib/foolio/listener.rb +26 -0
  258. data/lib/foolio/loop.rb +79 -0
  259. data/lib/foolio/stream.rb +109 -0
  260. data/lib/foolio/version.rb +3 -0
  261. metadata +309 -0
@@ -0,0 +1,557 @@
1
+ /* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
2
+ *
3
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
4
+ * of this software and associated documentation files (the "Software"), to
5
+ * deal in the Software without restriction, including without limitation the
6
+ * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7
+ * sell copies of the Software, and to permit persons to whom the Software is
8
+ * furnished to do so, subject to the following conditions:
9
+ *
10
+ * The above copyright notice and this permission notice shall be included in
11
+ * all copies or substantial portions of the Software.
12
+ *
13
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18
+ * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
19
+ * IN THE SOFTWARE.
20
+ */
21
+
22
+ #include <assert.h>
23
+
24
+ #include "uv.h"
25
+ #include "internal.h"
26
+
27
+
28
+ /* Whether ipv6 is supported */
29
+ int uv_allow_ipv6;
30
+
31
+ /* Whether there are any non-IFS LSPs stacked on TCP */
32
+ int uv_tcp_non_ifs_lsp_ipv4;
33
+ int uv_tcp_non_ifs_lsp_ipv6;
34
+
35
+ /* Ip address used to bind to any port at any interface */
36
+ struct sockaddr_in uv_addr_ip4_any_;
37
+ struct sockaddr_in6 uv_addr_ip6_any_;
38
+
39
+
40
+ /*
41
+ * Retrieves the pointer to a winsock extension function.
42
+ */
43
+ static BOOL uv_get_extension_function(SOCKET socket, GUID guid,
44
+ void **target) {
45
+ DWORD result, bytes;
46
+
47
+ result = WSAIoctl(socket,
48
+ SIO_GET_EXTENSION_FUNCTION_POINTER,
49
+ &guid,
50
+ sizeof(guid),
51
+ (void*)target,
52
+ sizeof(*target),
53
+ &bytes,
54
+ NULL,
55
+ NULL);
56
+
57
+ if (result == SOCKET_ERROR) {
58
+ *target = NULL;
59
+ return FALSE;
60
+ } else {
61
+ return TRUE;
62
+ }
63
+ }
64
+
65
+
66
+ BOOL uv_get_acceptex_function(SOCKET socket, LPFN_ACCEPTEX* target) {
67
+ const GUID wsaid_acceptex = WSAID_ACCEPTEX;
68
+ return uv_get_extension_function(socket, wsaid_acceptex, (void**)target);
69
+ }
70
+
71
+
72
+ BOOL uv_get_connectex_function(SOCKET socket, LPFN_CONNECTEX* target) {
73
+ const GUID wsaid_connectex = WSAID_CONNECTEX;
74
+ return uv_get_extension_function(socket, wsaid_connectex, (void**)target);
75
+ }
76
+
77
+
78
+ void uv_winsock_init() {
79
+ const GUID wsaid_connectex = WSAID_CONNECTEX;
80
+ const GUID wsaid_acceptex = WSAID_ACCEPTEX;
81
+ const GUID wsaid_getacceptexsockaddrs = WSAID_GETACCEPTEXSOCKADDRS;
82
+ const GUID wsaid_disconnectex = WSAID_DISCONNECTEX;
83
+ const GUID wsaid_transmitfile = WSAID_TRANSMITFILE;
84
+
85
+ WSADATA wsa_data;
86
+ int errorno;
87
+ SOCKET dummy;
88
+ WSAPROTOCOL_INFOW protocol_info;
89
+ int opt_len;
90
+
91
+ /* Initialize winsock */
92
+ errorno = WSAStartup(MAKEWORD(2, 2), &wsa_data);
93
+ if (errorno != 0) {
94
+ uv_fatal_error(errorno, "WSAStartup");
95
+ }
96
+
97
+ /* Set implicit binding address used by connectEx */
98
+ uv_addr_ip4_any_ = uv_ip4_addr("0.0.0.0", 0);
99
+ uv_addr_ip6_any_ = uv_ip6_addr("::", 0);
100
+
101
+ /* Detect non-IFS LSPs */
102
+ dummy = socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
103
+ if (dummy == INVALID_SOCKET) {
104
+ uv_fatal_error(WSAGetLastError(), "socket");
105
+ }
106
+
107
+ opt_len = (int) sizeof protocol_info;
108
+ if (!getsockopt(dummy,
109
+ SOL_SOCKET,
110
+ SO_PROTOCOL_INFOW,
111
+ (char*) &protocol_info,
112
+ &opt_len) == SOCKET_ERROR) {
113
+ uv_fatal_error(WSAGetLastError(), "socket");
114
+ }
115
+
116
+ if (!(protocol_info.dwServiceFlags1 & XP1_IFS_HANDLES)) {
117
+ uv_tcp_non_ifs_lsp_ipv4 = 1;
118
+ }
119
+
120
+ if (closesocket(dummy) == SOCKET_ERROR) {
121
+ uv_fatal_error(WSAGetLastError(), "closesocket");
122
+ }
123
+
124
+ /* Detect IPV6 support and non-IFS LSPs */
125
+ dummy = socket(AF_INET6, SOCK_STREAM, IPPROTO_IP);
126
+ if (dummy != INVALID_SOCKET) {
127
+ uv_allow_ipv6 = TRUE;
128
+
129
+ opt_len = (int) sizeof protocol_info;
130
+ if (!getsockopt(dummy,
131
+ SOL_SOCKET,
132
+ SO_PROTOCOL_INFOW,
133
+ (char*) &protocol_info,
134
+ &opt_len) == SOCKET_ERROR) {
135
+ uv_fatal_error(WSAGetLastError(), "socket");
136
+ }
137
+
138
+ if (!(protocol_info.dwServiceFlags1 & XP1_IFS_HANDLES)) {
139
+ uv_tcp_non_ifs_lsp_ipv6 = 1;
140
+ }
141
+
142
+ if (closesocket(dummy) == SOCKET_ERROR) {
143
+ uv_fatal_error(WSAGetLastError(), "closesocket");
144
+ }
145
+ }
146
+ }
147
+
148
+
149
+ int uv_ntstatus_to_winsock_error(NTSTATUS status) {
150
+ switch (status) {
151
+ case STATUS_SUCCESS:
152
+ return ERROR_SUCCESS;
153
+
154
+ case STATUS_PENDING:
155
+ return ERROR_IO_PENDING;
156
+
157
+ case STATUS_INVALID_HANDLE:
158
+ case STATUS_OBJECT_TYPE_MISMATCH:
159
+ return WSAENOTSOCK;
160
+
161
+ case STATUS_INSUFFICIENT_RESOURCES:
162
+ case STATUS_PAGEFILE_QUOTA:
163
+ case STATUS_COMMITMENT_LIMIT:
164
+ case STATUS_WORKING_SET_QUOTA:
165
+ case STATUS_NO_MEMORY:
166
+ case STATUS_CONFLICTING_ADDRESSES:
167
+ case STATUS_QUOTA_EXCEEDED:
168
+ case STATUS_TOO_MANY_PAGING_FILES:
169
+ case STATUS_REMOTE_RESOURCES:
170
+ case STATUS_TOO_MANY_ADDRESSES:
171
+ return WSAENOBUFS;
172
+
173
+ case STATUS_SHARING_VIOLATION:
174
+ case STATUS_ADDRESS_ALREADY_EXISTS:
175
+ return WSAEADDRINUSE;
176
+
177
+ case STATUS_LINK_TIMEOUT:
178
+ case STATUS_IO_TIMEOUT:
179
+ case STATUS_TIMEOUT:
180
+ return WSAETIMEDOUT;
181
+
182
+ case STATUS_GRACEFUL_DISCONNECT:
183
+ return WSAEDISCON;
184
+
185
+ case STATUS_REMOTE_DISCONNECT:
186
+ case STATUS_CONNECTION_RESET:
187
+ case STATUS_LINK_FAILED:
188
+ case STATUS_CONNECTION_DISCONNECTED:
189
+ case STATUS_PORT_UNREACHABLE:
190
+ case STATUS_HOPLIMIT_EXCEEDED:
191
+ return WSAECONNRESET;
192
+
193
+ case STATUS_LOCAL_DISCONNECT:
194
+ case STATUS_TRANSACTION_ABORTED:
195
+ case STATUS_CONNECTION_ABORTED:
196
+ return WSAECONNABORTED;
197
+
198
+ case STATUS_BAD_NETWORK_PATH:
199
+ case STATUS_NETWORK_UNREACHABLE:
200
+ case STATUS_PROTOCOL_UNREACHABLE:
201
+ return WSAENETUNREACH;
202
+
203
+ case STATUS_HOST_UNREACHABLE:
204
+ return WSAEHOSTUNREACH;
205
+
206
+ case STATUS_CANCELLED:
207
+ case STATUS_REQUEST_ABORTED:
208
+ return WSAEINTR;
209
+
210
+ case STATUS_BUFFER_OVERFLOW:
211
+ case STATUS_INVALID_BUFFER_SIZE:
212
+ return WSAEMSGSIZE;
213
+
214
+ case STATUS_BUFFER_TOO_SMALL:
215
+ case STATUS_ACCESS_VIOLATION:
216
+ return WSAEFAULT;
217
+
218
+ case STATUS_DEVICE_NOT_READY:
219
+ case STATUS_REQUEST_NOT_ACCEPTED:
220
+ return WSAEWOULDBLOCK;
221
+
222
+ case STATUS_INVALID_NETWORK_RESPONSE:
223
+ case STATUS_NETWORK_BUSY:
224
+ case STATUS_NO_SUCH_DEVICE:
225
+ case STATUS_NO_SUCH_FILE:
226
+ case STATUS_OBJECT_PATH_NOT_FOUND:
227
+ case STATUS_OBJECT_NAME_NOT_FOUND:
228
+ case STATUS_UNEXPECTED_NETWORK_ERROR:
229
+ return WSAENETDOWN;
230
+
231
+ case STATUS_INVALID_CONNECTION:
232
+ return WSAENOTCONN;
233
+
234
+ case STATUS_REMOTE_NOT_LISTENING:
235
+ case STATUS_CONNECTION_REFUSED:
236
+ return WSAECONNREFUSED;
237
+
238
+ case STATUS_PIPE_DISCONNECTED:
239
+ return WSAESHUTDOWN;
240
+
241
+ case STATUS_INVALID_ADDRESS:
242
+ case STATUS_INVALID_ADDRESS_COMPONENT:
243
+ return WSAEADDRNOTAVAIL;
244
+
245
+ case STATUS_NOT_SUPPORTED:
246
+ case STATUS_NOT_IMPLEMENTED:
247
+ return WSAEOPNOTSUPP;
248
+
249
+ case STATUS_ACCESS_DENIED:
250
+ return WSAEACCES;
251
+
252
+ default:
253
+ if ((status & (FACILITY_NTWIN32 << 16)) == (FACILITY_NTWIN32 << 16) &&
254
+ (status & (ERROR_SEVERITY_ERROR | ERROR_SEVERITY_WARNING))) {
255
+ /* It's a windows error that has been previously mapped to an */
256
+ /* ntstatus code. */
257
+ return (DWORD) (status & 0xffff);
258
+ } else {
259
+ /* The default fallback for unmappable ntstatus codes. */
260
+ return WSAEINVAL;
261
+ }
262
+ }
263
+ }
264
+
265
+
266
+ /*
267
+ * This function provides a workaround for a bug in the winsock implementation
268
+ * of WSARecv. The problem is that when SetFileCompletionNotificationModes is
269
+ * used to avoid IOCP notifications of completed reads, WSARecv does not
270
+ * reliably indicate whether we can expect a completion package to be posted
271
+ * when the receive buffer is smaller than the received datagram.
272
+ *
273
+ * However it is desirable to use SetFileCompletionNotificationModes because
274
+ * it yields a massive performance increase.
275
+ *
276
+ * This function provides a workaround for that bug, but it only works for the
277
+ * specific case that we need it for. E.g. it assumes that the "avoid iocp"
278
+ * bit has been set, and supports only overlapped operation. It also requires
279
+ * the user to use the default msafd driver, doesn't work when other LSPs are
280
+ * stacked on top of it.
281
+ */
282
+ int WSAAPI uv_wsarecv_workaround(SOCKET socket, WSABUF* buffers,
283
+ DWORD buffer_count, DWORD* bytes, DWORD* flags, WSAOVERLAPPED *overlapped,
284
+ LPWSAOVERLAPPED_COMPLETION_ROUTINE completion_routine) {
285
+ NTSTATUS status;
286
+ void* apc_context;
287
+ IO_STATUS_BLOCK* iosb = (IO_STATUS_BLOCK*) &overlapped->Internal;
288
+ AFD_RECV_INFO info;
289
+ DWORD error;
290
+
291
+ if (overlapped == NULL || completion_routine != NULL) {
292
+ WSASetLastError(WSAEINVAL);
293
+ return SOCKET_ERROR;
294
+ }
295
+
296
+ info.BufferArray = buffers;
297
+ info.BufferCount = buffer_count;
298
+ info.AfdFlags = AFD_OVERLAPPED;
299
+ info.TdiFlags = TDI_RECEIVE_NORMAL;
300
+
301
+ if (*flags & MSG_PEEK) {
302
+ info.TdiFlags |= TDI_RECEIVE_PEEK;
303
+ }
304
+
305
+ if (*flags & MSG_PARTIAL) {
306
+ info.TdiFlags |= TDI_RECEIVE_PARTIAL;
307
+ }
308
+
309
+ if (!((intptr_t) overlapped->hEvent & 1)) {
310
+ apc_context = (void*) overlapped;
311
+ } else {
312
+ apc_context = NULL;
313
+ }
314
+
315
+ iosb->Status = STATUS_PENDING;
316
+ iosb->Pointer = 0;
317
+
318
+ status = pNtDeviceIoControlFile((HANDLE) socket,
319
+ overlapped->hEvent,
320
+ NULL,
321
+ apc_context,
322
+ iosb,
323
+ IOCTL_AFD_RECEIVE,
324
+ &info,
325
+ sizeof(info),
326
+ NULL,
327
+ 0);
328
+
329
+ *flags = 0;
330
+ *bytes = (DWORD) iosb->Information;
331
+
332
+ switch (status) {
333
+ case STATUS_SUCCESS:
334
+ error = ERROR_SUCCESS;
335
+ break;
336
+
337
+ case STATUS_PENDING:
338
+ error = WSA_IO_PENDING;
339
+ break;
340
+
341
+ case STATUS_BUFFER_OVERFLOW:
342
+ error = WSAEMSGSIZE;
343
+ break;
344
+
345
+ case STATUS_RECEIVE_EXPEDITED:
346
+ error = ERROR_SUCCESS;
347
+ *flags = MSG_OOB;
348
+ break;
349
+
350
+ case STATUS_RECEIVE_PARTIAL_EXPEDITED:
351
+ error = ERROR_SUCCESS;
352
+ *flags = MSG_PARTIAL | MSG_OOB;
353
+ break;
354
+
355
+ case STATUS_RECEIVE_PARTIAL:
356
+ error = ERROR_SUCCESS;
357
+ *flags = MSG_PARTIAL;
358
+ break;
359
+
360
+ default:
361
+ error = uv_ntstatus_to_winsock_error(status);
362
+ break;
363
+ }
364
+
365
+ WSASetLastError(error);
366
+
367
+ if (error == ERROR_SUCCESS) {
368
+ return 0;
369
+ } else {
370
+ return SOCKET_ERROR;
371
+ }
372
+ }
373
+
374
+
375
+ /* See description of uv_wsarecv_workaround. */
376
+ int WSAAPI uv_wsarecvfrom_workaround(SOCKET socket, WSABUF* buffers,
377
+ DWORD buffer_count, DWORD* bytes, DWORD* flags, struct sockaddr* addr,
378
+ int* addr_len, WSAOVERLAPPED *overlapped,
379
+ LPWSAOVERLAPPED_COMPLETION_ROUTINE completion_routine) {
380
+ NTSTATUS status;
381
+ void* apc_context;
382
+ IO_STATUS_BLOCK* iosb = (IO_STATUS_BLOCK*) &overlapped->Internal;
383
+ AFD_RECV_DATAGRAM_INFO info;
384
+ DWORD error;
385
+
386
+ if (overlapped == NULL || addr == NULL || addr_len == NULL ||
387
+ completion_routine != NULL) {
388
+ WSASetLastError(WSAEINVAL);
389
+ return SOCKET_ERROR;
390
+ }
391
+
392
+ info.BufferArray = buffers;
393
+ info.BufferCount = buffer_count;
394
+ info.AfdFlags = AFD_OVERLAPPED;
395
+ info.TdiFlags = TDI_RECEIVE_NORMAL;
396
+ info.Address = addr;
397
+ info.AddressLength = addr_len;
398
+
399
+ if (*flags & MSG_PEEK) {
400
+ info.TdiFlags |= TDI_RECEIVE_PEEK;
401
+ }
402
+
403
+ if (*flags & MSG_PARTIAL) {
404
+ info.TdiFlags |= TDI_RECEIVE_PARTIAL;
405
+ }
406
+
407
+ if (!((intptr_t) overlapped->hEvent & 1)) {
408
+ apc_context = (void*) overlapped;
409
+ } else {
410
+ apc_context = NULL;
411
+ }
412
+
413
+ iosb->Status = STATUS_PENDING;
414
+ iosb->Pointer = 0;
415
+
416
+ status = pNtDeviceIoControlFile((HANDLE) socket,
417
+ overlapped->hEvent,
418
+ NULL,
419
+ apc_context,
420
+ iosb,
421
+ IOCTL_AFD_RECEIVE_DATAGRAM,
422
+ &info,
423
+ sizeof(info),
424
+ NULL,
425
+ 0);
426
+
427
+ *flags = 0;
428
+ *bytes = (DWORD) iosb->Information;
429
+
430
+ switch (status) {
431
+ case STATUS_SUCCESS:
432
+ error = ERROR_SUCCESS;
433
+ break;
434
+
435
+ case STATUS_PENDING:
436
+ error = WSA_IO_PENDING;
437
+ break;
438
+
439
+ case STATUS_BUFFER_OVERFLOW:
440
+ error = WSAEMSGSIZE;
441
+ break;
442
+
443
+ case STATUS_RECEIVE_EXPEDITED:
444
+ error = ERROR_SUCCESS;
445
+ *flags = MSG_OOB;
446
+ break;
447
+
448
+ case STATUS_RECEIVE_PARTIAL_EXPEDITED:
449
+ error = ERROR_SUCCESS;
450
+ *flags = MSG_PARTIAL | MSG_OOB;
451
+ break;
452
+
453
+ case STATUS_RECEIVE_PARTIAL:
454
+ error = ERROR_SUCCESS;
455
+ *flags = MSG_PARTIAL;
456
+ break;
457
+
458
+ default:
459
+ error = uv_ntstatus_to_winsock_error(status);
460
+ break;
461
+ }
462
+
463
+ WSASetLastError(error);
464
+
465
+ if (error == ERROR_SUCCESS) {
466
+ return 0;
467
+ } else {
468
+ return SOCKET_ERROR;
469
+ }
470
+ }
471
+
472
+
473
+ int WSAAPI uv_msafd_poll(SOCKET socket, AFD_POLL_INFO* info,
474
+ OVERLAPPED* overlapped) {
475
+ IO_STATUS_BLOCK iosb;
476
+ IO_STATUS_BLOCK* iosb_ptr;
477
+ HANDLE event = NULL;
478
+ void* apc_context;
479
+ NTSTATUS status;
480
+ DWORD error;
481
+
482
+ if (overlapped != NULL) {
483
+ /* Overlapped operation. */
484
+ iosb_ptr = (IO_STATUS_BLOCK*) &overlapped->Internal;
485
+ event = overlapped->hEvent;
486
+
487
+ /* Do not report iocp completion if hEvent is tagged. */
488
+ if ((uintptr_t) event & 1) {
489
+ event = (HANDLE)((uintptr_t) event & ~(uintptr_t) 1);
490
+ apc_context = NULL;
491
+ } else {
492
+ apc_context = overlapped;
493
+ }
494
+
495
+ } else {
496
+ /* Blocking operation. */
497
+ iosb_ptr = &iosb;
498
+ event = CreateEvent(NULL, FALSE, FALSE, NULL);
499
+ if (event == NULL) {
500
+ return SOCKET_ERROR;
501
+ }
502
+ apc_context = NULL;
503
+ }
504
+
505
+ iosb_ptr->Status = STATUS_PENDING;
506
+ status = pNtDeviceIoControlFile((HANDLE) socket,
507
+ event,
508
+ NULL,
509
+ apc_context,
510
+ iosb_ptr,
511
+ IOCTL_AFD_POLL,
512
+ info,
513
+ sizeof *info,
514
+ info,
515
+ sizeof *info);
516
+
517
+ if (overlapped == NULL) {
518
+ /* If this is a blocking operation, wait for the event to become */
519
+ /* signaled, and then grab the real status from the io status block. */
520
+ if (status == STATUS_PENDING) {
521
+ DWORD r = WaitForSingleObject(event, INFINITE);
522
+
523
+ if (r == WAIT_FAILED) {
524
+ DWORD saved_error = GetLastError();
525
+ CloseHandle(event);
526
+ WSASetLastError(saved_error);
527
+ return SOCKET_ERROR;
528
+ }
529
+
530
+ status = iosb.Status;
531
+ }
532
+
533
+ CloseHandle(event);
534
+ }
535
+
536
+ switch (status) {
537
+ case STATUS_SUCCESS:
538
+ error = ERROR_SUCCESS;
539
+ break;
540
+
541
+ case STATUS_PENDING:
542
+ error = WSA_IO_PENDING;
543
+ break;
544
+
545
+ default:
546
+ error = uv_ntstatus_to_winsock_error(status);
547
+ break;
548
+ }
549
+
550
+ WSASetLastError(error);
551
+
552
+ if (error == ERROR_SUCCESS) {
553
+ return 0;
554
+ } else {
555
+ return SOCKET_ERROR;
556
+ }
557
+ }