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,118 @@
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 "uv.h"
23
+ #include "task.h"
24
+
25
+ #include <stdio.h>
26
+ #include <stdlib.h>
27
+
28
+ typedef struct {
29
+ uv_tcp_t handle;
30
+ uv_shutdown_t shutdown_req;
31
+ } conn_rec;
32
+
33
+ static uv_tcp_t tcp_server;
34
+
35
+ static void connection_cb(uv_stream_t* stream, int status);
36
+ static uv_buf_t alloc_cb(uv_handle_t* handle, size_t suggested_size);
37
+ static void read_cb(uv_stream_t* stream, ssize_t nread, uv_buf_t buf);
38
+ static void shutdown_cb(uv_shutdown_t* req, int status);
39
+ static void close_cb(uv_handle_t* handle);
40
+
41
+
42
+ static void connection_cb(uv_stream_t* stream, int status) {
43
+ conn_rec* conn;
44
+ int r;
45
+
46
+ ASSERT(status == 0);
47
+ ASSERT(stream == (uv_stream_t*)&tcp_server);
48
+
49
+ conn = malloc(sizeof *conn);
50
+ ASSERT(conn != NULL);
51
+
52
+ r = uv_tcp_init(stream->loop, &conn->handle);
53
+ ASSERT(r == 0);
54
+
55
+ r = uv_accept(stream, (uv_stream_t*)&conn->handle);
56
+ ASSERT(r == 0);
57
+
58
+ r = uv_read_start((uv_stream_t*)&conn->handle, alloc_cb, read_cb);
59
+ ASSERT(r == 0);
60
+ }
61
+
62
+
63
+ static uv_buf_t alloc_cb(uv_handle_t* handle, size_t suggested_size) {
64
+ static char buf[65536];
65
+ return uv_buf_init(buf, sizeof buf);
66
+ }
67
+
68
+
69
+ static void read_cb(uv_stream_t* stream, ssize_t nread, uv_buf_t buf) {
70
+ conn_rec* conn;
71
+ int r;
72
+
73
+ if (nread >= 0)
74
+ return;
75
+
76
+ ASSERT(uv_last_error(stream->loop).code == UV_EOF);
77
+
78
+ conn = container_of(stream, conn_rec, handle);
79
+
80
+ r = uv_shutdown(&conn->shutdown_req, stream, shutdown_cb);
81
+ ASSERT(r == 0);
82
+ }
83
+
84
+
85
+ static void shutdown_cb(uv_shutdown_t* req, int status) {
86
+ conn_rec* conn = container_of(req, conn_rec, shutdown_req);
87
+ uv_close((uv_handle_t*)&conn->handle, close_cb);
88
+ }
89
+
90
+
91
+ static void close_cb(uv_handle_t* handle) {
92
+ conn_rec* conn = container_of(handle, conn_rec, handle);
93
+ free(conn);
94
+ }
95
+
96
+
97
+ HELPER_IMPL(tcp4_blackhole_server) {
98
+ struct sockaddr_in addr;
99
+ uv_loop_t* loop;
100
+ int r;
101
+
102
+ loop = uv_default_loop();
103
+ addr = uv_ip4_addr("127.0.0.1", TEST_PORT);
104
+
105
+ r = uv_tcp_init(loop, &tcp_server);
106
+ ASSERT(r == 0);
107
+
108
+ r = uv_tcp_bind(&tcp_server, addr);
109
+ ASSERT(r == 0);
110
+
111
+ r = uv_listen((uv_stream_t*)&tcp_server, 128, connection_cb);
112
+ ASSERT(r == 0);
113
+
114
+ r = uv_run(loop);
115
+ ASSERT(0 && "Blackhole server dropped out of event loop.");
116
+
117
+ return 0;
118
+ }
@@ -0,0 +1,321 @@
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 "uv.h"
23
+ #include "task.h"
24
+ #include <stdio.h>
25
+ #include <stdlib.h>
26
+ #include <string.h>
27
+
28
+
29
+ typedef struct {
30
+ uv_write_t req;
31
+ uv_buf_t buf;
32
+ } write_req_t;
33
+
34
+
35
+ /* used to track multiple DNS requests received */
36
+ typedef struct {
37
+ char* prevbuf_ptr;
38
+ int prevbuf_pos;
39
+ int prevbuf_rem;
40
+ } dnsstate;
41
+
42
+
43
+ /* modify handle to append dnsstate */
44
+ typedef struct {
45
+ uv_tcp_t handle;
46
+ dnsstate state;
47
+ } dnshandle;
48
+
49
+
50
+ static uv_loop_t* loop;
51
+
52
+
53
+ static uv_tcp_t server;
54
+
55
+
56
+ static void after_write(uv_write_t* req, int status);
57
+ static void after_read(uv_stream_t*, ssize_t nread, uv_buf_t buf);
58
+ static void on_close(uv_handle_t* peer);
59
+ static void on_connection(uv_stream_t*, int status);
60
+
61
+ #define WRITE_BUF_LEN (64*1024)
62
+ #define DNSREC_LEN (4)
63
+
64
+ #define LEN_OFFSET 0
65
+ #define QUERYID_OFFSET 2
66
+ unsigned char DNSRsp[] = {0, 43, 0, 0, 0x81, 0x80, 0, 1, 0, 1, 0, 0, 0, 0 };
67
+ unsigned char qrecord[] = {5, 'e', 'c', 'h', 'o', 's', 3, 's', 'r', 'v', 0, 0, 1, 0, 1};
68
+ unsigned char arecord[] = {0xc0, 0x0c, 0, 1, 0, 1, 0, 0, 5, 0xbd, 0, 4, 10, 0, 1, 1 };
69
+
70
+
71
+ static void after_write(uv_write_t* req, int status) {
72
+ write_req_t* wr;
73
+
74
+ if (status) {
75
+ uv_err_t err = uv_last_error(loop);
76
+ fprintf(stderr, "uv_write error: %s\n", uv_strerror(err));
77
+ ASSERT(0);
78
+ }
79
+
80
+ wr = (write_req_t*) req;
81
+
82
+ /* Free the read/write buffer and the request */
83
+ free(wr->buf.base);
84
+ free(wr);
85
+ }
86
+
87
+
88
+ static void after_shutdown(uv_shutdown_t* req, int status) {
89
+ uv_close((uv_handle_t*) req->handle, on_close);
90
+ free(req);
91
+ }
92
+
93
+
94
+ static void addrsp(write_req_t* wr, char* hdr) {
95
+ char * dnsrsp;
96
+ short int rsplen;
97
+ short int* reclen;
98
+
99
+ rsplen = sizeof(DNSRsp) + sizeof(qrecord) + sizeof(arecord);
100
+
101
+ ASSERT (rsplen + wr->buf.len < WRITE_BUF_LEN);
102
+
103
+ dnsrsp = wr->buf.base + wr->buf.len;
104
+
105
+ /* copy stock response */
106
+ memcpy(dnsrsp, DNSRsp, sizeof(DNSRsp));
107
+ memcpy(dnsrsp + sizeof(DNSRsp), qrecord, sizeof(qrecord));
108
+ memcpy(dnsrsp + sizeof(DNSRsp) + sizeof(qrecord), arecord, sizeof(arecord));
109
+
110
+ /* overwrite with network order length and id from request header */
111
+ reclen = (short int*)dnsrsp;
112
+ *reclen = htons(rsplen-2);
113
+ dnsrsp[QUERYID_OFFSET] = hdr[QUERYID_OFFSET];
114
+ dnsrsp[QUERYID_OFFSET+1] = hdr[QUERYID_OFFSET+1];
115
+
116
+ wr->buf.len += rsplen;
117
+ }
118
+
119
+ static void process_req(uv_stream_t* handle, ssize_t nread, uv_buf_t buf) {
120
+ write_req_t* wr;
121
+ dnshandle* dns = (dnshandle*)handle;
122
+ char hdrbuf[DNSREC_LEN];
123
+ int hdrbuf_remaining = DNSREC_LEN;
124
+ int rec_remaining = 0;
125
+ int readbuf_remaining;
126
+ char* dnsreq;
127
+ char* hdrstart;
128
+ int usingprev = 0;
129
+
130
+ wr = (write_req_t*) malloc(sizeof *wr);
131
+ wr->buf.base = (char*)malloc(WRITE_BUF_LEN);
132
+ wr->buf.len = 0;
133
+
134
+ if (dns->state.prevbuf_ptr != NULL) {
135
+ dnsreq = dns->state.prevbuf_ptr + dns->state.prevbuf_pos;
136
+ readbuf_remaining = dns->state.prevbuf_rem;
137
+ usingprev = 1;
138
+ } else {
139
+ dnsreq = buf.base;
140
+ readbuf_remaining = nread;
141
+ }
142
+ hdrstart = dnsreq;
143
+
144
+ while (dnsreq != NULL) {
145
+ /* something to process */
146
+ while (readbuf_remaining > 0) {
147
+ /* something to process in current buffer */
148
+ if (hdrbuf_remaining > 0) {
149
+ /* process len and id */
150
+ if (readbuf_remaining < hdrbuf_remaining) {
151
+ /* too little to get request header. save for next buffer */
152
+ memcpy(&hdrbuf[DNSREC_LEN - hdrbuf_remaining], dnsreq, readbuf_remaining);
153
+ hdrbuf_remaining = DNSREC_LEN - readbuf_remaining;
154
+ break;
155
+ } else {
156
+ short int reclen_n;
157
+ /* save header */
158
+ memcpy(&hdrbuf[DNSREC_LEN - hdrbuf_remaining], dnsreq, hdrbuf_remaining);
159
+ dnsreq += hdrbuf_remaining;
160
+ readbuf_remaining -= hdrbuf_remaining;
161
+ hdrbuf_remaining = 0;
162
+
163
+ /* get record length */
164
+ reclen_n = *((short int*)hdrbuf);
165
+ rec_remaining = ntohs(reclen_n) - (DNSREC_LEN - 2);
166
+ }
167
+ }
168
+
169
+ if (rec_remaining <= readbuf_remaining) {
170
+ /* prepare reply */
171
+ addrsp(wr, hdrbuf);
172
+
173
+ /* move to next record */
174
+ dnsreq += rec_remaining;
175
+ hdrstart = dnsreq;
176
+ readbuf_remaining -= rec_remaining;
177
+ rec_remaining = 0;
178
+ hdrbuf_remaining = DNSREC_LEN;
179
+ } else {
180
+ /* otherwise this buffer is done. */
181
+ rec_remaining -= readbuf_remaining;
182
+ break;
183
+ }
184
+ }
185
+
186
+ /* if we had to use bytes from prev buffer, start processing the current one */
187
+ if (usingprev == 1) {
188
+ /* free previous buffer */
189
+ free(dns->state.prevbuf_ptr);
190
+ dnsreq = buf.base;
191
+ readbuf_remaining = nread;
192
+ usingprev = 0;
193
+ } else {
194
+ dnsreq = NULL;
195
+ }
196
+ }
197
+
198
+ /* send write buffer */
199
+ if (wr->buf.len > 0) {
200
+ if (uv_write((uv_write_t*) &wr->req, handle, &wr->buf, 1, after_write)) {
201
+ FATAL("uv_write failed");
202
+ }
203
+ }
204
+
205
+ if (readbuf_remaining > 0) {
206
+ /* save start of record position, so we can continue on next read */
207
+ dns->state.prevbuf_ptr = buf.base;
208
+ dns->state.prevbuf_pos = hdrstart - buf.base;
209
+ dns->state.prevbuf_rem = nread - dns->state.prevbuf_pos;
210
+ } else {
211
+ /* nothing left in this buffer */
212
+ dns->state.prevbuf_ptr = NULL;
213
+ dns->state.prevbuf_pos = 0;
214
+ dns->state.prevbuf_rem = 0;
215
+ free(buf.base);
216
+ }
217
+ }
218
+
219
+ static void after_read(uv_stream_t* handle, ssize_t nread, uv_buf_t buf) {
220
+ uv_shutdown_t* req;
221
+
222
+ if (nread < 0) {
223
+ /* Error or EOF */
224
+ ASSERT (uv_last_error(loop).code == UV_EOF);
225
+
226
+ if (buf.base) {
227
+ free(buf.base);
228
+ }
229
+
230
+ req = malloc(sizeof *req);
231
+ uv_shutdown(req, handle, after_shutdown);
232
+
233
+ return;
234
+ }
235
+
236
+ if (nread == 0) {
237
+ /* Everything OK, but nothing read. */
238
+ free(buf.base);
239
+ return;
240
+ }
241
+ /* process requests and send responses */
242
+ process_req(handle, nread, buf);
243
+ }
244
+
245
+
246
+ static void on_close(uv_handle_t* peer) {
247
+ free(peer);
248
+ }
249
+
250
+
251
+ static uv_buf_t buf_alloc(uv_handle_t* handle, size_t suggested_size) {
252
+ uv_buf_t buf;
253
+ buf.base = (char*) malloc(suggested_size);
254
+ buf.len = suggested_size;
255
+ return buf;
256
+ }
257
+
258
+
259
+ static void on_connection(uv_stream_t* server, int status) {
260
+ dnshandle* handle;
261
+ int r;
262
+
263
+ ASSERT(status == 0);
264
+
265
+ handle = (dnshandle*) malloc(sizeof *handle);
266
+ ASSERT(handle != NULL);
267
+
268
+ /* initialize read buffer state */
269
+ handle->state.prevbuf_ptr = 0;
270
+ handle->state.prevbuf_pos = 0;
271
+ handle->state.prevbuf_rem = 0;
272
+
273
+ r = uv_tcp_init(loop, (uv_tcp_t*)handle);
274
+ ASSERT(r == 0);
275
+
276
+ r = uv_accept(server, (uv_stream_t*)handle);
277
+ ASSERT(r == 0);
278
+
279
+ r = uv_read_start((uv_stream_t*)handle, buf_alloc, after_read);
280
+ ASSERT(r == 0);
281
+ }
282
+
283
+
284
+ static int dns_start(int port) {
285
+ struct sockaddr_in addr = uv_ip4_addr("0.0.0.0", port);
286
+ int r;
287
+
288
+ r = uv_tcp_init(loop, &server);
289
+ if (r) {
290
+ /* TODO: Error codes */
291
+ fprintf(stderr, "Socket creation error\n");
292
+ return 1;
293
+ }
294
+
295
+ r = uv_tcp_bind(&server, addr);
296
+ if (r) {
297
+ /* TODO: Error codes */
298
+ fprintf(stderr, "Bind error\n");
299
+ return 1;
300
+ }
301
+
302
+ r = uv_listen((uv_stream_t*)&server, 128, on_connection);
303
+ if (r) {
304
+ /* TODO: Error codes */
305
+ fprintf(stderr, "Listen error\n");
306
+ return 1;
307
+ }
308
+
309
+ return 0;
310
+ }
311
+
312
+
313
+ HELPER_IMPL(dns_server) {
314
+ loop = uv_default_loop();
315
+
316
+ if (dns_start(TEST_PORT_2))
317
+ return 1;
318
+
319
+ uv_run(loop);
320
+ return 0;
321
+ }
@@ -0,0 +1,378 @@
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 "uv.h"
23
+ #include "task.h"
24
+ #include <stdio.h>
25
+ #include <stdlib.h>
26
+
27
+ typedef struct {
28
+ uv_write_t req;
29
+ uv_buf_t buf;
30
+ } write_req_t;
31
+
32
+ static uv_loop_t* loop;
33
+
34
+ static int server_closed;
35
+ static stream_type serverType;
36
+ static uv_tcp_t tcpServer;
37
+ static uv_udp_t udpServer;
38
+ static uv_pipe_t pipeServer;
39
+ static uv_handle_t* server;
40
+
41
+ static void after_write(uv_write_t* req, int status);
42
+ static void after_read(uv_stream_t*, ssize_t nread, uv_buf_t buf);
43
+ static void on_close(uv_handle_t* peer);
44
+ static void on_server_close(uv_handle_t* handle);
45
+ static void on_connection(uv_stream_t*, int status);
46
+
47
+
48
+ static void after_write(uv_write_t* req, int status) {
49
+ write_req_t* wr;
50
+
51
+ if (status) {
52
+ uv_err_t err = uv_last_error(loop);
53
+ fprintf(stderr, "uv_write error: %s\n", uv_strerror(err));
54
+ ASSERT(0);
55
+ }
56
+
57
+ wr = (write_req_t*) req;
58
+
59
+ /* Free the read/write buffer and the request */
60
+ free(wr->buf.base);
61
+ free(wr);
62
+ }
63
+
64
+
65
+ static void after_shutdown(uv_shutdown_t* req, int status) {
66
+ uv_close((uv_handle_t*)req->handle, on_close);
67
+ free(req);
68
+ }
69
+
70
+
71
+ static void after_read(uv_stream_t* handle, ssize_t nread, uv_buf_t buf) {
72
+ int i;
73
+ write_req_t *wr;
74
+ uv_shutdown_t* req;
75
+
76
+ if (nread < 0) {
77
+ /* Error or EOF */
78
+ ASSERT (uv_last_error(loop).code == UV_EOF);
79
+
80
+ if (buf.base) {
81
+ free(buf.base);
82
+ }
83
+
84
+ req = (uv_shutdown_t*) malloc(sizeof *req);
85
+ uv_shutdown(req, handle, after_shutdown);
86
+
87
+ return;
88
+ }
89
+
90
+ if (nread == 0) {
91
+ /* Everything OK, but nothing read. */
92
+ free(buf.base);
93
+ return;
94
+ }
95
+
96
+ /*
97
+ * Scan for the letter Q which signals that we should quit the server.
98
+ * If we get QS it means close the stream.
99
+ */
100
+ if (!server_closed) {
101
+ for (i = 0; i < nread; i++) {
102
+ if (buf.base[i] == 'Q') {
103
+ if (i + 1 < nread && buf.base[i + 1] == 'S') {
104
+ free(buf.base);
105
+ uv_close((uv_handle_t*)handle, on_close);
106
+ return;
107
+ } else {
108
+ uv_close(server, on_server_close);
109
+ server_closed = 1;
110
+ }
111
+ }
112
+ }
113
+ }
114
+
115
+ wr = (write_req_t*) malloc(sizeof *wr);
116
+
117
+ wr->buf = uv_buf_init(buf.base, nread);
118
+ if (uv_write(&wr->req, handle, &wr->buf, 1, after_write)) {
119
+ FATAL("uv_write failed");
120
+ }
121
+ }
122
+
123
+
124
+ static void on_close(uv_handle_t* peer) {
125
+ free(peer);
126
+ }
127
+
128
+
129
+ static uv_buf_t echo_alloc(uv_handle_t* handle, size_t suggested_size) {
130
+ return uv_buf_init(malloc(suggested_size), suggested_size);
131
+ }
132
+
133
+
134
+ static void on_connection(uv_stream_t* server, int status) {
135
+ uv_stream_t* stream;
136
+ int r;
137
+
138
+ if (status != 0) {
139
+ fprintf(stderr, "Connect error %d\n",
140
+ uv_last_error(loop).code);
141
+ }
142
+ ASSERT(status == 0);
143
+
144
+ switch (serverType) {
145
+ case TCP:
146
+ stream = malloc(sizeof(uv_tcp_t));
147
+ ASSERT(stream != NULL);
148
+ r = uv_tcp_init(loop, (uv_tcp_t*)stream);
149
+ ASSERT(r == 0);
150
+ break;
151
+
152
+ case PIPE:
153
+ stream = malloc(sizeof(uv_pipe_t));
154
+ ASSERT(stream != NULL);
155
+ r = uv_pipe_init(loop, (uv_pipe_t*)stream, 0);
156
+ ASSERT(r == 0);
157
+ break;
158
+
159
+ default:
160
+ ASSERT(0 && "Bad serverType");
161
+ abort();
162
+ }
163
+
164
+ /* associate server with stream */
165
+ stream->data = server;
166
+
167
+ r = uv_accept(server, stream);
168
+ ASSERT(r == 0);
169
+
170
+ r = uv_read_start(stream, echo_alloc, after_read);
171
+ ASSERT(r == 0);
172
+ }
173
+
174
+
175
+ static void on_server_close(uv_handle_t* handle) {
176
+ ASSERT(handle == server);
177
+ }
178
+
179
+
180
+ static void on_send(uv_udp_send_t* req, int status);
181
+
182
+
183
+ static void on_recv(uv_udp_t* handle,
184
+ ssize_t nread,
185
+ uv_buf_t buf,
186
+ struct sockaddr* addr,
187
+ unsigned flags) {
188
+ uv_udp_send_t* req;
189
+ int r;
190
+
191
+ ASSERT(nread > 0);
192
+ ASSERT(addr->sa_family == AF_INET);
193
+
194
+ req = malloc(sizeof(*req));
195
+ ASSERT(req != NULL);
196
+
197
+ r = uv_udp_send(req, handle, &buf, 1, *(struct sockaddr_in*)addr, on_send);
198
+ ASSERT(r == 0);
199
+ }
200
+
201
+
202
+ static void on_send(uv_udp_send_t* req, int status) {
203
+ ASSERT(status == 0);
204
+ free(req);
205
+ }
206
+
207
+
208
+ static int tcp4_echo_start(int port) {
209
+ struct sockaddr_in addr = uv_ip4_addr("0.0.0.0", port);
210
+ int r;
211
+
212
+ server = (uv_handle_t*)&tcpServer;
213
+ serverType = TCP;
214
+
215
+ r = uv_tcp_init(loop, &tcpServer);
216
+ if (r) {
217
+ /* TODO: Error codes */
218
+ fprintf(stderr, "Socket creation error\n");
219
+ return 1;
220
+ }
221
+
222
+ r = uv_tcp_bind(&tcpServer, addr);
223
+ if (r) {
224
+ /* TODO: Error codes */
225
+ fprintf(stderr, "Bind error\n");
226
+ return 1;
227
+ }
228
+
229
+ r = uv_listen((uv_stream_t*)&tcpServer, SOMAXCONN, on_connection);
230
+ if (r) {
231
+ /* TODO: Error codes */
232
+ fprintf(stderr, "Listen error %s\n",
233
+ uv_err_name(uv_last_error(loop)));
234
+ return 1;
235
+ }
236
+
237
+ return 0;
238
+ }
239
+
240
+
241
+ static int tcp6_echo_start(int port) {
242
+ struct sockaddr_in6 addr6 = uv_ip6_addr("::1", port);
243
+ int r;
244
+
245
+ server = (uv_handle_t*)&tcpServer;
246
+ serverType = TCP;
247
+
248
+ r = uv_tcp_init(loop, &tcpServer);
249
+ if (r) {
250
+ /* TODO: Error codes */
251
+ fprintf(stderr, "Socket creation error\n");
252
+ return 1;
253
+ }
254
+
255
+ /* IPv6 is optional as not all platforms support it */
256
+ r = uv_tcp_bind6(&tcpServer, addr6);
257
+ if (r) {
258
+ /* show message but return OK */
259
+ fprintf(stderr, "IPv6 not supported\n");
260
+ return 0;
261
+ }
262
+
263
+ r = uv_listen((uv_stream_t*)&tcpServer, SOMAXCONN, on_connection);
264
+ if (r) {
265
+ /* TODO: Error codes */
266
+ fprintf(stderr, "Listen error\n");
267
+ return 1;
268
+ }
269
+
270
+ return 0;
271
+ }
272
+
273
+
274
+ static int udp4_echo_start(int port) {
275
+ int r;
276
+
277
+ server = (uv_handle_t*)&udpServer;
278
+ serverType = UDP;
279
+
280
+ r = uv_udp_init(loop, &udpServer);
281
+ if (r) {
282
+ fprintf(stderr, "uv_udp_init: %s\n",
283
+ uv_strerror(uv_last_error(loop)));
284
+ return 1;
285
+ }
286
+
287
+ r = uv_udp_recv_start(&udpServer, echo_alloc, on_recv);
288
+ if (r) {
289
+ fprintf(stderr, "uv_udp_recv_start: %s\n",
290
+ uv_strerror(uv_last_error(loop)));
291
+ return 1;
292
+ }
293
+
294
+ return 0;
295
+ }
296
+
297
+
298
+ static int pipe_echo_start(char* pipeName) {
299
+ int r;
300
+
301
+ #ifndef _WIN32
302
+ {
303
+ uv_fs_t req;
304
+ uv_fs_unlink(uv_default_loop(), &req, pipeName, NULL);
305
+ uv_fs_req_cleanup(&req);
306
+ }
307
+ #endif
308
+
309
+ server = (uv_handle_t*)&pipeServer;
310
+ serverType = PIPE;
311
+
312
+ r = uv_pipe_init(loop, &pipeServer, 0);
313
+ if (r) {
314
+ fprintf(stderr, "uv_pipe_init: %s\n",
315
+ uv_strerror(uv_last_error(loop)));
316
+ return 1;
317
+ }
318
+
319
+ r = uv_pipe_bind(&pipeServer, pipeName);
320
+ if (r) {
321
+ fprintf(stderr, "uv_pipe_bind: %s\n",
322
+ uv_strerror(uv_last_error(loop)));
323
+ return 1;
324
+ }
325
+
326
+ r = uv_listen((uv_stream_t*)&pipeServer, SOMAXCONN, on_connection);
327
+ if (r) {
328
+ fprintf(stderr, "uv_pipe_listen: %s\n",
329
+ uv_strerror(uv_last_error(loop)));
330
+ return 1;
331
+ }
332
+
333
+ return 0;
334
+ }
335
+
336
+
337
+ HELPER_IMPL(tcp4_echo_server) {
338
+ loop = uv_default_loop();
339
+
340
+ if (tcp4_echo_start(TEST_PORT))
341
+ return 1;
342
+
343
+ uv_run(loop);
344
+ return 0;
345
+ }
346
+
347
+
348
+ HELPER_IMPL(tcp6_echo_server) {
349
+ loop = uv_default_loop();
350
+
351
+ if (tcp6_echo_start(TEST_PORT))
352
+ return 1;
353
+
354
+ uv_run(loop);
355
+ return 0;
356
+ }
357
+
358
+
359
+ HELPER_IMPL(pipe_echo_server) {
360
+ loop = uv_default_loop();
361
+
362
+ if (pipe_echo_start(TEST_PIPENAME))
363
+ return 1;
364
+
365
+ uv_run(loop);
366
+ return 0;
367
+ }
368
+
369
+
370
+ HELPER_IMPL(udp4_echo_server) {
371
+ loop = uv_default_loop();
372
+
373
+ if (udp4_echo_start(TEST_PORT))
374
+ return 1;
375
+
376
+ uv_run(loop);
377
+ return 0;
378
+ }