remove the CVSish $Id$ lines
[platform/upstream/curl.git] / include / curl / typecheck-gcc.h
1 #ifndef __CURL_TYPECHECK_GCC_H
2 #define __CURL_TYPECHECK_GCC_H
3 /***************************************************************************
4  *                                  _   _ ____  _
5  *  Project                     ___| | | |  _ \| |
6  *                             / __| | | | |_) | |
7  *                            | (__| |_| |  _ <| |___
8  *                             \___|\___/|_| \_\_____|
9  *
10  * Copyright (C) 1998 - 2009, Daniel Stenberg, <daniel@haxx.se>, et al.
11  *
12  * This software is licensed as described in the file COPYING, which
13  * you should have received as part of this distribution. The terms
14  * are also available at http://curl.haxx.se/docs/copyright.html.
15  *
16  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
17  * copies of the Software, and permit persons to whom the Software is
18  * furnished to do so, under the terms of the COPYING file.
19  *
20  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
21  * KIND, either express or implied.
22  *
23  ***************************************************************************/
24
25 /* wraps curl_easy_setopt() with typechecking */
26
27 /* To add a new kind of warning, add an
28  *   if(_curl_is_sometype_option(_curl_opt) && ! _curl_is_sometype(value))
29  *     _curl_easy_setopt_err_sometype();
30  * block and define _curl_is_sometype_option, _curl_is_sometype and
31  * _curl_easy_setopt_err_sometype below
32  *
33  * To add an option that uses the same type as an existing option, you'll just
34  * need to extend the appropriate _curl_*_option macro
35  */
36 #define curl_easy_setopt(handle, option, value)                               \
37 __extension__ ({                                                              \
38   __typeof__ (option) _curl_opt = option;                                     \
39   if (__builtin_constant_p(_curl_opt)) {                                      \
40     if (_curl_is_long_option(_curl_opt) && !_curl_is_long(value))             \
41       _curl_easy_setopt_err_long();                                           \
42     if (_curl_is_off_t_option(_curl_opt) && !_curl_is_off_t(value))           \
43       _curl_easy_setopt_err_curl_off_t();                                     \
44     if (_curl_is_string_option(_curl_opt) && !_curl_is_string(value))         \
45       _curl_easy_setopt_err_string();                                         \
46     if (_curl_is_write_cb_option(_curl_opt) && !_curl_is_write_cb(value))     \
47       _curl_easy_setopt_err_write_callback();                                 \
48     if ((_curl_opt) == CURLOPT_READFUNCTION && !_curl_is_read_cb(value))      \
49       _curl_easy_setopt_err_read_cb();                                        \
50     if ((_curl_opt) == CURLOPT_IOCTLFUNCTION && !_curl_is_ioctl_cb(value))    \
51       _curl_easy_setopt_err_ioctl_cb();                                       \
52     if ((_curl_opt) == CURLOPT_SOCKOPTFUNCTION && !_curl_is_sockopt_cb(value))\
53       _curl_easy_setopt_err_sockopt_cb();                                     \
54     if ((_curl_opt) == CURLOPT_OPENSOCKETFUNCTION &&                          \
55             !_curl_is_opensocket_cb(value))                                   \
56       _curl_easy_setopt_err_opensocket_cb();                                  \
57     if ((_curl_opt) == CURLOPT_PROGRESSFUNCTION &&                            \
58             !_curl_is_progress_cb(value))                                     \
59       _curl_easy_setopt_err_progress_cb();                                    \
60     if ((_curl_opt) == CURLOPT_DEBUGFUNCTION && !_curl_is_debug_cb(value))    \
61       _curl_easy_setopt_err_debug_cb();                                       \
62     if ((_curl_opt) == CURLOPT_SSL_CTX_FUNCTION &&                            \
63             !_curl_is_ssl_ctx_cb(value))                                      \
64       _curl_easy_setopt_err_ssl_ctx_cb();                                     \
65     if (_curl_is_conv_cb_option(_curl_opt) && !_curl_is_conv_cb(value))       \
66       _curl_easy_setopt_err_conv_cb();                                        \
67     if ((_curl_opt) == CURLOPT_SEEKFUNCTION && !_curl_is_seek_cb(value))      \
68       _curl_easy_setopt_err_seek_cb();                                        \
69     if (_curl_is_cb_data_option(_curl_opt) && !_curl_is_cb_data(value))       \
70       _curl_easy_setopt_err_cb_data();                                        \
71     if ((_curl_opt) == CURLOPT_ERRORBUFFER && !_curl_is_error_buffer(value))  \
72       _curl_easy_setopt_err_error_buffer();                                   \
73     if ((_curl_opt) == CURLOPT_STDERR && !_curl_is_FILE(value))               \
74       _curl_easy_setopt_err_FILE();                                           \
75     if (_curl_is_postfields_option(_curl_opt) && !_curl_is_postfields(value)) \
76       _curl_easy_setopt_err_postfields();                                     \
77     if ((_curl_opt) == CURLOPT_HTTPPOST &&                                    \
78             !_curl_is_arr((value), struct curl_httppost))                     \
79       _curl_easy_setopt_err_curl_httpost();                                   \
80     if (_curl_is_slist_option(_curl_opt) &&                                   \
81             !_curl_is_arr((value), struct curl_slist))                        \
82       _curl_easy_setopt_err_curl_slist();                                     \
83     if ((_curl_opt) == CURLOPT_SHARE && !_curl_is_ptr((value), CURLSH))       \
84       _curl_easy_setopt_err_CURLSH();                                         \
85   }                                                                           \
86   curl_easy_setopt(handle, _curl_opt, value);                                 \
87 })
88
89 /* wraps curl_easy_getinfo() with typechecking */
90 /* FIXME: don't allow const pointers */
91 #define curl_easy_getinfo(handle, info, arg)                                  \
92 __extension__ ({                                                              \
93   __typeof__ (info) _curl_info = info;                                        \
94   if (__builtin_constant_p(_curl_info)) {                                     \
95     if (_curl_is_string_info(_curl_info) && !_curl_is_arr((arg), char *))     \
96       _curl_easy_getinfo_err_string();                                        \
97     if (_curl_is_long_info(_curl_info) && !_curl_is_arr((arg), long))         \
98       _curl_easy_getinfo_err_long();                                          \
99     if (_curl_is_double_info(_curl_info) && !_curl_is_arr((arg), double))     \
100       _curl_easy_getinfo_err_double();                                        \
101     if (_curl_is_slist_info(_curl_info) &&                                    \
102            !_curl_is_arr((arg), struct curl_slist *))                         \
103       _curl_easy_getinfo_err_curl_slist();                                    \
104   }                                                                           \
105   curl_easy_getinfo(handle, _curl_info, arg);                                 \
106 })
107
108 /* TODO: typechecking for curl_share_setopt() and curl_multi_setopt(),
109  * for now just make sure that the functions are called with three
110  * arguments
111  */
112 #define curl_share_setopt(share,opt,param) curl_share_setopt(share,opt,param)
113 #define curl_multi_setopt(handle,opt,param) curl_multi_setopt(handle,opt,param)
114
115
116 /* the actual warnings, triggered by calling the _curl_easy_setopt_err*
117  * functions */
118
119 /* To define a new warning, use _CURL_WARNING(identifier, "message") */
120 #define _CURL_WARNING(id, message)                                            \
121   static void __attribute__((warning(message))) __attribute__((unused))       \
122   __attribute__((noinline)) id(void) { __asm__(""); }
123
124 _CURL_WARNING(_curl_easy_setopt_err_long,
125   "curl_easy_setopt expects a long argument for this option")
126 _CURL_WARNING(_curl_easy_setopt_err_curl_off_t,
127   "curl_easy_setopt expects a curl_off_t argument for this option")
128 _CURL_WARNING(_curl_easy_setopt_err_string,
129   "curl_easy_setopt expects a string (char* or char[]) argument for this option"
130   )
131 _CURL_WARNING(_curl_easy_setopt_err_write_callback,
132   "curl_easy_setopt expects a curl_write_callback argument for this option")
133 _CURL_WARNING(_curl_easy_setopt_err_read_cb,
134   "curl_easy_setopt expects a curl_read_callback argument for this option")
135 _CURL_WARNING(_curl_easy_setopt_err_ioctl_cb,
136   "curl_easy_setopt expects a curl_ioctl_callback argument for this option")
137 _CURL_WARNING(_curl_easy_setopt_err_sockopt_cb,
138   "curl_easy_setopt expects a curl_sockopt_callback argument for this option")
139 _CURL_WARNING(_curl_easy_setopt_err_opensocket_cb,
140   "curl_easy_setopt expects a curl_opensocket_callback argument for this option"
141   )
142 _CURL_WARNING(_curl_easy_setopt_err_progress_cb,
143   "curl_easy_setopt expects a curl_progress_callback argument for this option")
144 _CURL_WARNING(_curl_easy_setopt_err_debug_cb,
145   "curl_easy_setopt expects a curl_debug_callback argument for this option")
146 _CURL_WARNING(_curl_easy_setopt_err_ssl_ctx_cb,
147   "curl_easy_setopt expects a curl_ssl_ctx_callback argument for this option")
148 _CURL_WARNING(_curl_easy_setopt_err_conv_cb,
149   "curl_easy_setopt expects a curl_conv_callback argument for this option")
150 _CURL_WARNING(_curl_easy_setopt_err_seek_cb,
151   "curl_easy_setopt expects a curl_seek_callback argument for this option")
152 _CURL_WARNING(_curl_easy_setopt_err_cb_data,
153   "curl_easy_setopt expects a private data pointer as argument for this option")
154 _CURL_WARNING(_curl_easy_setopt_err_error_buffer,
155   "curl_easy_setopt expects a char buffer of CURL_ERROR_SIZE as argument for this option")
156 _CURL_WARNING(_curl_easy_setopt_err_FILE,
157   "curl_easy_setopt expects a FILE* argument for this option")
158 _CURL_WARNING(_curl_easy_setopt_err_postfields,
159   "curl_easy_setopt expects a void* or char* argument for this option")
160 _CURL_WARNING(_curl_easy_setopt_err_curl_httpost,
161   "curl_easy_setopt expects a struct curl_httppost* argument for this option")
162 _CURL_WARNING(_curl_easy_setopt_err_curl_slist,
163   "curl_easy_setopt expects a struct curl_slist* argument for this option")
164 _CURL_WARNING(_curl_easy_setopt_err_CURLSH,
165   "curl_easy_setopt expects a CURLSH* argument for this option")
166
167 _CURL_WARNING(_curl_easy_getinfo_err_string,
168   "curl_easy_getinfo expects a pointer to char * for this info")
169 _CURL_WARNING(_curl_easy_getinfo_err_long,
170   "curl_easy_getinfo expects a pointer to long for this info")
171 _CURL_WARNING(_curl_easy_getinfo_err_double,
172   "curl_easy_getinfo expects a pointer to double for this info")
173 _CURL_WARNING(_curl_easy_getinfo_err_curl_slist,
174   "curl_easy_getinfo expects a pointer to struct curl_slist * for this info")
175
176 /* groups of curl_easy_setops options that take the same type of argument */
177
178 /* To add a new option to one of the groups, just add
179  *   (option) == CURLOPT_SOMETHING
180  * to the or-expression. If the option takes a long or curl_off_t, you don't
181  * have to do anything
182  */
183
184 /* evaluates to true if option takes a long argument */
185 #define _curl_is_long_option(option)                                          \
186   (0 < (option) && (option) < CURLOPTTYPE_OBJECTPOINT)
187
188 #define _curl_is_off_t_option(option)                                         \
189   ((option) > CURLOPTTYPE_OFF_T)
190
191 /* evaluates to true if option takes a char* argument */
192 #define _curl_is_string_option(option)                                        \
193   ((option) == CURLOPT_URL ||                                                 \
194    (option) == CURLOPT_PROXY ||                                               \
195    (option) == CURLOPT_INTERFACE ||                                           \
196    (option) == CURLOPT_NETRC_FILE ||                                          \
197    (option) == CURLOPT_USERPWD ||                                             \
198    (option) == CURLOPT_USERNAME ||                                            \
199    (option) == CURLOPT_PASSWORD ||                                            \
200    (option) == CURLOPT_PROXYUSERPWD ||                                        \
201    (option) == CURLOPT_PROXYUSERNAME ||                                       \
202    (option) == CURLOPT_PROXYPASSWORD ||                                       \
203    (option) == CURLOPT_NOPROXY ||                                             \
204    (option) == CURLOPT_ENCODING ||                                            \
205    (option) == CURLOPT_REFERER ||                                             \
206    (option) == CURLOPT_USERAGENT ||                                           \
207    (option) == CURLOPT_COOKIE ||                                              \
208    (option) == CURLOPT_COOKIEFILE ||                                          \
209    (option) == CURLOPT_COOKIEJAR ||                                           \
210    (option) == CURLOPT_COOKIELIST ||                                          \
211    (option) == CURLOPT_FTPPORT ||                                             \
212    (option) == CURLOPT_FTP_ALTERNATIVE_TO_USER ||                             \
213    (option) == CURLOPT_FTP_ACCOUNT ||                                         \
214    (option) == CURLOPT_RANGE ||                                               \
215    (option) == CURLOPT_CUSTOMREQUEST ||                                       \
216    (option) == CURLOPT_SSLCERT ||                                             \
217    (option) == CURLOPT_SSLCERTTYPE ||                                         \
218    (option) == CURLOPT_SSLKEY ||                                              \
219    (option) == CURLOPT_SSLKEYTYPE ||                                          \
220    (option) == CURLOPT_KEYPASSWD ||                                           \
221    (option) == CURLOPT_SSLENGINE ||                                           \
222    (option) == CURLOPT_CAINFO ||                                              \
223    (option) == CURLOPT_CAPATH ||                                              \
224    (option) == CURLOPT_RANDOM_FILE ||                                         \
225    (option) == CURLOPT_EGDSOCKET ||                                           \
226    (option) == CURLOPT_SSL_CIPHER_LIST ||                                     \
227    (option) == CURLOPT_KRBLEVEL ||                                            \
228    (option) == CURLOPT_SSH_HOST_PUBLIC_KEY_MD5 ||                             \
229    (option) == CURLOPT_SSH_PUBLIC_KEYFILE ||                                  \
230    (option) == CURLOPT_SSH_PRIVATE_KEYFILE ||                                 \
231    (option) == CURLOPT_CRLFILE ||                                             \
232    (option) == CURLOPT_ISSUERCERT ||                                          \
233    0)
234
235 /* evaluates to true if option takes a curl_write_callback argument */
236 #define _curl_is_write_cb_option(option)                                      \
237   ((option) == CURLOPT_HEADERFUNCTION ||                                      \
238    (option) == CURLOPT_WRITEFUNCTION)
239
240 /* evaluates to true if option takes a curl_conv_callback argument */
241 #define _curl_is_conv_cb_option(option)                                       \
242   ((option) == CURLOPT_CONV_TO_NETWORK_FUNCTION ||                            \
243    (option) == CURLOPT_CONV_FROM_NETWORK_FUNCTION ||                          \
244    (option) == CURLOPT_CONV_FROM_UTF8_FUNCTION)
245
246 /* evaluates to true if option takes a data argument to pass to a callback */
247 #define _curl_is_cb_data_option(option)                                       \
248   ((option) == CURLOPT_WRITEDATA ||                                           \
249    (option) == CURLOPT_READDATA ||                                            \
250    (option) == CURLOPT_IOCTLDATA ||                                           \
251    (option) == CURLOPT_SOCKOPTDATA ||                                         \
252    (option) == CURLOPT_OPENSOCKETDATA ||                                      \
253    (option) == CURLOPT_PROGRESSDATA ||                                        \
254    (option) == CURLOPT_WRITEHEADER ||                                         \
255    (option) == CURLOPT_DEBUGDATA ||                                           \
256    (option) == CURLOPT_SSL_CTX_DATA ||                                        \
257    (option) == CURLOPT_SEEKDATA ||                                            \
258    (option) == CURLOPT_PRIVATE ||                                             \
259    0)
260
261 /* evaluates to true if option takes a POST data argument (void* or char*) */
262 #define _curl_is_postfields_option(option)                                    \
263   ((option) == CURLOPT_POSTFIELDS ||                                          \
264    (option) == CURLOPT_COPYPOSTFIELDS ||                                      \
265    0)
266
267 /* evaluates to true if option takes a struct curl_slist * argument */
268 #define _curl_is_slist_option(option)                                         \
269   ((option) == CURLOPT_HTTPHEADER ||                                          \
270    (option) == CURLOPT_HTTP200ALIASES ||                                      \
271    (option) == CURLOPT_QUOTE ||                                               \
272    (option) == CURLOPT_POSTQUOTE ||                                           \
273    (option) == CURLOPT_PREQUOTE ||                                            \
274    (option) == CURLOPT_TELNETOPTIONS ||                                       \
275    0)
276
277 /* groups of curl_easy_getinfo infos that take the same type of argument */
278
279 /* evaluates to true if info expects a pointer to char * argument */
280 #define _curl_is_string_info(info)                                            \
281   (CURLINFO_STRING < (info) && (info) < CURLINFO_LONG)
282
283 /* evaluates to true if info expects a pointer to long argument */
284 #define _curl_is_long_info(info)                                              \
285   (CURLINFO_LONG < (info) && (info) < CURLINFO_DOUBLE)
286
287 /* evaluates to true if info expects a pointer to double argument */
288 #define _curl_is_double_info(info)                                            \
289   (CURLINFO_DOUBLE < (info) && (info) < CURLINFO_SLIST)
290
291 /* true if info expects a pointer to struct curl_slist * argument */
292 #define _curl_is_slist_info(info)                                             \
293   (CURLINFO_SLIST < (info))
294
295
296 /* typecheck helpers -- check whether given expression has requested type*/
297
298 /* For pointers, you can use the _curl_is_ptr/_curl_is_arr macros,
299  * otherwise define a new macro. Search for __builtin_types_compatible_p
300  * in the GCC manual.
301  * NOTE: these macros MUST NOT EVALUATE their arguments! The argument is
302  * the actual expression passed to the curl_easy_setopt macro. This
303  * means that you can only apply the sizeof and __typeof__ operators, no
304  * == or whatsoever.
305  */
306
307 /* XXX: should evaluate to true iff expr is a pointer */
308 #define _curl_is_any_ptr(expr)                                                \
309   (sizeof(expr) == sizeof(void*))
310
311 /* evaluates to true if expr is NULL */
312 /* XXX: must not evaluate expr, so this check is not accurate */
313 #define _curl_is_NULL(expr)                                                   \
314   (__builtin_types_compatible_p(__typeof__(expr), __typeof__(NULL)))
315
316 /* evaluates to true if expr is type*, const type* or NULL */
317 #define _curl_is_ptr(expr, type)                                              \
318   (_curl_is_NULL(expr) ||                                                     \
319    __builtin_types_compatible_p(__typeof__(expr), type *) ||                  \
320    __builtin_types_compatible_p(__typeof__(expr), const type *))
321
322 /* evaluates to true if expr is one of type[], type*, NULL or const type* */
323 #define _curl_is_arr(expr, type)                                              \
324   (_curl_is_ptr((expr), type) ||                                              \
325    __builtin_types_compatible_p(__typeof__(expr), type []))
326
327 /* evaluates to true if expr is a string */
328 #define _curl_is_string(expr)                                                 \
329   (_curl_is_arr((expr), char) ||                                              \
330    _curl_is_arr((expr), signed char) ||                                       \
331    _curl_is_arr((expr), unsigned char))
332
333 /* evaluates to true if expr is a long (no matter the signedness)
334  * XXX: for now, int is also accepted (and therefore short and char, which
335  * are promoted to int when passed to a variadic function) */
336 #define _curl_is_long(expr)                                                   \
337   (__builtin_types_compatible_p(__typeof__(expr), long) ||                    \
338    __builtin_types_compatible_p(__typeof__(expr), signed long) ||             \
339    __builtin_types_compatible_p(__typeof__(expr), unsigned long) ||           \
340    __builtin_types_compatible_p(__typeof__(expr), int) ||                     \
341    __builtin_types_compatible_p(__typeof__(expr), signed int) ||              \
342    __builtin_types_compatible_p(__typeof__(expr), unsigned int) ||            \
343    __builtin_types_compatible_p(__typeof__(expr), short) ||                   \
344    __builtin_types_compatible_p(__typeof__(expr), signed short) ||            \
345    __builtin_types_compatible_p(__typeof__(expr), unsigned short) ||          \
346    __builtin_types_compatible_p(__typeof__(expr), char) ||                    \
347    __builtin_types_compatible_p(__typeof__(expr), signed char) ||             \
348    __builtin_types_compatible_p(__typeof__(expr), unsigned char))
349
350 /* evaluates to true if expr is of type curl_off_t */
351 #define _curl_is_off_t(expr)                                                  \
352   (__builtin_types_compatible_p(__typeof__(expr), curl_off_t))
353
354 /* evaluates to true if expr is abuffer suitable for CURLOPT_ERRORBUFFER */
355 /* XXX: also check size of an char[] array? */
356 #define _curl_is_error_buffer(expr)                                           \
357   (__builtin_types_compatible_p(__typeof__(expr), char *) ||                  \
358    __builtin_types_compatible_p(__typeof__(expr), char[]))
359
360 /* evaluates to true if expr is of type (const) void* or (const) FILE* */
361 #if 0
362 #define _curl_is_cb_data(expr)                                                \
363   (_curl_is_ptr((expr), void) ||                                              \
364    _curl_is_ptr((expr), FILE))
365 #else /* be less strict */
366 #define _curl_is_cb_data(expr)                                                \
367   _curl_is_any_ptr(expr)
368 #endif
369
370 /* evaluates to true if expr is of type FILE* */
371 #define _curl_is_FILE(expr)                                                   \
372   (__builtin_types_compatible_p(__typeof__(expr), FILE *))
373
374 /* evaluates to true if expr can be passed as POST data (void* or char*) */
375 #define _curl_is_postfields(expr)                                             \
376   (_curl_is_ptr((expr), void) ||                                              \
377    _curl_is_arr((expr), char))
378
379 /* FIXME: the whole callback checking is messy...
380  * The idea is to tolerate char vs. void and const vs. not const
381  * pointers in arguments at least
382  */
383 /* helper: __builtin_types_compatible_p distinguishes between functions and
384  * function pointers, hide it */
385 #define _curl_callback_compatible(func, type)                                 \
386   (__builtin_types_compatible_p(__typeof__(func), type) ||                    \
387    __builtin_types_compatible_p(__typeof__(func), type*))
388
389 /* evaluates to true if expr is of type curl_read_callback or "similar" */
390 #define _curl_is_read_cb(expr)                                          \
391   (_curl_is_NULL(expr) ||                                                     \
392    __builtin_types_compatible_p(__typeof__(expr), __typeof__(fread)) ||       \
393    __builtin_types_compatible_p(__typeof__(expr), curl_read_callback) ||      \
394    _curl_callback_compatible((expr), _curl_read_callback1) ||                 \
395    _curl_callback_compatible((expr), _curl_read_callback2) ||                 \
396    _curl_callback_compatible((expr), _curl_read_callback3) ||                 \
397    _curl_callback_compatible((expr), _curl_read_callback4) ||                 \
398    _curl_callback_compatible((expr), _curl_read_callback5) ||                 \
399    _curl_callback_compatible((expr), _curl_read_callback6))
400 typedef size_t (_curl_read_callback1)(char *, size_t, size_t, void*);
401 typedef size_t (_curl_read_callback2)(char *, size_t, size_t, const void*);
402 typedef size_t (_curl_read_callback3)(char *, size_t, size_t, FILE*);
403 typedef size_t (_curl_read_callback4)(void *, size_t, size_t, void*);
404 typedef size_t (_curl_read_callback5)(void *, size_t, size_t, const void*);
405 typedef size_t (_curl_read_callback6)(void *, size_t, size_t, FILE*);
406
407 /* evaluates to true if expr is of type curl_write_callback or "similar" */
408 #define _curl_is_write_cb(expr)                                               \
409   (_curl_is_read_cb(expr) ||                                            \
410    __builtin_types_compatible_p(__typeof__(expr), __typeof__(fwrite)) ||      \
411    __builtin_types_compatible_p(__typeof__(expr), curl_write_callback) ||     \
412    _curl_callback_compatible((expr), _curl_write_callback1) ||                \
413    _curl_callback_compatible((expr), _curl_write_callback2) ||                \
414    _curl_callback_compatible((expr), _curl_write_callback3) ||                \
415    _curl_callback_compatible((expr), _curl_write_callback4) ||                \
416    _curl_callback_compatible((expr), _curl_write_callback5) ||                \
417    _curl_callback_compatible((expr), _curl_write_callback6))
418 typedef size_t (_curl_write_callback1)(const char *, size_t, size_t, void*);
419 typedef size_t (_curl_write_callback2)(const char *, size_t, size_t,
420                                        const void*);
421 typedef size_t (_curl_write_callback3)(const char *, size_t, size_t, FILE*);
422 typedef size_t (_curl_write_callback4)(const void *, size_t, size_t, void*);
423 typedef size_t (_curl_write_callback5)(const void *, size_t, size_t,
424                                        const void*);
425 typedef size_t (_curl_write_callback6)(const void *, size_t, size_t, FILE*);
426
427 /* evaluates to true if expr is of type curl_ioctl_callback or "similar" */
428 #define _curl_is_ioctl_cb(expr)                                         \
429   (_curl_is_NULL(expr) ||                                                     \
430    __builtin_types_compatible_p(__typeof__(expr), curl_ioctl_callback) ||     \
431    _curl_callback_compatible((expr), _curl_ioctl_callback1) ||                \
432    _curl_callback_compatible((expr), _curl_ioctl_callback2) ||                \
433    _curl_callback_compatible((expr), _curl_ioctl_callback3) ||                \
434    _curl_callback_compatible((expr), _curl_ioctl_callback4))
435 typedef curlioerr (_curl_ioctl_callback1)(CURL *, int, void*);
436 typedef curlioerr (_curl_ioctl_callback2)(CURL *, int, const void*);
437 typedef curlioerr (_curl_ioctl_callback3)(CURL *, curliocmd, void*);
438 typedef curlioerr (_curl_ioctl_callback4)(CURL *, curliocmd, const void*);
439
440 /* evaluates to true if expr is of type curl_sockopt_callback or "similar" */
441 #define _curl_is_sockopt_cb(expr)                                       \
442   (_curl_is_NULL(expr) ||                                                     \
443    __builtin_types_compatible_p(__typeof__(expr), curl_sockopt_callback) ||   \
444    _curl_callback_compatible((expr), _curl_sockopt_callback1) ||              \
445    _curl_callback_compatible((expr), _curl_sockopt_callback2))
446 typedef int (_curl_sockopt_callback1)(void *, curl_socket_t, curlsocktype);
447 typedef int (_curl_sockopt_callback2)(const void *, curl_socket_t,
448                                       curlsocktype);
449
450 /* evaluates to true if expr is of type curl_opensocket_callback or "similar" */
451 #define _curl_is_opensocket_cb(expr)                                    \
452   (_curl_is_NULL(expr) ||                                                     \
453    __builtin_types_compatible_p(__typeof__(expr), curl_opensocket_callback) ||\
454    _curl_callback_compatible((expr), _curl_opensocket_callback1) ||           \
455    _curl_callback_compatible((expr), _curl_opensocket_callback2) ||           \
456    _curl_callback_compatible((expr), _curl_opensocket_callback3) ||           \
457    _curl_callback_compatible((expr), _curl_opensocket_callback4))
458 typedef curl_socket_t (_curl_opensocket_callback1)
459   (void *, curlsocktype, struct curl_sockaddr *);
460 typedef curl_socket_t (_curl_opensocket_callback2)
461   (void *, curlsocktype, const struct curl_sockaddr *);
462 typedef curl_socket_t (_curl_opensocket_callback3)
463   (const void *, curlsocktype, struct curl_sockaddr *);
464 typedef curl_socket_t (_curl_opensocket_callback4)
465   (const void *, curlsocktype, const struct curl_sockaddr *);
466
467 /* evaluates to true if expr is of type curl_progress_callback or "similar" */
468 #define _curl_is_progress_cb(expr)                                      \
469   (_curl_is_NULL(expr) ||                                                     \
470    __builtin_types_compatible_p(__typeof__(expr), curl_progress_callback) ||  \
471    _curl_callback_compatible((expr), _curl_progress_callback1) ||             \
472    _curl_callback_compatible((expr), _curl_progress_callback2))
473 typedef int (_curl_progress_callback1)(void *,
474     double, double, double, double);
475 typedef int (_curl_progress_callback2)(const void *,
476     double, double, double, double);
477
478 /* evaluates to true if expr is of type curl_debug_callback or "similar" */
479 #define _curl_is_debug_cb(expr)                                         \
480   (_curl_is_NULL(expr) ||                                                     \
481    __builtin_types_compatible_p(__typeof__(expr), curl_debug_callback) ||     \
482    _curl_callback_compatible((expr), _curl_debug_callback1) ||                \
483    _curl_callback_compatible((expr), _curl_debug_callback2) ||                \
484    _curl_callback_compatible((expr), _curl_debug_callback3) ||                \
485    _curl_callback_compatible((expr), _curl_debug_callback4))
486 typedef int (_curl_debug_callback1) (CURL *,
487     curl_infotype, char *, size_t, void *);
488 typedef int (_curl_debug_callback2) (CURL *,
489     curl_infotype, char *, size_t, const void *);
490 typedef int (_curl_debug_callback3) (CURL *,
491     curl_infotype, const char *, size_t, void *);
492 typedef int (_curl_debug_callback4) (CURL *,
493     curl_infotype, const char *, size_t, const void *);
494
495 /* evaluates to true if expr is of type curl_ssl_ctx_callback or "similar" */
496 /* this is getting even messier... */
497 #define _curl_is_ssl_ctx_cb(expr)                                       \
498   (_curl_is_NULL(expr) ||                                                     \
499    __builtin_types_compatible_p(__typeof__(expr), curl_ssl_ctx_callback) ||   \
500    _curl_callback_compatible((expr), _curl_ssl_ctx_callback1) ||              \
501    _curl_callback_compatible((expr), _curl_ssl_ctx_callback2) ||              \
502    _curl_callback_compatible((expr), _curl_ssl_ctx_callback3) ||              \
503    _curl_callback_compatible((expr), _curl_ssl_ctx_callback4) ||              \
504    _curl_callback_compatible((expr), _curl_ssl_ctx_callback5) ||              \
505    _curl_callback_compatible((expr), _curl_ssl_ctx_callback6) ||              \
506    _curl_callback_compatible((expr), _curl_ssl_ctx_callback7) ||              \
507    _curl_callback_compatible((expr), _curl_ssl_ctx_callback8))
508 typedef CURLcode (_curl_ssl_ctx_callback1)(CURL *, void *, void *);
509 typedef CURLcode (_curl_ssl_ctx_callback2)(CURL *, void *, const void *);
510 typedef CURLcode (_curl_ssl_ctx_callback3)(CURL *, const void *, void *);
511 typedef CURLcode (_curl_ssl_ctx_callback4)(CURL *, const void *, const void *);
512 #ifdef HEADER_SSL_H
513 /* hack: if we included OpenSSL's ssl.h, we know about SSL_CTX
514  * this will of course break if we're included before OpenSSL headers...
515  */
516 typedef CURLcode (_curl_ssl_ctx_callback5)(CURL *, SSL_CTX, void *);
517 typedef CURLcode (_curl_ssl_ctx_callback6)(CURL *, SSL_CTX, const void *);
518 typedef CURLcode (_curl_ssl_ctx_callback7)(CURL *, const SSL_CTX, void *);
519 typedef CURLcode (_curl_ssl_ctx_callback8)(CURL *, const SSL_CTX, const void *);
520 #else
521 typedef _curl_ssl_ctx_callback1 _curl_ssl_ctx_callback5;
522 typedef _curl_ssl_ctx_callback1 _curl_ssl_ctx_callback6;
523 typedef _curl_ssl_ctx_callback1 _curl_ssl_ctx_callback7;
524 typedef _curl_ssl_ctx_callback1 _curl_ssl_ctx_callback8;
525 #endif
526
527 /* evaluates to true if expr is of type curl_conv_callback or "similar" */
528 #define _curl_is_conv_cb(expr)                                          \
529   (_curl_is_NULL(expr) ||                                                     \
530    __builtin_types_compatible_p(__typeof__(expr), curl_conv_callback) ||      \
531    _curl_callback_compatible((expr), _curl_conv_callback1) ||                 \
532    _curl_callback_compatible((expr), _curl_conv_callback2) ||                 \
533    _curl_callback_compatible((expr), _curl_conv_callback3) ||                 \
534    _curl_callback_compatible((expr), _curl_conv_callback4))
535 typedef CURLcode (*_curl_conv_callback1)(char *, size_t length);
536 typedef CURLcode (*_curl_conv_callback2)(const char *, size_t length);
537 typedef CURLcode (*_curl_conv_callback3)(void *, size_t length);
538 typedef CURLcode (*_curl_conv_callback4)(const void *, size_t length);
539
540 /* evaluates to true if expr is of type curl_seek_callback or "similar" */
541 #define _curl_is_seek_cb(expr)                                          \
542   (_curl_is_NULL(expr) ||                                                     \
543    __builtin_types_compatible_p(__typeof__(expr), curl_seek_callback) ||      \
544    _curl_callback_compatible((expr), _curl_seek_callback1) ||                 \
545    _curl_callback_compatible((expr), _curl_seek_callback2))
546 typedef CURLcode (*_curl_seek_callback1)(void *, curl_off_t, int);
547 typedef CURLcode (*_curl_seek_callback2)(const void *, curl_off_t, int);
548
549
550 #endif /* __CURL_TYPECHECK_GCC_H */