1*0e552da7Schristos /* Copyright Bert Belder, and other libuv contributors. All rights reserved.
2*0e552da7Schristos *
3*0e552da7Schristos * Permission is hereby granted, free of charge, to any person obtaining a copy
4*0e552da7Schristos * of this software and associated documentation files (the "Software"), to
5*0e552da7Schristos * deal in the Software without restriction, including without limitation the
6*0e552da7Schristos * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7*0e552da7Schristos * sell copies of the Software, and to permit persons to whom the Software is
8*0e552da7Schristos * furnished to do so, subject to the following conditions:
9*0e552da7Schristos *
10*0e552da7Schristos * The above copyright notice and this permission notice shall be included in
11*0e552da7Schristos * all copies or substantial portions of the Software.
12*0e552da7Schristos *
13*0e552da7Schristos * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14*0e552da7Schristos * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15*0e552da7Schristos * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16*0e552da7Schristos * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17*0e552da7Schristos * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18*0e552da7Schristos * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
19*0e552da7Schristos * IN THE SOFTWARE.
20*0e552da7Schristos */
21*0e552da7Schristos
22*0e552da7Schristos #include <errno.h>
23*0e552da7Schristos #include <stdio.h>
24*0e552da7Schristos
25*0e552da7Schristos #include "uv.h"
26*0e552da7Schristos #include "task.h"
27*0e552da7Schristos
28*0e552da7Schristos #ifdef _MSC_VER /* msvc */
29*0e552da7Schristos # define NO_INLINE __declspec(noinline)
30*0e552da7Schristos #else /* gcc */
31*0e552da7Schristos # define NO_INLINE __attribute__ ((noinline))
32*0e552da7Schristos #endif
33*0e552da7Schristos
34*0e552da7Schristos #ifdef _WIN32
35*0e552da7Schristos static uv_os_sock_t sock;
36*0e552da7Schristos static uv_poll_t handle;
37*0e552da7Schristos static int close_cb_called = 0;
38*0e552da7Schristos
39*0e552da7Schristos
close_cb(uv_handle_t * h)40*0e552da7Schristos static void close_cb(uv_handle_t* h) {
41*0e552da7Schristos close_cb_called++;
42*0e552da7Schristos }
43*0e552da7Schristos
44*0e552da7Schristos
poll_cb(uv_poll_t * h,int status,int events)45*0e552da7Schristos static void poll_cb(uv_poll_t* h, int status, int events) {
46*0e552da7Schristos ASSERT(0 && "should never get here");
47*0e552da7Schristos }
48*0e552da7Schristos
49*0e552da7Schristos
close_socket_and_verify_stack(void)50*0e552da7Schristos static void NO_INLINE close_socket_and_verify_stack(void) {
51*0e552da7Schristos const uint32_t MARKER = 0xDEADBEEF;
52*0e552da7Schristos const int VERIFY_AFTER = 10; /* ms */
53*0e552da7Schristos int r;
54*0e552da7Schristos
55*0e552da7Schristos volatile uint32_t data[65536];
56*0e552da7Schristos size_t i;
57*0e552da7Schristos
58*0e552da7Schristos for (i = 0; i < ARRAY_SIZE(data); i++)
59*0e552da7Schristos data[i] = MARKER;
60*0e552da7Schristos
61*0e552da7Schristos r = closesocket(sock);
62*0e552da7Schristos ASSERT(r == 0);
63*0e552da7Schristos
64*0e552da7Schristos uv_sleep(VERIFY_AFTER);
65*0e552da7Schristos
66*0e552da7Schristos for (i = 0; i < ARRAY_SIZE(data); i++)
67*0e552da7Schristos ASSERT(data[i] == MARKER);
68*0e552da7Schristos }
69*0e552da7Schristos #endif
70*0e552da7Schristos
71*0e552da7Schristos
TEST_IMPL(poll_close_doesnt_corrupt_stack)72*0e552da7Schristos TEST_IMPL(poll_close_doesnt_corrupt_stack) {
73*0e552da7Schristos #ifndef _WIN32
74*0e552da7Schristos RETURN_SKIP("Test only relevant on Windows");
75*0e552da7Schristos #else
76*0e552da7Schristos struct WSAData wsa_data;
77*0e552da7Schristos int r;
78*0e552da7Schristos unsigned long on;
79*0e552da7Schristos struct sockaddr_in addr;
80*0e552da7Schristos
81*0e552da7Schristos r = WSAStartup(MAKEWORD(2, 2), &wsa_data);
82*0e552da7Schristos ASSERT(r == 0);
83*0e552da7Schristos
84*0e552da7Schristos sock = socket(AF_INET, SOCK_STREAM, 0);
85*0e552da7Schristos ASSERT(sock != INVALID_SOCKET);
86*0e552da7Schristos on = 1;
87*0e552da7Schristos r = ioctlsocket(sock, FIONBIO, &on);
88*0e552da7Schristos ASSERT(r == 0);
89*0e552da7Schristos
90*0e552da7Schristos r = uv_ip4_addr("127.0.0.1", TEST_PORT, &addr);
91*0e552da7Schristos ASSERT(r == 0);
92*0e552da7Schristos
93*0e552da7Schristos r = connect(sock, (const struct sockaddr*) &addr, sizeof addr);
94*0e552da7Schristos ASSERT(r != 0);
95*0e552da7Schristos ASSERT(WSAGetLastError() == WSAEWOULDBLOCK);
96*0e552da7Schristos
97*0e552da7Schristos r = uv_poll_init_socket(uv_default_loop(), &handle, sock);
98*0e552da7Schristos ASSERT(r == 0);
99*0e552da7Schristos r = uv_poll_start(&handle, UV_READABLE | UV_WRITABLE, poll_cb);
100*0e552da7Schristos ASSERT(r == 0);
101*0e552da7Schristos
102*0e552da7Schristos uv_close((uv_handle_t*) &handle, close_cb);
103*0e552da7Schristos
104*0e552da7Schristos close_socket_and_verify_stack();
105*0e552da7Schristos
106*0e552da7Schristos r = uv_run(uv_default_loop(), UV_RUN_DEFAULT);
107*0e552da7Schristos ASSERT(r == 0);
108*0e552da7Schristos
109*0e552da7Schristos ASSERT(close_cb_called == 1);
110*0e552da7Schristos
111*0e552da7Schristos MAKE_VALGRIND_HAPPY();
112*0e552da7Schristos return 0;
113*0e552da7Schristos #endif
114*0e552da7Schristos }
115