foolio 0.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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
+ }