1/***************************************************************************
2 * _ _ ____ _
3 * Project ___| | | | _ \| |
4 * / __| | | | |_) | |
5 * | (__| |_| | _ <| |___
6 * \___|\___/|_| \_\_____|
7 *
8 * Copyright (C) 1998 - 2016, Daniel Stenberg, <daniel@haxx.se>, et al.
9 *
10 * This software is licensed as described in the file COPYING, which
11 * you should have received as part of this distribution. The terms
12 * are also available at https://curl.haxx.se/docs/copyright.html.
13 *
14 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15 * copies of the Software, and permit persons to whom the Software is
16 * furnished to do so, under the terms of the COPYING file.
17 *
18 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19 * KIND, either express or implied.
20 *
21 ***************************************************************************/
22
23/* !checksrc! disable ASSIGNWITHINCONDITION all */
24
25/* Now include the curl_setup.h file from libcurl's private libdir (the source
26 version, but that might include "curl_config.h" from the build dir so we
27 need both of them in the include path), so that we get good in-depth
28 knowledge about the system we're building this on */
29
30#define CURL_NO_OLDIES
31
32#include "curl_setup.h"
33
34#include <curl/curl.h>
35
36#ifdef HAVE_SYS_SELECT_H
37/* since so many tests use select(), we can just as well include it here */
38#include <sys/select.h>
39#endif
40
41#ifdef TPF
42# include "select.h"
43#endif
44
45#include "curl_printf.h"
46
47#define test_setopt(A,B,C) \
48 if((res = curl_easy_setopt((A), (B), (C))) != CURLE_OK) goto test_cleanup
49
50#define test_multi_setopt(A,B,C) \
51 if((res = curl_multi_setopt((A), (B), (C))) != CURLE_OK) goto test_cleanup
52
53extern char *libtest_arg2; /* set by first.c to the argv[2] or NULL */
54extern char *libtest_arg3; /* set by first.c to the argv[3] or NULL */
55
56/* argc and argv as passed in to the main() function */
57extern int test_argc;
58extern char **test_argv;
59
60extern struct timeval tv_test_start; /* for test timing */
61
62extern int select_wrapper(int nfds, fd_set *rd, fd_set *wr, fd_set *exc,
63 struct timeval *tv);
64
65extern void wait_ms(int ms); /* wait this many milliseconds */
66
67extern int test(char *URL); /* the actual test function provided by each
68 individual libXXX.c file */
69
70extern char *hexdump(const unsigned char *buffer, size_t len);
71
72#ifdef UNITTESTS
73extern int unitfail;
74#endif
75
76/*
77** TEST_ERR_* values must be greater than CURL_LAST CURLcode in order
78** to avoid confusion with any CURLcode or CURLMcode. These TEST_ERR_*
79** codes are returned to signal test specific situations and should
80** not get mixed with CURLcode or CURLMcode values.
81**
82** For portability reasons TEST_ERR_* values should be less than 127.
83*/
84
85#define TEST_ERR_MAJOR_BAD 126
86#define TEST_ERR_RUNS_FOREVER 125
87#define TEST_ERR_EASY_INIT 124
88#define TEST_ERR_MULTI_INIT 123
89#define TEST_ERR_NUM_HANDLES 122
90#define TEST_ERR_SELECT 121
91#define TEST_ERR_SUCCESS 120
92#define TEST_ERR_FAILURE 119
93#define TEST_ERR_USAGE 118
94#define TEST_ERR_FOPEN 117
95#define TEST_ERR_FSTAT 116
96#define TEST_ERR_BAD_TIMEOUT 115
97
98/*
99** Macros for test source code readability/maintainability.
100**
101** All of the following macros require that an int data type 'res' variable
102** exists in scope where macro is used, and that it has been initialized to
103** zero before the macro is used.
104**
105** exe_* and chk_* macros are helper macros not intended to be used from
106** outside of this header file. Arguments 'Y' and 'Z' of these represent
107** source code file and line number, while Arguments 'A', 'B', etc, are
108** the arguments used to actually call a libcurl function.
109**
110** All easy_* and multi_* macros call a libcurl function and evaluate if
111** the function has succeeded or failed. When the function succeeds 'res'
112** variable is not set nor cleared and program continues normal flow. On
113** the other hand if function fails 'res' variable is set and a jump to
114** label 'test_cleanup' is performed.
115**
116** Every easy_* and multi_* macros have a res_easy_* and res_multi_* macro
117** counterpart that operates in the same way with the exception that no
118** jump takes place in case of failure. res_easy_* and res_multi_* macros
119** should be immediately followed by checking if 'res' variable has been
120** set.
121**
122** 'res' variable when set will hold a CURLcode, CURLMcode, or any of the
123** TEST_ERR_* values defined above. It is advisable to return this value
124** as test result.
125*/
126
127/* ---------------------------------------------------------------- */
128
129#define exe_easy_init(A,Y,Z) do { \
130 if(((A) = curl_easy_init()) == NULL) { \
131 fprintf(stderr, "%s:%d curl_easy_init() failed\n", (Y), (Z)); \
132 res = TEST_ERR_EASY_INIT; \
133 } \
134} WHILE_FALSE
135
136#define res_easy_init(A) \
137 exe_easy_init((A), (__FILE__), (__LINE__))
138
139#define chk_easy_init(A,Y,Z) do { \
140 exe_easy_init((A), (Y), (Z)); \
141 if(res) \
142 goto test_cleanup; \
143} WHILE_FALSE
144
145#define easy_init(A) \
146 chk_easy_init((A), (__FILE__), (__LINE__))
147
148/* ---------------------------------------------------------------- */
149
150#define exe_multi_init(A,Y,Z) do { \
151 if(((A) = curl_multi_init()) == NULL) { \
152 fprintf(stderr, "%s:%d curl_multi_init() failed\n", (Y), (Z)); \
153 res = TEST_ERR_MULTI_INIT; \
154 } \
155} WHILE_FALSE
156
157#define res_multi_init(A) \
158 exe_multi_init((A), (__FILE__), (__LINE__))
159
160#define chk_multi_init(A,Y,Z) do { \
161 exe_multi_init((A), (Y), (Z)); \
162 if(res) \
163 goto test_cleanup; \
164} WHILE_FALSE
165
166#define multi_init(A) \
167 chk_multi_init((A), (__FILE__), (__LINE__))
168
169/* ---------------------------------------------------------------- */
170
171#define exe_easy_setopt(A,B,C,Y,Z) do { \
172 CURLcode ec; \
173 if((ec = curl_easy_setopt((A), (B), (C))) != CURLE_OK) { \
174 fprintf(stderr, "%s:%d curl_easy_setopt() failed, " \
175 "with code %d (%s)\n", \
176 (Y), (Z), (int)ec, curl_easy_strerror(ec)); \
177 res = (int)ec; \
178 } \
179} WHILE_FALSE
180
181#define res_easy_setopt(A, B, C) \
182 exe_easy_setopt((A), (B), (C), (__FILE__), (__LINE__))
183
184#define chk_easy_setopt(A, B, C, Y, Z) do { \
185 exe_easy_setopt((A), (B), (C), (Y), (Z)); \
186 if(res) \
187 goto test_cleanup; \
188} WHILE_FALSE
189
190#define easy_setopt(A, B, C) \
191 chk_easy_setopt((A), (B), (C), (__FILE__), (__LINE__))
192
193/* ---------------------------------------------------------------- */
194
195#define exe_multi_setopt(A, B, C, Y, Z) do { \
196 CURLMcode ec; \
197 if((ec = curl_multi_setopt((A), (B), (C))) != CURLM_OK) { \
198 fprintf(stderr, "%s:%d curl_multi_setopt() failed, " \
199 "with code %d (%s)\n", \
200 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
201 res = (int)ec; \
202 } \
203} WHILE_FALSE
204
205#define res_multi_setopt(A,B,C) \
206 exe_multi_setopt((A), (B), (C), (__FILE__), (__LINE__))
207
208#define chk_multi_setopt(A,B,C,Y,Z) do { \
209 exe_multi_setopt((A), (B), (C), (Y), (Z)); \
210 if(res) \
211 goto test_cleanup; \
212} WHILE_FALSE
213
214#define multi_setopt(A,B,C) \
215 chk_multi_setopt((A), (B), (C), (__FILE__), (__LINE__))
216
217/* ---------------------------------------------------------------- */
218
219#define exe_multi_add_handle(A,B,Y,Z) do { \
220 CURLMcode ec; \
221 if((ec = curl_multi_add_handle((A), (B))) != CURLM_OK) { \
222 fprintf(stderr, "%s:%d curl_multi_add_handle() failed, " \
223 "with code %d (%s)\n", \
224 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
225 res = (int)ec; \
226 } \
227} WHILE_FALSE
228
229#define res_multi_add_handle(A, B) \
230 exe_multi_add_handle((A), (B), (__FILE__), (__LINE__))
231
232#define chk_multi_add_handle(A, B, Y, Z) do { \
233 exe_multi_add_handle((A), (B), (Y), (Z)); \
234 if(res) \
235 goto test_cleanup; \
236} WHILE_FALSE
237
238#define multi_add_handle(A, B) \
239 chk_multi_add_handle((A), (B), (__FILE__), (__LINE__))
240
241/* ---------------------------------------------------------------- */
242
243#define exe_multi_remove_handle(A,B,Y,Z) do { \
244 CURLMcode ec; \
245 if((ec = curl_multi_remove_handle((A), (B))) != CURLM_OK) { \
246 fprintf(stderr, "%s:%d curl_multi_remove_handle() failed, " \
247 "with code %d (%s)\n", \
248 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
249 res = (int)ec; \
250 } \
251} WHILE_FALSE
252
253#define res_multi_remove_handle(A, B) \
254 exe_multi_remove_handle((A), (B), (__FILE__), (__LINE__))
255
256#define chk_multi_remove_handle(A, B, Y, Z) do { \
257 exe_multi_remove_handle((A), (B), (Y), (Z)); \
258 if(res) \
259 goto test_cleanup; \
260} WHILE_FALSE
261
262
263#define multi_remove_handle(A, B) \
264 chk_multi_remove_handle((A), (B), (__FILE__), (__LINE__))
265
266/* ---------------------------------------------------------------- */
267
268#define exe_multi_perform(A,B,Y,Z) do { \
269 CURLMcode ec; \
270 if((ec = curl_multi_perform((A), (B))) != CURLM_OK) { \
271 fprintf(stderr, "%s:%d curl_multi_perform() failed, " \
272 "with code %d (%s)\n", \
273 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
274 res = (int)ec; \
275 } \
276 else if(*((B)) < 0) { \
277 fprintf(stderr, "%s:%d curl_multi_perform() succeeded, " \
278 "but returned invalid running_handles value (%d)\n", \
279 (Y), (Z), (int)*((B))); \
280 res = TEST_ERR_NUM_HANDLES; \
281 } \
282} WHILE_FALSE
283
284#define res_multi_perform(A, B) \
285 exe_multi_perform((A), (B), (__FILE__), (__LINE__))
286
287#define chk_multi_perform(A, B, Y, Z) do { \
288 exe_multi_perform((A), (B), (Y), (Z)); \
289 if(res) \
290 goto test_cleanup; \
291} WHILE_FALSE
292
293#define multi_perform(A,B) \
294 chk_multi_perform((A), (B), (__FILE__), (__LINE__))
295
296/* ---------------------------------------------------------------- */
297
298#define exe_multi_fdset(A, B, C, D, E, Y, Z) do { \
299 CURLMcode ec; \
300 if((ec = curl_multi_fdset((A), (B), (C), (D), (E))) != CURLM_OK) { \
301 fprintf(stderr, "%s:%d curl_multi_fdset() failed, " \
302 "with code %d (%s)\n", \
303 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
304 res = (int)ec; \
305 } \
306 else if(*((E)) < -1) { \
307 fprintf(stderr, "%s:%d curl_multi_fdset() succeeded, " \
308 "but returned invalid max_fd value (%d)\n", \
309 (Y), (Z), (int)*((E))); \
310 res = TEST_ERR_NUM_HANDLES; \
311 } \
312} WHILE_FALSE
313
314#define res_multi_fdset(A, B, C, D, E) \
315 exe_multi_fdset((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
316
317#define chk_multi_fdset(A, B, C, D, E, Y, Z) do { \
318 exe_multi_fdset((A), (B), (C), (D), (E), (Y), (Z)); \
319 if(res) \
320 goto test_cleanup; \
321 } WHILE_FALSE
322
323#define multi_fdset(A, B, C, D, E) \
324 chk_multi_fdset((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
325
326/* ---------------------------------------------------------------- */
327
328#define exe_multi_timeout(A,B,Y,Z) do { \
329 CURLMcode ec; \
330 if((ec = curl_multi_timeout((A), (B))) != CURLM_OK) { \
331 fprintf(stderr, "%s:%d curl_multi_timeout() failed, " \
332 "with code %d (%s)\n", \
333 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
334 res = (int)ec; \
335 } \
336 else if(*((B)) < -1L) { \
337 fprintf(stderr, "%s:%d curl_multi_timeout() succeeded, " \
338 "but returned invalid timeout value (%ld)\n", \
339 (Y), (Z), (long)*((B))); \
340 res = TEST_ERR_BAD_TIMEOUT; \
341 } \
342} WHILE_FALSE
343
344#define res_multi_timeout(A, B) \
345 exe_multi_timeout((A), (B), (__FILE__), (__LINE__))
346
347#define chk_multi_timeout(A, B, Y, Z) do { \
348 exe_multi_timeout((A), (B), (Y), (Z)); \
349 if(res) \
350 goto test_cleanup; \
351 } WHILE_FALSE
352
353#define multi_timeout(A, B) \
354 chk_multi_timeout((A), (B), (__FILE__), (__LINE__))
355
356/* ---------------------------------------------------------------- */
357
358#define exe_multi_poll(A,B,C,D,E,Y,Z) do { \
359 CURLMcode ec; \
360 if((ec = curl_multi_poll((A), (B), (C), (D), (E))) != CURLM_OK) { \
361 fprintf(stderr, "%s:%d curl_multi_poll() failed, " \
362 "with code %d (%s)\n", \
363 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
364 res = (int)ec; \
365 } \
366 else if(*((E)) < 0) { \
367 fprintf(stderr, "%s:%d curl_multi_poll() succeeded, " \
368 "but returned invalid numfds value (%d)\n", \
369 (Y), (Z), (int)*((E))); \
370 res = TEST_ERR_NUM_HANDLES; \
371 } \
372} WHILE_FALSE
373
374#define res_multi_poll(A, B, C, D, E) \
375 exe_multi_poll((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
376
377#define chk_multi_poll(A, B, C, D, E, Y, Z) do { \
378 exe_multi_poll((A), (B), (C), (D), (E), (Y), (Z)); \
379 if(res) \
380 goto test_cleanup; \
381} WHILE_FALSE
382
383#define multi_poll(A, B, C, D, E) \
384 chk_multi_poll((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
385
386/* ---------------------------------------------------------------- */
387
388#define exe_multi_wakeup(A,Y,Z) do { \
389 CURLMcode ec; \
390 if((ec = curl_multi_wakeup((A))) != CURLM_OK) { \
391 fprintf(stderr, "%s:%d curl_multi_wakeup() failed, " \
392 "with code %d (%s)\n", \
393 (Y), (Z), (int)ec, curl_multi_strerror(ec)); \
394 res = (int)ec; \
395 } \
396} WHILE_FALSE
397
398#define res_multi_wakeup(A) \
399 exe_multi_wakeup((A), (__FILE__), (__LINE__))
400
401#define chk_multi_wakeup(A, Y, Z) do { \
402 exe_multi_wakeup((A), (Y), (Z)); \
403 if(res) \
404 goto test_cleanup; \
405} WHILE_FALSE
406
407#define multi_wakeup(A) \
408 chk_multi_wakeup((A), (__FILE__), (__LINE__))
409
410/* ---------------------------------------------------------------- */
411
412#define exe_select_test(A, B, C, D, E, Y, Z) do { \
413 int ec; \
414 if(select_wrapper((A), (B), (C), (D), (E)) == -1) { \
415 ec = SOCKERRNO; \
416 fprintf(stderr, "%s:%d select() failed, with " \
417 "errno %d (%s)\n", \
418 (Y), (Z), ec, strerror(ec)); \
419 res = TEST_ERR_SELECT; \
420 } \
421 } WHILE_FALSE
422
423#define res_select_test(A, B, C, D, E) \
424 exe_select_test((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
425
426#define chk_select_test(A, B, C, D, E, Y, Z) do { \
427 exe_select_test((A), (B), (C), (D), (E), (Y), (Z)); \
428 if(res) \
429 goto test_cleanup; \
430 } WHILE_FALSE
431
432#define select_test(A, B, C, D, E) \
433 chk_select_test((A), (B), (C), (D), (E), (__FILE__), (__LINE__))
434
435/* ---------------------------------------------------------------- */
436
437#define start_test_timing() do { \
438 tv_test_start = tutil_tvnow(); \
439} WHILE_FALSE
440
441#define exe_test_timedout(Y,Z) do { \
442 if(tutil_tvdiff(tutil_tvnow(), tv_test_start) > TEST_HANG_TIMEOUT) { \
443 fprintf(stderr, "%s:%d ABORTING TEST, since it seems " \
444 "that it would have run forever.\n", (Y), (Z)); \
445 res = TEST_ERR_RUNS_FOREVER; \
446 } \
447} WHILE_FALSE
448
449#define res_test_timedout() \
450 exe_test_timedout((__FILE__), (__LINE__))
451
452#define chk_test_timedout(Y, Z) do { \
453 exe_test_timedout(Y, Z); \
454 if(res) \
455 goto test_cleanup; \
456 } WHILE_FALSE
457
458#define abort_on_test_timeout() \
459 chk_test_timedout((__FILE__), (__LINE__))
460
461/* ---------------------------------------------------------------- */
462
463#define exe_global_init(A,Y,Z) do { \
464 CURLcode ec; \
465 if((ec = curl_global_init((A))) != CURLE_OK) { \
466 fprintf(stderr, "%s:%d curl_global_init() failed, " \
467 "with code %d (%s)\n", \
468 (Y), (Z), (int)ec, curl_easy_strerror(ec)); \
469 res = (int)ec; \
470 } \
471} WHILE_FALSE
472
473#define res_global_init(A) \
474 exe_global_init((A), (__FILE__), (__LINE__))
475
476#define chk_global_init(A, Y, Z) do { \
477 exe_global_init((A), (Y), (Z)); \
478 if(res) \
479 return res; \
480 } WHILE_FALSE
481
482/* global_init() is different than other macros. In case of
483 failure it 'return's instead of going to 'test_cleanup'. */
484
485#define global_init(A) \
486 chk_global_init((A), (__FILE__), (__LINE__))
487
488/* ---------------------------------------------------------------- */
489