-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathcurl_fuzzer.h
353 lines (296 loc) · 12 KB
/
curl_fuzzer.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
/***************************************************************************
* _ _ ____ _
* Project ___| | | | _ \| |
* / __| | | | |_) | |
* | (__| |_| | _ <| |___
* \___|\___/|_| \_\_____|
*
* Copyright (C) 2017, Max Dymond, <cmeister2@gmail.com>, et al.
*
* This software is licensed as described in the file COPYING, which
* you should have received as part of this distribution. The terms
* are also available at https://curl.se/docs/copyright.html.
*
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
* copies of the Software, and permit persons to whom the Software is
* furnished to do so, under the terms of the COPYING file.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
***************************************************************************/
#include <inttypes.h>
#include <curl/curl.h>
#include "testinput.h"
/**
* TLV types.
*/
#define TLV_TYPE_URL 1
#define TLV_TYPE_RESPONSE0 2
#define TLV_TYPE_USERNAME 3
#define TLV_TYPE_PASSWORD 4
#define TLV_TYPE_POSTFIELDS 5
#define TLV_TYPE_HEADER 6
#define TLV_TYPE_COOKIE 7
#define TLV_TYPE_UPLOAD1 8
#define TLV_TYPE_RANGE 9
#define TLV_TYPE_CUSTOMREQUEST 10
#define TLV_TYPE_MAIL_RECIPIENT 11
#define TLV_TYPE_MAIL_FROM 12
#define TLV_TYPE_MIME_PART 13
#define TLV_TYPE_MIME_PART_NAME 14
#define TLV_TYPE_MIME_PART_DATA 15
#define TLV_TYPE_HTTPAUTH 16
#define TLV_TYPE_RESPONSE1 17
#define TLV_TYPE_RESPONSE2 18
#define TLV_TYPE_RESPONSE3 19
#define TLV_TYPE_RESPONSE4 20
#define TLV_TYPE_RESPONSE5 21
#define TLV_TYPE_RESPONSE6 22
#define TLV_TYPE_RESPONSE7 23
#define TLV_TYPE_RESPONSE8 24
#define TLV_TYPE_RESPONSE9 25
#define TLV_TYPE_RESPONSE10 26
#define TLV_TYPE_OPTHEADER 27
#define TLV_TYPE_NOBODY 28
#define TLV_TYPE_FOLLOWLOCATION 29
#define TLV_TYPE_ACCEPTENCODING 30
#define TLV_TYPE_SECOND_RESPONSE0 31
#define TLV_TYPE_SECOND_RESPONSE1 32
#define TLV_TYPE_WILDCARDMATCH 33
#define TLV_TYPE_RTSP_REQUEST 34
#define TLV_TYPE_RTSP_SESSION_ID 35
#define TLV_TYPE_RTSP_STREAM_URI 36
#define TLV_TYPE_RTSP_TRANSPORT 37
#define TLV_TYPE_RTSP_CLIENT_CSEQ 38
#define TLV_TYPE_MAIL_AUTH 39
#define TLV_TYPE_HTTP_VERSION 40
#define TLV_TYPE_DOH_URL 41
#define TLV_TYPE_LOGIN_OPTIONS 42
#define TLV_TYPE_XOAUTH2_BEARER 43
#define TLV_TYPE_USERPWD 44
#define TLV_TYPE_USERAGENT 45
#define TLV_TYPE_NETRC 46
#define TLV_TYPE_SSH_HOST_PUBLIC_KEY_SHA256 47
#define TLV_TYPE_POST 48
#define TLV_TYPE_WS_OPTIONS 49
#define TLV_TYPE_CONNECT_ONLY 50
#define TLV_TYPE_HSTS 51
#define TLV_TYPE_HTTPPOSTBODY 52
/**
* TLV function return codes.
*/
#define TLV_RC_NO_ERROR 0
#define TLV_RC_NO_MORE_TLVS 1
#define TLV_RC_SIZE_ERROR 2
/* Temporary write array size */
#define TEMP_WRITE_ARRAY_SIZE 10
/* Maximum write size in bytes to stop unbounded writes (50MB) */
#define MAXIMUM_WRITE_LENGTH 52428800
/* convenience string for HTTPPOST body name */
#define FUZZ_HTTPPOST_NAME "test"
/* Cookie-jar WRITE (CURLOPT_COOKIEJAR) path. */
#define FUZZ_COOKIE_JAR_PATH "/dev/null"
/* Cookie-jar READ (CURLOPT_COOKIEFILE) path. */
#define FUZZ_RO_COOKIE_FILE_PATH "/dev/null"
/* Alt-Svc header cache path */
#define FUZZ_ALT_SVC_HEADER_CACHE_PATH "/dev/null"
/* HSTS header cache path */
#define FUZZ_HSTS_HEADER_CACHE_PATH "/dev/null"
/* Certificate Revocation List file path */
#define FUZZ_CRL_FILE_PATH "/dev/null"
/* .netrc file path */
#define FUZZ_NETRC_FILE_PATH "/dev/null"
/* Number of supported responses */
#define TLV_MAX_NUM_RESPONSES 11
/* Number of allowed CURLOPT_HEADERs */
#define TLV_MAX_NUM_CURLOPT_HEADER 2000
/* Space variable for all CURLOPTs. */
#define FUZZ_CURLOPT_TRACKER_SPACE 500
/* Number of connections allowed to be opened */
#define FUZZ_NUM_CONNECTIONS 2
typedef enum fuzz_sock_state {
FUZZ_SOCK_CLOSED,
FUZZ_SOCK_OPEN,
FUZZ_SOCK_SHUTDOWN
} FUZZ_SOCK_STATE;
/**
* Byte stream representation of the TLV header. Casting the byte stream
* to a TLV_RAW allows us to examine the type and length.
*/
typedef struct tlv_raw
{
/* Type of the TLV - 16 bits. */
uint8_t raw_type[2];
/* Length of the TLV data - 32 bits. */
uint8_t raw_length[4];
} TLV_RAW;
typedef struct tlv
{
/* Type of the TLV */
uint16_t type;
/* Length of the TLV data */
uint32_t length;
/* Pointer to data if length > 0. */
const uint8_t *value;
} TLV;
/**
* Internal state when parsing a TLV data stream.
*/
typedef struct fuzz_parse_state
{
/* Data stream */
const uint8_t *data;
size_t data_len;
/* Current position of our "cursor" in processing the data stream. */
size_t data_pos;
} FUZZ_PARSE_STATE;
/**
* Structure to use for responses.
*/
typedef struct fuzz_response
{
/* Response data and length */
const uint8_t *data;
size_t data_len;
} FUZZ_RESPONSE;
typedef struct fuzz_socket_manager
{
unsigned char index;
/* Responses. Response 0 is sent as soon as the socket is connected. Further
responses are sent when the socket becomes readable. */
FUZZ_RESPONSE responses[TLV_MAX_NUM_RESPONSES];
int response_index;
/* Server file descriptor. */
FUZZ_SOCK_STATE fd_state;
curl_socket_t fd;
} FUZZ_SOCKET_MANAGER;
/**
* Data local to a fuzzing run.
*/
typedef struct fuzz_data
{
/* CURL easy object */
CURL *easy;
/* Parser state */
FUZZ_PARSE_STATE state;
/* Temporary writefunction state */
char write_array[TEMP_WRITE_ARRAY_SIZE];
/* Cumulative length of "written" data */
size_t written_data;
/* Upload data and length; */
const uint8_t *upload1_data;
size_t upload1_data_len;
size_t upload1_data_written;
/* Singleton option tracker. Options should only be set once. */
unsigned char options[FUZZ_CURLOPT_TRACKER_SPACE];
/* CURLOPT_POSTFIELDS data. */
char *postfields;
/* List of headers */
int header_list_count;
struct curl_slist *header_list;
/* List of mail recipients */
struct curl_slist *mail_recipients_list;
/* List of connect_to strings */
struct curl_slist *connect_to_list;
/* Mime data */
curl_mime *mime;
curl_mimepart *part;
/* httppost data */
struct curl_httppost *httppost;
struct curl_httppost *last_post_part;
char *post_body;
/* Server socket managers. Primarily socket manager 0 is used, but some
protocols (FTP) use two sockets. */
FUZZ_SOCKET_MANAGER sockman[FUZZ_NUM_CONNECTIONS];
/* Verbose mode. */
int verbose;
} FUZZ_DATA;
/* Function prototypes */
uint32_t to_u32(const uint8_t b[4]);
uint16_t to_u16(const uint8_t b[2]);
int fuzz_initialize_fuzz_data(FUZZ_DATA *fuzz,
const uint8_t *data,
size_t data_len);
int fuzz_set_easy_options(FUZZ_DATA *fuzz);
void fuzz_terminate_fuzz_data(FUZZ_DATA *fuzz);
void fuzz_free(void **ptr);
curl_socket_t fuzz_open_socket(void *ptr,
curlsocktype purpose,
struct curl_sockaddr *address);
int fuzz_sockopt_callback(void *ptr,
curl_socket_t curlfd,
curlsocktype purpose);
size_t fuzz_read_callback(char *buffer,
size_t size,
size_t nitems,
void *ptr);
size_t fuzz_write_callback(void *contents,
size_t size,
size_t nmemb,
void *ptr);
int fuzz_get_first_tlv(FUZZ_DATA *fuzz, TLV *tlv);
int fuzz_get_next_tlv(FUZZ_DATA *fuzz, TLV *tlv);
int fuzz_get_tlv_comn(FUZZ_DATA *fuzz, TLV *tlv);
int fuzz_parse_tlv(FUZZ_DATA *fuzz, TLV *tlv);
char *fuzz_tlv_to_string(TLV *tlv);
void fuzz_setup_http_post(FUZZ_DATA *fuzz, TLV *tlv);
int fuzz_add_mime_part(TLV *src_tlv, curl_mimepart *part);
int fuzz_parse_mime_tlv(curl_mimepart *part, TLV *tlv);
int fuzz_handle_transfer(FUZZ_DATA *fuzz);
int fuzz_send_next_response(FUZZ_DATA *fuzz, FUZZ_SOCKET_MANAGER *sockman);
int fuzz_select(int nfds,
fd_set *readfds,
fd_set *writefds,
fd_set *exceptfds,
struct timeval *timeout);
int fuzz_set_allowed_protocols(FUZZ_DATA *fuzz);
/* Macros */
#define FTRY(FUNC) \
{ \
int _func_rc = (FUNC); \
if (_func_rc) \
{ \
rc = _func_rc; \
goto EXIT_LABEL; \
} \
}
#define FCHECK(COND) \
{ \
if (!(COND)) \
{ \
rc = 255; \
goto EXIT_LABEL; \
} \
}
#define FSET_OPTION(FUZZP, OPTNAME, OPTVALUE) \
FTRY(curl_easy_setopt((FUZZP)->easy, OPTNAME, OPTVALUE)); \
(FUZZP)->options[OPTNAME % 1000] = 1
#define FCHECK_OPTION_UNSET(FUZZP, OPTNAME) \
FCHECK((FUZZP)->options[OPTNAME % 1000] == 0)
#define FSINGLETONTLV(FUZZP, TLVNAME, OPTNAME) \
case TLVNAME: \
FCHECK_OPTION_UNSET(FUZZP, OPTNAME); \
tmp = fuzz_tlv_to_string(tlv); \
FSET_OPTION(FUZZP, OPTNAME, tmp); \
break
#define FRESPONSETLV(SMAN, TLVNAME, INDEX) \
case TLVNAME: \
(SMAN)->responses[(INDEX)].data = tlv->value; \
(SMAN)->responses[(INDEX)].data_len = tlv->length; \
break
#define FU32TLV(FUZZP, TLVNAME, OPTNAME) \
case TLVNAME: \
if(tlv->length != 4) { \
rc = 255; \
goto EXIT_LABEL; \
} \
FCHECK_OPTION_UNSET(FUZZP, OPTNAME); \
tmp_u32 = to_u32(tlv->value); \
FSET_OPTION(FUZZP, OPTNAME, tmp_u32); \
break
#define FV_PRINTF(FUZZP, ...) \
if((FUZZP)->verbose) { \
printf(__VA_ARGS__); \
}
#define FUZZ_MAX(A, B) ((A) > (B) ? (A) : (B))