Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 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 "uv.h" |
| 23 | #include "task.h" |
| 24 | |
| 25 | #include <stdio.h> |
| 26 | #include <stdlib.h> |
| 27 | #include <string.h> |
| 28 | |
| 29 | static void connection_cb(uv_stream_t* server, int status); |
| 30 | static void connect_cb(uv_connect_t* req, int status); |
| 31 | static void write_cb(uv_write_t* req, int status); |
| 32 | static void read_cb(uv_stream_t* stream, ssize_t nread, uv_buf_t buf); |
| 33 | static uv_buf_t alloc_cb(uv_handle_t* handle, size_t suggested_size); |
| 34 | |
| 35 | static uv_tcp_t tcp_server; |
| 36 | static uv_tcp_t tcp_client; |
| 37 | static uv_tcp_t tcp_peer; /* client socket as accept()-ed by server */ |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 38 | static uv_connect_t connect_req; |
| 39 | |
| 40 | static int write_cb_called; |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 41 | static int write_cb_error_called; |
| 42 | |
| 43 | typedef struct { |
| 44 | uv_write_t req; |
| 45 | uv_buf_t buf; |
| 46 | } write_req_t; |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 47 | |
| 48 | |
| 49 | static void connection_cb(uv_stream_t* server, int status) { |
| 50 | int r; |
| 51 | |
| 52 | ASSERT(server == (uv_stream_t*)&tcp_server); |
| 53 | ASSERT(status == 0); |
| 54 | |
| 55 | r = uv_tcp_init(server->loop, &tcp_peer); |
| 56 | ASSERT(r == 0); |
| 57 | |
| 58 | r = uv_accept(server, (uv_stream_t*)&tcp_peer); |
| 59 | ASSERT(r == 0); |
| 60 | |
| 61 | r = uv_read_start((uv_stream_t*)&tcp_peer, alloc_cb, read_cb); |
| 62 | ASSERT(r == 0); |
| 63 | } |
| 64 | |
| 65 | |
| 66 | static uv_buf_t alloc_cb(uv_handle_t* handle, size_t suggested_size) { |
| 67 | static char slab[1024]; |
| 68 | return uv_buf_init(slab, sizeof slab); |
| 69 | } |
| 70 | |
| 71 | |
| 72 | static void read_cb(uv_stream_t* stream, ssize_t nread, uv_buf_t buf) { |
| 73 | uv_close((uv_handle_t*)&tcp_server, NULL); |
| 74 | uv_close((uv_handle_t*)&tcp_peer, NULL); |
| 75 | } |
| 76 | |
| 77 | |
| 78 | static void connect_cb(uv_connect_t* req, int status) { |
| 79 | uv_buf_t buf; |
| 80 | size_t size; |
| 81 | char* data; |
| 82 | int r; |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 83 | write_req_t* wr; |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 84 | |
| 85 | ASSERT(req == &connect_req); |
| 86 | ASSERT(status == 0); |
| 87 | |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 88 | while (1) { |
| 89 | size = 10 * 1024 * 1024; |
| 90 | data = malloc(size); |
| 91 | ASSERT(data != NULL); |
| 92 | |
| 93 | memset(data, '$', size); |
| 94 | buf = uv_buf_init(data, size); |
| 95 | |
| 96 | wr = (write_req_t*) malloc(sizeof *wr); |
| 97 | wr->buf = buf; |
| 98 | wr->req.data = data; |
| 99 | |
| 100 | r = uv_write(&(wr->req), req->handle, &wr->buf, 1, write_cb); |
| 101 | ASSERT(r == 0); |
| 102 | |
Ben Noordhuis | 039fac6 | 2012-05-17 05:13:29 | [diff] [blame^] | 103 | if (req->handle->write_queue_size >= size * 2) { |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 104 | break; |
| 105 | } |
| 106 | } |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 107 | } |
| 108 | |
| 109 | |
| 110 | static void write_cb(uv_write_t* req, int status) { |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 111 | write_req_t* wr; |
| 112 | wr = (write_req_t*) req; |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 113 | |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 114 | if (status == -1) { |
| 115 | write_cb_error_called++; |
| 116 | } |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 117 | |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 118 | if (req->handle->write_queue_size == 0) { |
| 119 | uv_close((uv_handle_t*)&tcp_client, NULL); |
| 120 | } |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 121 | |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 122 | free(wr->buf.base); |
| 123 | free(wr); |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 124 | |
| 125 | write_cb_called++; |
| 126 | } |
| 127 | |
| 128 | |
| 129 | /* |
| 130 | * Assert that a failing write does not leave |
| 131 | * the stream's write_queue_size in an inconsistent state. |
| 132 | */ |
| 133 | TEST_IMPL(tcp_write_error) { |
| 134 | uv_loop_t* loop; |
| 135 | int r; |
| 136 | |
| 137 | loop = uv_default_loop(); |
| 138 | ASSERT(loop != NULL); |
| 139 | |
| 140 | r = uv_tcp_init(loop, &tcp_server); |
| 141 | ASSERT(r == 0); |
| 142 | |
| 143 | r = uv_tcp_bind(&tcp_server, uv_ip4_addr("127.0.0.1", TEST_PORT)); |
| 144 | ASSERT(r == 0); |
| 145 | |
| 146 | r = uv_listen((uv_stream_t*)&tcp_server, 1, connection_cb); |
| 147 | ASSERT(r == 0); |
| 148 | |
| 149 | r = uv_tcp_init(loop, &tcp_client); |
| 150 | ASSERT(r == 0); |
| 151 | |
| 152 | r = uv_tcp_connect(&connect_req, |
| 153 | &tcp_client, |
| 154 | uv_ip4_addr("127.0.0.1", TEST_PORT), |
| 155 | connect_cb); |
| 156 | ASSERT(r == 0); |
| 157 | |
| 158 | ASSERT(write_cb_called == 0); |
| 159 | |
| 160 | r = uv_run(loop); |
| 161 | ASSERT(r == 0); |
| 162 | |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 163 | ASSERT(write_cb_called > 0); |
Ben Noordhuis | 039fac6 | 2012-05-17 05:13:29 | [diff] [blame^] | 164 | ASSERT(write_cb_error_called >= 1); |
Bert Belder | c06af22 | 2011-09-27 14:16:59 | [diff] [blame] | 165 | ASSERT(tcp_client.write_queue_size == 0); |
Ryan Dahl | d750927 | 2011-09-16 23:19:18 | [diff] [blame] | 166 | |
| 167 | return 0; |
| 168 | } |