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,372 @@
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
+ #include <limits.h>
24
+
25
+ #include "uv.h"
26
+ #include "internal.h"
27
+
28
+
29
+ #define HAVE_SRWLOCK_API() (pTryAcquireSRWLockShared != NULL)
30
+
31
+ #ifdef _MSC_VER /* msvc */
32
+ # define inline __inline
33
+ # define NOINLINE __declspec (noinline)
34
+ #else /* gcc */
35
+ # define inline inline
36
+ # define NOINLINE __attribute__ ((noinline))
37
+ #endif
38
+
39
+
40
+ inline static int uv__rwlock_srwlock_init(uv_rwlock_t* rwlock);
41
+ inline static void uv__rwlock_srwlock_destroy(uv_rwlock_t* rwlock);
42
+ inline static void uv__rwlock_srwlock_rdlock(uv_rwlock_t* rwlock);
43
+ inline static int uv__rwlock_srwlock_tryrdlock(uv_rwlock_t* rwlock);
44
+ inline static void uv__rwlock_srwlock_rdunlock(uv_rwlock_t* rwlock);
45
+ inline static void uv__rwlock_srwlock_wrlock(uv_rwlock_t* rwlock);
46
+ inline static int uv__rwlock_srwlock_trywrlock(uv_rwlock_t* rwlock);
47
+ inline static void uv__rwlock_srwlock_wrunlock(uv_rwlock_t* rwlock);
48
+
49
+ inline static int uv__rwlock_fallback_init(uv_rwlock_t* rwlock);
50
+ inline static void uv__rwlock_fallback_destroy(uv_rwlock_t* rwlock);
51
+ inline static void uv__rwlock_fallback_rdlock(uv_rwlock_t* rwlock);
52
+ inline static int uv__rwlock_fallback_tryrdlock(uv_rwlock_t* rwlock);
53
+ inline static void uv__rwlock_fallback_rdunlock(uv_rwlock_t* rwlock);
54
+ inline static void uv__rwlock_fallback_wrlock(uv_rwlock_t* rwlock);
55
+ inline static int uv__rwlock_fallback_trywrlock(uv_rwlock_t* rwlock);
56
+ inline static void uv__rwlock_fallback_wrunlock(uv_rwlock_t* rwlock);
57
+
58
+
59
+ static NOINLINE void uv__once_inner(uv_once_t* guard,
60
+ void (*callback)(void)) {
61
+ DWORD result;
62
+ HANDLE existing_event, created_event;
63
+ HANDLE* event_ptr;
64
+
65
+ /* Fetch and align event_ptr */
66
+ event_ptr = (HANDLE*) (((uintptr_t) &guard->event + (sizeof(HANDLE) - 1)) &
67
+ ~(sizeof(HANDLE) - 1));
68
+
69
+ created_event = CreateEvent(NULL, 1, 0, NULL);
70
+ if (created_event == 0) {
71
+ /* Could fail in a low-memory situation? */
72
+ uv_fatal_error(GetLastError(), "CreateEvent");
73
+ }
74
+
75
+ existing_event = InterlockedCompareExchangePointer(event_ptr,
76
+ created_event,
77
+ NULL);
78
+
79
+ if (existing_event == NULL) {
80
+ /* We won the race */
81
+ callback();
82
+
83
+ result = SetEvent(created_event);
84
+ assert(result);
85
+ guard->ran = 1;
86
+
87
+ } else {
88
+ /* We lost the race. Destroy the event we created and wait for the */
89
+ /* existing one to become signaled. */
90
+ CloseHandle(created_event);
91
+ result = WaitForSingleObject(existing_event, INFINITE);
92
+ assert(result == WAIT_OBJECT_0);
93
+ }
94
+ }
95
+
96
+
97
+ void uv_once(uv_once_t* guard, void (*callback)(void)) {
98
+ /* Fast case - avoid WaitForSingleObject. */
99
+ if (guard->ran) {
100
+ return;
101
+ }
102
+
103
+ uv__once_inner(guard, callback);
104
+ }
105
+
106
+
107
+ int uv_thread_join(uv_thread_t *tid) {
108
+ if (WaitForSingleObject(*tid, INFINITE))
109
+ return -1;
110
+ else {
111
+ CloseHandle(*tid);
112
+ *tid = 0;
113
+ return 0;
114
+ }
115
+ }
116
+
117
+
118
+ int uv_mutex_init(uv_mutex_t* mutex) {
119
+ InitializeCriticalSection(mutex);
120
+ return 0;
121
+ }
122
+
123
+
124
+ void uv_mutex_destroy(uv_mutex_t* mutex) {
125
+ DeleteCriticalSection(mutex);
126
+ }
127
+
128
+
129
+ void uv_mutex_lock(uv_mutex_t* mutex) {
130
+ EnterCriticalSection(mutex);
131
+ }
132
+
133
+
134
+ int uv_mutex_trylock(uv_mutex_t* mutex) {
135
+ if (TryEnterCriticalSection(mutex))
136
+ return 0;
137
+ else
138
+ return -1;
139
+ }
140
+
141
+
142
+ void uv_mutex_unlock(uv_mutex_t* mutex) {
143
+ LeaveCriticalSection(mutex);
144
+ }
145
+
146
+
147
+ int uv_rwlock_init(uv_rwlock_t* rwlock) {
148
+ if (HAVE_SRWLOCK_API())
149
+ return uv__rwlock_srwlock_init(rwlock);
150
+ else
151
+ return uv__rwlock_fallback_init(rwlock);
152
+ }
153
+
154
+
155
+ void uv_rwlock_destroy(uv_rwlock_t* rwlock) {
156
+ if (HAVE_SRWLOCK_API())
157
+ uv__rwlock_srwlock_destroy(rwlock);
158
+ else
159
+ uv__rwlock_fallback_destroy(rwlock);
160
+ }
161
+
162
+
163
+ void uv_rwlock_rdlock(uv_rwlock_t* rwlock) {
164
+ if (HAVE_SRWLOCK_API())
165
+ uv__rwlock_srwlock_rdlock(rwlock);
166
+ else
167
+ uv__rwlock_fallback_rdlock(rwlock);
168
+ }
169
+
170
+
171
+ int uv_rwlock_tryrdlock(uv_rwlock_t* rwlock) {
172
+ if (HAVE_SRWLOCK_API())
173
+ return uv__rwlock_srwlock_tryrdlock(rwlock);
174
+ else
175
+ return uv__rwlock_fallback_tryrdlock(rwlock);
176
+ }
177
+
178
+
179
+ void uv_rwlock_rdunlock(uv_rwlock_t* rwlock) {
180
+ if (HAVE_SRWLOCK_API())
181
+ uv__rwlock_srwlock_rdunlock(rwlock);
182
+ else
183
+ uv__rwlock_fallback_rdunlock(rwlock);
184
+ }
185
+
186
+
187
+ void uv_rwlock_wrlock(uv_rwlock_t* rwlock) {
188
+ if (HAVE_SRWLOCK_API())
189
+ uv__rwlock_srwlock_wrlock(rwlock);
190
+ else
191
+ uv__rwlock_fallback_wrlock(rwlock);
192
+ }
193
+
194
+
195
+ int uv_rwlock_trywrlock(uv_rwlock_t* rwlock) {
196
+ if (HAVE_SRWLOCK_API())
197
+ return uv__rwlock_srwlock_trywrlock(rwlock);
198
+ else
199
+ return uv__rwlock_fallback_trywrlock(rwlock);
200
+ }
201
+
202
+
203
+ void uv_rwlock_wrunlock(uv_rwlock_t* rwlock) {
204
+ if (HAVE_SRWLOCK_API())
205
+ uv__rwlock_srwlock_wrunlock(rwlock);
206
+ else
207
+ uv__rwlock_fallback_wrunlock(rwlock);
208
+ }
209
+
210
+
211
+ int uv_sem_init(uv_sem_t* sem, unsigned int value) {
212
+ *sem = CreateSemaphore(NULL, value, INT_MAX, NULL);
213
+ return *sem ? 0 : -1;
214
+ }
215
+
216
+
217
+ void uv_sem_destroy(uv_sem_t* sem) {
218
+ if (!CloseHandle(*sem))
219
+ abort();
220
+ }
221
+
222
+
223
+ void uv_sem_post(uv_sem_t* sem) {
224
+ if (!ReleaseSemaphore(*sem, 1, NULL))
225
+ abort();
226
+ }
227
+
228
+
229
+ void uv_sem_wait(uv_sem_t* sem) {
230
+ if (WaitForSingleObject(*sem, INFINITE) != WAIT_OBJECT_0)
231
+ abort();
232
+ }
233
+
234
+
235
+ int uv_sem_trywait(uv_sem_t* sem) {
236
+ DWORD r = WaitForSingleObject(*sem, 0);
237
+
238
+ if (r == WAIT_OBJECT_0)
239
+ return 0;
240
+
241
+ if (r == WAIT_TIMEOUT)
242
+ return -1;
243
+
244
+ abort();
245
+ return -1; /* Satisfy the compiler. */
246
+ }
247
+
248
+
249
+ inline static int uv__rwlock_srwlock_init(uv_rwlock_t* rwlock) {
250
+ pInitializeSRWLock(&rwlock->srwlock_);
251
+ return 0;
252
+ }
253
+
254
+
255
+ inline static void uv__rwlock_srwlock_destroy(uv_rwlock_t* rwlock) {
256
+ (void) rwlock;
257
+ }
258
+
259
+
260
+ inline static void uv__rwlock_srwlock_rdlock(uv_rwlock_t* rwlock) {
261
+ pAcquireSRWLockShared(&rwlock->srwlock_);
262
+ }
263
+
264
+
265
+ inline static int uv__rwlock_srwlock_tryrdlock(uv_rwlock_t* rwlock) {
266
+ if (pTryAcquireSRWLockShared(&rwlock->srwlock_))
267
+ return 0;
268
+ else
269
+ return -1;
270
+ }
271
+
272
+
273
+ inline static void uv__rwlock_srwlock_rdunlock(uv_rwlock_t* rwlock) {
274
+ pReleaseSRWLockShared(&rwlock->srwlock_);
275
+ }
276
+
277
+
278
+ inline static void uv__rwlock_srwlock_wrlock(uv_rwlock_t* rwlock) {
279
+ pAcquireSRWLockExclusive(&rwlock->srwlock_);
280
+ }
281
+
282
+
283
+ inline static int uv__rwlock_srwlock_trywrlock(uv_rwlock_t* rwlock) {
284
+ if (pTryAcquireSRWLockExclusive(&rwlock->srwlock_))
285
+ return 0;
286
+ else
287
+ return -1;
288
+ }
289
+
290
+
291
+ inline static void uv__rwlock_srwlock_wrunlock(uv_rwlock_t* rwlock) {
292
+ pReleaseSRWLockExclusive(&rwlock->srwlock_);
293
+ }
294
+
295
+
296
+ inline static int uv__rwlock_fallback_init(uv_rwlock_t* rwlock) {
297
+ if (uv_mutex_init(&rwlock->fallback_.read_mutex_))
298
+ return -1;
299
+
300
+ if (uv_mutex_init(&rwlock->fallback_.write_mutex_)) {
301
+ uv_mutex_destroy(&rwlock->fallback_.read_mutex_);
302
+ return -1;
303
+ }
304
+
305
+ rwlock->fallback_.num_readers_ = 0;
306
+
307
+ return 0;
308
+ }
309
+
310
+
311
+ inline static void uv__rwlock_fallback_destroy(uv_rwlock_t* rwlock) {
312
+ uv_mutex_destroy(&rwlock->fallback_.read_mutex_);
313
+ uv_mutex_destroy(&rwlock->fallback_.write_mutex_);
314
+ }
315
+
316
+
317
+ inline static void uv__rwlock_fallback_rdlock(uv_rwlock_t* rwlock) {
318
+ uv_mutex_lock(&rwlock->fallback_.read_mutex_);
319
+
320
+ if (++rwlock->fallback_.num_readers_ == 1)
321
+ uv_mutex_lock(&rwlock->fallback_.write_mutex_);
322
+
323
+ uv_mutex_unlock(&rwlock->fallback_.read_mutex_);
324
+ }
325
+
326
+
327
+ inline static int uv__rwlock_fallback_tryrdlock(uv_rwlock_t* rwlock) {
328
+ int ret;
329
+
330
+ ret = -1;
331
+
332
+ if (uv_mutex_trylock(&rwlock->fallback_.read_mutex_))
333
+ goto out;
334
+
335
+ if (rwlock->fallback_.num_readers_ == 0)
336
+ ret = uv_mutex_trylock(&rwlock->fallback_.write_mutex_);
337
+ else
338
+ ret = 0;
339
+
340
+ if (ret == 0)
341
+ rwlock->fallback_.num_readers_++;
342
+
343
+ uv_mutex_unlock(&rwlock->fallback_.read_mutex_);
344
+
345
+ out:
346
+ return ret;
347
+ }
348
+
349
+
350
+ inline static void uv__rwlock_fallback_rdunlock(uv_rwlock_t* rwlock) {
351
+ uv_mutex_lock(&rwlock->fallback_.read_mutex_);
352
+
353
+ if (--rwlock->fallback_.num_readers_ == 0)
354
+ uv_mutex_unlock(&rwlock->fallback_.write_mutex_);
355
+
356
+ uv_mutex_unlock(&rwlock->fallback_.read_mutex_);
357
+ }
358
+
359
+
360
+ inline static void uv__rwlock_fallback_wrlock(uv_rwlock_t* rwlock) {
361
+ uv_mutex_lock(&rwlock->fallback_.write_mutex_);
362
+ }
363
+
364
+
365
+ inline static int uv__rwlock_fallback_trywrlock(uv_rwlock_t* rwlock) {
366
+ return uv_mutex_trylock(&rwlock->fallback_.write_mutex_);
367
+ }
368
+
369
+
370
+ inline static void uv__rwlock_fallback_wrunlock(uv_rwlock_t* rwlock) {
371
+ uv_mutex_unlock(&rwlock->fallback_.write_mutex_);
372
+ }
@@ -0,0 +1,74 @@
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
+ #include "req-inl.h"
27
+
28
+
29
+ static void uv_work_req_init(uv_loop_t* loop, uv_work_t* req,
30
+ uv_work_cb work_cb, uv_after_work_cb after_work_cb) {
31
+ uv_req_init(loop, (uv_req_t*) req);
32
+ req->type = UV_WORK;
33
+ req->loop = loop;
34
+ req->work_cb = work_cb;
35
+ req->after_work_cb = after_work_cb;
36
+ memset(&req->overlapped, 0, sizeof(req->overlapped));
37
+ }
38
+
39
+
40
+ static DWORD WINAPI uv_work_thread_proc(void* parameter) {
41
+ uv_work_t* req = (uv_work_t*)parameter;
42
+ uv_loop_t* loop = req->loop;
43
+
44
+ assert(req != NULL);
45
+ assert(req->type == UV_WORK);
46
+ assert(req->work_cb);
47
+
48
+ req->work_cb(req);
49
+
50
+ POST_COMPLETION_FOR_REQ(loop, req);
51
+
52
+ return 0;
53
+ }
54
+
55
+
56
+ int uv_queue_work(uv_loop_t* loop, uv_work_t* req, uv_work_cb work_cb,
57
+ uv_after_work_cb after_work_cb) {
58
+ uv_work_req_init(loop, req, work_cb, after_work_cb);
59
+
60
+ if (!QueueUserWorkItem(&uv_work_thread_proc, req, WT_EXECUTELONGFUNCTION)) {
61
+ uv__set_sys_error(loop, GetLastError());
62
+ return -1;
63
+ }
64
+
65
+ uv__req_register(loop, req);
66
+ return 0;
67
+ }
68
+
69
+
70
+ void uv_process_work_req(uv_loop_t* loop, uv_work_t* req) {
71
+ assert(req->after_work_cb);
72
+ uv__req_unregister(loop, req);
73
+ req->after_work_cb(req);
74
+ }
@@ -0,0 +1,224 @@
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
+ #include <limits.h>
24
+
25
+ #include "uv.h"
26
+ #include "internal.h"
27
+ #include "tree.h"
28
+ #include "handle-inl.h"
29
+
30
+
31
+ void uv_update_time(uv_loop_t* loop) {
32
+ DWORD ticks = GetTickCount();
33
+
34
+ /* The assumption is made that LARGE_INTEGER.QuadPart has the same type */
35
+ /* loop->time, which happens to be. Is there any way to assert this? */
36
+ LARGE_INTEGER* time = (LARGE_INTEGER*) &loop->time;
37
+
38
+ /* If the timer has wrapped, add 1 to it's high-order dword. */
39
+ /* uv_poll must make sure that the timer can never overflow more than */
40
+ /* once between two subsequent uv_update_time calls. */
41
+ if (ticks < time->LowPart) {
42
+ time->HighPart += 1;
43
+ }
44
+ time->LowPart = ticks;
45
+ }
46
+
47
+
48
+ int64_t uv_now(uv_loop_t* loop) {
49
+ return loop->time;
50
+ }
51
+
52
+
53
+ static int uv_timer_compare(uv_timer_t* a, uv_timer_t* b) {
54
+ if (a->due < b->due)
55
+ return -1;
56
+ if (a->due > b->due)
57
+ return 1;
58
+ if ((intptr_t)a < (intptr_t)b)
59
+ return -1;
60
+ if ((intptr_t)a > (intptr_t)b)
61
+ return 1;
62
+ return 0;
63
+ }
64
+
65
+
66
+ RB_GENERATE_STATIC(uv_timer_tree_s, uv_timer_s, tree_entry, uv_timer_compare);
67
+
68
+
69
+ int uv_timer_init(uv_loop_t* loop, uv_timer_t* handle) {
70
+ uv__handle_init(loop, (uv_handle_t*) handle, UV_TIMER);
71
+ handle->timer_cb = NULL;
72
+ handle->repeat = 0;
73
+
74
+ return 0;
75
+ }
76
+
77
+
78
+ void uv_timer_endgame(uv_loop_t* loop, uv_timer_t* handle) {
79
+ if (handle->flags & UV_HANDLE_CLOSING) {
80
+ assert(!(handle->flags & UV_HANDLE_CLOSED));
81
+ uv__handle_stop(handle);
82
+ uv__handle_close(handle);
83
+ }
84
+ }
85
+
86
+
87
+ int uv_timer_start(uv_timer_t* handle, uv_timer_cb timer_cb, int64_t timeout,
88
+ int64_t repeat) {
89
+ uv_loop_t* loop = handle->loop;
90
+ uv_timer_t* old;
91
+
92
+ if (handle->flags & UV_HANDLE_ACTIVE) {
93
+ RB_REMOVE(uv_timer_tree_s, &loop->timers, handle);
94
+ }
95
+
96
+ handle->timer_cb = timer_cb;
97
+ handle->due = loop->time + timeout;
98
+ handle->repeat = repeat;
99
+ handle->flags |= UV_HANDLE_ACTIVE;
100
+ uv__handle_start(handle);
101
+
102
+ old = RB_INSERT(uv_timer_tree_s, &loop->timers, handle);
103
+ assert(old == NULL);
104
+
105
+ return 0;
106
+ }
107
+
108
+
109
+ int uv_timer_stop(uv_timer_t* handle) {
110
+ uv_loop_t* loop = handle->loop;
111
+
112
+ if (!(handle->flags & UV_HANDLE_ACTIVE))
113
+ return 0;
114
+
115
+ RB_REMOVE(uv_timer_tree_s, &loop->timers, handle);
116
+
117
+ handle->flags &= ~UV_HANDLE_ACTIVE;
118
+ uv__handle_stop(handle);
119
+
120
+ return 0;
121
+ }
122
+
123
+
124
+ int uv_timer_again(uv_timer_t* handle) {
125
+ uv_loop_t* loop = handle->loop;
126
+
127
+ /* If timer_cb is NULL that means that the timer was never started. */
128
+ if (!handle->timer_cb) {
129
+ uv__set_sys_error(loop, ERROR_INVALID_DATA);
130
+ return -1;
131
+ }
132
+
133
+ if (handle->flags & UV_HANDLE_ACTIVE) {
134
+ RB_REMOVE(uv_timer_tree_s, &loop->timers, handle);
135
+ handle->flags &= ~UV_HANDLE_ACTIVE;
136
+ uv__handle_stop(handle);
137
+ }
138
+
139
+ if (handle->repeat) {
140
+ handle->due = loop->time + handle->repeat;
141
+
142
+ if (RB_INSERT(uv_timer_tree_s, &loop->timers, handle) != NULL) {
143
+ uv_fatal_error(ERROR_INVALID_DATA, "RB_INSERT");
144
+ }
145
+
146
+ handle->flags |= UV_HANDLE_ACTIVE;
147
+ uv__handle_start(handle);
148
+ }
149
+
150
+ return 0;
151
+ }
152
+
153
+
154
+ void uv_timer_set_repeat(uv_timer_t* handle, int64_t repeat) {
155
+ assert(handle->type == UV_TIMER);
156
+ handle->repeat = repeat;
157
+ }
158
+
159
+
160
+ int64_t uv_timer_get_repeat(uv_timer_t* handle) {
161
+ assert(handle->type == UV_TIMER);
162
+ return handle->repeat;
163
+ }
164
+
165
+
166
+ DWORD uv_get_poll_timeout(uv_loop_t* loop) {
167
+ uv_timer_t* timer;
168
+ int64_t delta;
169
+
170
+ /* Check if there are any running timers */
171
+ timer = RB_MIN(uv_timer_tree_s, &loop->timers);
172
+ if (timer) {
173
+ uv_update_time(loop);
174
+
175
+ delta = timer->due - loop->time;
176
+ if (delta >= UINT_MAX >> 1) {
177
+ /* A timeout value of UINT_MAX means infinite, so that's no good. But */
178
+ /* more importantly, there's always the risk that GetTickCount wraps. */
179
+ /* uv_update_time can detect this, but we must make sure that the */
180
+ /* tick counter never overflows twice between two subsequent */
181
+ /* uv_update_time calls. We do this by never sleeping more than half */
182
+ /* the time it takes to wrap the counter - which is huge overkill, */
183
+ /* but hey, it's not so bad to wake up every 25 days. */
184
+ return UINT_MAX >> 1;
185
+ } else if (delta < 0) {
186
+ /* Negative timeout values are not allowed */
187
+ return 0;
188
+ } else {
189
+ return (DWORD)delta;
190
+ }
191
+ } else {
192
+ /* No timers */
193
+ return INFINITE;
194
+ }
195
+ }
196
+
197
+
198
+ void uv_process_timers(uv_loop_t* loop) {
199
+ uv_timer_t* timer;
200
+
201
+ /* Call timer callbacks */
202
+ for (timer = RB_MIN(uv_timer_tree_s, &loop->timers);
203
+ timer != NULL && timer->due <= loop->time;
204
+ timer = RB_MIN(uv_timer_tree_s, &loop->timers)) {
205
+ RB_REMOVE(uv_timer_tree_s, &loop->timers, timer);
206
+
207
+ if (timer->repeat != 0) {
208
+ /* If it is a repeating timer, reschedule with repeat timeout. */
209
+ timer->due += timer->repeat;
210
+ if (timer->due < loop->time) {
211
+ timer->due = loop->time;
212
+ }
213
+ if (RB_INSERT(uv_timer_tree_s, &loop->timers, timer) != NULL) {
214
+ uv_fatal_error(ERROR_INVALID_DATA, "RB_INSERT");
215
+ }
216
+ } else {
217
+ /* If non-repeating, mark the timer as inactive. */
218
+ timer->flags &= ~UV_HANDLE_ACTIVE;
219
+ uv__handle_stop(timer);
220
+ }
221
+
222
+ timer->timer_cb((uv_timer_t*) timer, 0);
223
+ }
224
+ }