1*5f2f4271Schristos /* Copyright libuv project contributors. All rights reserved.
2*5f2f4271Schristos *
3*5f2f4271Schristos * Permission is hereby granted, free of charge, to any person obtaining a copy
4*5f2f4271Schristos * of this software and associated documentation files (the "Software"), to
5*5f2f4271Schristos * deal in the Software without restriction, including without limitation the
6*5f2f4271Schristos * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7*5f2f4271Schristos * sell copies of the Software, and to permit persons to whom the Software is
8*5f2f4271Schristos * furnished to do so, subject to the following conditions:
9*5f2f4271Schristos *
10*5f2f4271Schristos * The above copyright notice and this permission notice shall be included in
11*5f2f4271Schristos * all copies or substantial portions of the Software.
12*5f2f4271Schristos *
13*5f2f4271Schristos * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14*5f2f4271Schristos * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15*5f2f4271Schristos * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16*5f2f4271Schristos * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17*5f2f4271Schristos * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18*5f2f4271Schristos * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
19*5f2f4271Schristos * IN THE SOFTWARE.
20*5f2f4271Schristos */
21*5f2f4271Schristos
22*5f2f4271Schristos #include <errno.h>
23*5f2f4271Schristos
24*5f2f4271Schristos #ifndef _WIN32
25*5f2f4271Schristos # include <fcntl.h>
26*5f2f4271Schristos # include <sys/socket.h>
27*5f2f4271Schristos # include <unistd.h>
28*5f2f4271Schristos #endif
29*5f2f4271Schristos
30*5f2f4271Schristos #include "uv.h"
31*5f2f4271Schristos #include "task.h"
32*5f2f4271Schristos
33*5f2f4271Schristos
34*5f2f4271Schristos static int close_cb_called = 0;
35*5f2f4271Schristos
36*5f2f4271Schristos
close_cb(uv_handle_t * handle)37*5f2f4271Schristos static void close_cb(uv_handle_t* handle) {
38*5f2f4271Schristos close_cb_called++;
39*5f2f4271Schristos }
40*5f2f4271Schristos
poll_cb(uv_poll_t * handle,int status,int events)41*5f2f4271Schristos static void poll_cb(uv_poll_t* handle, int status, int events) {
42*5f2f4271Schristos /* Not a bound socket, linux immediately reports UV_READABLE, other OS do not */
43*5f2f4271Schristos ASSERT(events == UV_READABLE);
44*5f2f4271Schristos }
45*5f2f4271Schristos
TEST_IMPL(poll_multiple_handles)46*5f2f4271Schristos TEST_IMPL(poll_multiple_handles) {
47*5f2f4271Schristos uv_os_sock_t sock;
48*5f2f4271Schristos uv_poll_t first_poll_handle, second_poll_handle;
49*5f2f4271Schristos
50*5f2f4271Schristos #ifdef _WIN32
51*5f2f4271Schristos {
52*5f2f4271Schristos struct WSAData wsa_data;
53*5f2f4271Schristos int r = WSAStartup(MAKEWORD(2, 2), &wsa_data);
54*5f2f4271Schristos ASSERT(r == 0);
55*5f2f4271Schristos }
56*5f2f4271Schristos #endif
57*5f2f4271Schristos
58*5f2f4271Schristos sock = socket(AF_INET, SOCK_STREAM, 0);
59*5f2f4271Schristos #ifdef _WIN32
60*5f2f4271Schristos ASSERT(sock != INVALID_SOCKET);
61*5f2f4271Schristos #else
62*5f2f4271Schristos ASSERT(sock != -1);
63*5f2f4271Schristos #endif
64*5f2f4271Schristos ASSERT(0 == uv_poll_init_socket(uv_default_loop(), &first_poll_handle, sock));
65*5f2f4271Schristos ASSERT(0 == uv_poll_init_socket(uv_default_loop(), &second_poll_handle, sock));
66*5f2f4271Schristos
67*5f2f4271Schristos ASSERT(0 == uv_poll_start(&first_poll_handle, UV_READABLE, poll_cb));
68*5f2f4271Schristos
69*5f2f4271Schristos /* We may not start polling while another polling handle is active
70*5f2f4271Schristos * on that fd.
71*5f2f4271Schristos */
72*5f2f4271Schristos #ifndef _WIN32
73*5f2f4271Schristos /* We do not track handles in an O(1) lookupable way on Windows,
74*5f2f4271Schristos * so not checking that here.
75*5f2f4271Schristos */
76*5f2f4271Schristos ASSERT(uv_poll_start(&second_poll_handle, UV_READABLE, poll_cb) == UV_EEXIST);
77*5f2f4271Schristos #endif
78*5f2f4271Schristos
79*5f2f4271Schristos /* After stopping the other polling handle, we now should be able to poll */
80*5f2f4271Schristos ASSERT(0 == uv_poll_stop(&first_poll_handle));
81*5f2f4271Schristos ASSERT(0 == uv_poll_start(&second_poll_handle, UV_READABLE, poll_cb));
82*5f2f4271Schristos
83*5f2f4271Schristos /* Closing an already stopped polling handle is safe in any case */
84*5f2f4271Schristos uv_close((uv_handle_t*) &first_poll_handle, close_cb);
85*5f2f4271Schristos
86*5f2f4271Schristos uv_unref((uv_handle_t*) &second_poll_handle);
87*5f2f4271Schristos ASSERT(0 == uv_run(uv_default_loop(), UV_RUN_DEFAULT));
88*5f2f4271Schristos ASSERT(close_cb_called == 1);
89*5f2f4271Schristos uv_ref((uv_handle_t*) &second_poll_handle);
90*5f2f4271Schristos
91*5f2f4271Schristos ASSERT(uv_is_active((uv_handle_t*) &second_poll_handle));
92*5f2f4271Schristos uv_close((uv_handle_t*) &second_poll_handle, close_cb);
93*5f2f4271Schristos
94*5f2f4271Schristos ASSERT(0 == uv_run(uv_default_loop(), UV_RUN_DEFAULT));
95*5f2f4271Schristos ASSERT(close_cb_called == 2);
96*5f2f4271Schristos
97*5f2f4271Schristos MAKE_VALGRIND_HAPPY();
98*5f2f4271Schristos return 0;
99*5f2f4271Schristos }
100