xref: /netbsd-src/external/mit/libuv/dist/test/test-poll-close-doesnt-corrupt-stack.c (revision 0e552da7216834a96e91ad098e59272b41087480)
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