OLD | NEW |
| (Empty) |
1 /* packet-spdy.c | |
2 * Routines for SPDY packet disassembly | |
3 * For now, the protocol spec can be found at | |
4 * http://dev.chromium.org/spdy/spdy-protocol | |
5 * | |
6 * Copyright 2010, Google Inc. | |
7 * Eric Shienbrood <ers@google.com> | |
8 * | |
9 * $Id$ | |
10 * | |
11 * Wireshark - Network traffic analyzer | |
12 * By Gerald Combs <gerald@wireshark.org> | |
13 * Copyright 1998 Gerald Combs | |
14 * | |
15 * Originally based on packet-http.c | |
16 * | |
17 * This program is free software; you can redistribute it and/or | |
18 * modify it under the terms of the GNU General Public License | |
19 * as published by the Free Software Foundation; either version 2 | |
20 * of the License, or (at your option) any later version. | |
21 * | |
22 * This program is distributed in the hope that it will be useful, | |
23 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
24 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
25 * GNU General Public License for more details. | |
26 * | |
27 * You should have received a copy of the GNU General Public License | |
28 * along with this program; if not, write to the Free Software | |
29 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. | |
30 */ | |
31 | |
32 #ifdef HAVE_CONFIG_H | |
33 #include "config.h" | |
34 #endif | |
35 | |
36 #include <string.h> | |
37 #include <ctype.h> | |
38 | |
39 #include <glib.h> | |
40 #include <epan/conversation.h> | |
41 #include <epan/packet.h> | |
42 #include <epan/strutil.h> | |
43 #include <epan/base64.h> | |
44 #include <epan/emem.h> | |
45 #include <epan/stats_tree.h> | |
46 | |
47 #include <epan/req_resp_hdrs.h> | |
48 #include "packet-spdy.h" | |
49 #include <epan/dissectors/packet-tcp.h> | |
50 #include <epan/dissectors/packet-ssl.h> | |
51 #include <epan/prefs.h> | |
52 #include <epan/expert.h> | |
53 #include <epan/uat.h> | |
54 | |
55 #define SPDY_FIN 0x01 | |
56 | |
57 /* The types of SPDY control frames */ | |
58 typedef enum _spdy_type { | |
59 SPDY_DATA, | |
60 SPDY_SYN_STREAM, | |
61 SPDY_SYN_REPLY, | |
62 SPDY_FIN_STREAM, | |
63 SPDY_HELLO, | |
64 SPDY_NOOP, | |
65 SPDY_PING, | |
66 SPDY_INVALID | |
67 } spdy_frame_type_t; | |
68 | |
69 static const char *frame_type_names[] = { | |
70 "DATA", "SYN_STREAM", "SYN_REPLY", "FIN_STREAM", "HELLO", "NOOP", | |
71 "PING", "INVALID" | |
72 }; | |
73 | |
74 /* | |
75 * This structure will be tied to each SPDY frame. | |
76 * Note that there may be multiple SPDY frames | |
77 * in one packet. | |
78 */ | |
79 typedef struct _spdy_frame_info_t { | |
80 guint32 stream_id; | |
81 guint8 *header_block; | |
82 guint header_block_len; | |
83 guint16 frame_type; | |
84 } spdy_frame_info_t; | |
85 | |
86 /* | |
87 * This structures keeps track of all the data frames | |
88 * associated with a stream, so that they can be | |
89 * reassembled into a single chunk. | |
90 */ | |
91 typedef struct _spdy_data_frame_t { | |
92 guint8 *data; | |
93 guint32 length; | |
94 guint32 framenum; | |
95 } spdy_data_frame_t; | |
96 | |
97 typedef struct _spdy_stream_info_t { | |
98 gchar *content_type; | |
99 gchar *content_type_parameters; | |
100 gchar *content_encoding; | |
101 GSList *data_frames; | |
102 tvbuff_t *assembled_data; | |
103 guint num_data_frames; | |
104 } spdy_stream_info_t; | |
105 | |
106 #include <epan/tap.h> | |
107 | |
108 | |
109 static int spdy_tap = -1; | |
110 static int spdy_eo_tap = -1; | |
111 | |
112 static int proto_spdy = -1; | |
113 static int hf_spdy_syn_stream = -1; | |
114 static int hf_spdy_syn_reply = -1; | |
115 static int hf_spdy_control_bit = -1; | |
116 static int hf_spdy_version = -1; | |
117 static int hf_spdy_type = -1; | |
118 static int hf_spdy_flags = -1; | |
119 static int hf_spdy_flags_fin = -1; | |
120 static int hf_spdy_length = -1; | |
121 static int hf_spdy_header = -1; | |
122 static int hf_spdy_header_name = -1; | |
123 static int hf_spdy_header_name_text = -1; | |
124 static int hf_spdy_header_value = -1; | |
125 static int hf_spdy_header_value_text = -1; | |
126 static int hf_spdy_streamid = -1; | |
127 static int hf_spdy_associated_streamid = -1; | |
128 static int hf_spdy_priority = -1; | |
129 static int hf_spdy_num_headers = -1; | |
130 static int hf_spdy_num_headers_string = -1; | |
131 | |
132 static gint ett_spdy = -1; | |
133 static gint ett_spdy_syn_stream = -1; | |
134 static gint ett_spdy_syn_reply = -1; | |
135 static gint ett_spdy_fin_stream = -1; | |
136 static gint ett_spdy_flags = -1; | |
137 static gint ett_spdy_header = -1; | |
138 static gint ett_spdy_header_name = -1; | |
139 static gint ett_spdy_header_value = -1; | |
140 | |
141 static gint ett_spdy_encoded_entity = -1; | |
142 | |
143 static dissector_handle_t data_handle; | |
144 static dissector_handle_t media_handle; | |
145 static dissector_handle_t spdy_handle; | |
146 | |
147 /* Stuff for generation/handling of fields for custom HTTP headers */ | |
148 typedef struct _header_field_t { | |
149 gchar* header_name; | |
150 gchar* header_desc; | |
151 } header_field_t; | |
152 | |
153 /* | |
154 * desegmentation of SPDY control frames | |
155 * (when we are over TCP or another protocol providing the desegmentation API) | |
156 */ | |
157 static gboolean spdy_desegment_control_frames = TRUE; | |
158 | |
159 /* | |
160 * desegmentation of SPDY data frames bodies | |
161 * (when we are over TCP or another protocol providing the desegmentation API) | |
162 * TODO let the user filter on content-type the bodies he wants desegmented | |
163 */ | |
164 static gboolean spdy_desegment_data_frames = TRUE; | |
165 | |
166 static gboolean spdy_assemble_entity_bodies = TRUE; | |
167 | |
168 /* | |
169 * Decompression of zlib encoded entities. | |
170 */ | |
171 #ifdef HAVE_LIBZ | |
172 static gboolean spdy_decompress_body = TRUE; | |
173 static gboolean spdy_decompress_headers = TRUE; | |
174 #else | |
175 static gboolean spdy_decompress_body = FALSE; | |
176 static gboolean spdy_decompress_headers = FALSE; | |
177 #endif | |
178 static gboolean spdy_debug = FALSE; | |
179 | |
180 #define TCP_PORT_DAAP 3689 | |
181 | |
182 /* | |
183 * SSDP is implemented atop HTTP (yes, it really *does* run over UDP). | |
184 */ | |
185 #define TCP_PORT_SSDP 1900 | |
186 #define UDP_PORT_SSDP 1900 | |
187 | |
188 /* | |
189 * tcp and ssl ports | |
190 */ | |
191 | |
192 #define TCP_DEFAULT_RANGE "80,8080" | |
193 #define SSL_DEFAULT_RANGE "443" | |
194 | |
195 static range_t *global_spdy_tcp_range = NULL; | |
196 static range_t *global_spdy_ssl_range = NULL; | |
197 | |
198 static range_t *spdy_tcp_range = NULL; | |
199 static range_t *spdy_ssl_range = NULL; | |
200 | |
201 static const value_string vals_status_code[] = { | |
202 { 100, "Continue" }, | |
203 { 101, "Switching Protocols" }, | |
204 { 102, "Processing" }, | |
205 { 199, "Informational - Others" }, | |
206 | |
207 { 200, "OK"}, | |
208 { 201, "Created"}, | |
209 { 202, "Accepted"}, | |
210 { 203, "Non-authoritative Information"}, | |
211 { 204, "No Content"}, | |
212 { 205, "Reset Content"}, | |
213 { 206, "Partial Content"}, | |
214 { 207, "Multi-Status"}, | |
215 { 299, "Success - Others"}, | |
216 | |
217 { 300, "Multiple Choices"}, | |
218 { 301, "Moved Permanently"}, | |
219 { 302, "Found"}, | |
220 { 303, "See Other"}, | |
221 { 304, "Not Modified"}, | |
222 { 305, "Use Proxy"}, | |
223 { 307, "Temporary Redirect"}, | |
224 { 399, "Redirection - Others"}, | |
225 | |
226 { 400, "Bad Request"}, | |
227 { 401, "Unauthorized"}, | |
228 { 402, "Payment Required"}, | |
229 { 403, "Forbidden"}, | |
230 { 404, "Not Found"}, | |
231 { 405, "Method Not Allowed"}, | |
232 { 406, "Not Acceptable"}, | |
233 { 407, "Proxy Authentication Required"}, | |
234 { 408, "Request Time-out"}, | |
235 { 409, "Conflict"}, | |
236 { 410, "Gone"}, | |
237 { 411, "Length Required"}, | |
238 { 412, "Precondition Failed"}, | |
239 { 413, "Request Entity Too Large"}, | |
240 { 414, "Request-URI Too Long"}, | |
241 { 415, "Unsupported Media Type"}, | |
242 { 416, "Requested Range Not Satisfiable"}, | |
243 { 417, "Expectation Failed"}, | |
244 { 418, "I'm a teapot"}, /* RFC 2324 */ | |
245 { 422, "Unprocessable Entity"}, | |
246 { 423, "Locked"}, | |
247 { 424, "Failed Dependency"}, | |
248 { 499, "Client Error - Others"}, | |
249 | |
250 { 500, "Internal Server Error"}, | |
251 { 501, "Not Implemented"}, | |
252 { 502, "Bad Gateway"}, | |
253 { 503, "Service Unavailable"}, | |
254 { 504, "Gateway Time-out"}, | |
255 { 505, "HTTP Version not supported"}, | |
256 { 507, "Insufficient Storage"}, | |
257 { 599, "Server Error - Others"}, | |
258 | |
259 { 0, NULL} | |
260 }; | |
261 | |
262 static const char spdy_dictionary[] = | |
263 "optionsgetheadpostputdeletetraceacceptaccept-charsetaccept-encodingaccept-" | |
264 "languageauthorizationexpectfromhostif-modified-sinceif-matchif-none-matchi" | |
265 "f-rangeif-unmodifiedsincemax-forwardsproxy-authorizationrangerefererteuser" | |
266 "-agent10010120020120220320420520630030130230330430530630740040140240340440" | |
267 "5406407408409410411412413414415416417500501502503504505accept-rangesageeta" | |
268 "glocationproxy-authenticatepublicretry-afterservervarywarningwww-authentic" | |
269 "ateallowcontent-basecontent-encodingcache-controlconnectiondatetrailertran" | |
270 "sfer-encodingupgradeviawarningcontent-languagecontent-lengthcontent-locati" | |
271 "oncontent-md5content-rangecontent-typeetagexpireslast-modifiedset-cookieMo" | |
272 "ndayTuesdayWednesdayThursdayFridaySaturdaySundayJanFebMarAprMayJunJulAugSe" | |
273 "pOctNovDecchunkedtext/htmlimage/pngimage/jpgimage/gifapplication/xmlapplic" | |
274 "ation/xhtmltext/plainpublicmax-agecharset=iso-8859-1utf-8gzipdeflateHTTP/1" | |
275 ".1statusversionurl"; | |
276 | |
277 static void reset_decompressors(void) | |
278 { | |
279 if (spdy_debug) printf("Should reset SPDY decompressors\n"); | |
280 } | |
281 | |
282 static spdy_conv_t * | |
283 get_spdy_conversation_data(packet_info *pinfo) | |
284 { | |
285 conversation_t *conversation; | |
286 spdy_conv_t *conv_data; | |
287 int retcode; | |
288 | |
289 conversation = find_conversation(pinfo->fd->num, &pinfo->src, &pinfo->dst, p
info->ptype, pinfo->srcport, pinfo->destport, 0); | |
290 if (spdy_debug) { | |
291 printf("\n===========================================\n\n"); | |
292 printf("Conversation for frame #%d is %p\n", pinfo->fd->num, conversatio
n); | |
293 if (conversation) | |
294 printf(" conv_data=%p\n", conversation_get_proto_data(conversation,
proto_spdy)); | |
295 } | |
296 | |
297 if(!conversation) /* Conversation does not exist yet - create it */ | |
298 conversation = conversation_new(pinfo->fd->num, &pinfo->src, &pinfo->dst
, pinfo->ptype, pinfo->srcport, pinfo->destport, 0); | |
299 | |
300 /* Retrieve information from conversation | |
301 */ | |
302 conv_data = conversation_get_proto_data(conversation, proto_spdy); | |
303 if(!conv_data) { | |
304 /* Setup the conversation structure itself */ | |
305 conv_data = se_alloc0(sizeof(spdy_conv_t)); | |
306 | |
307 conv_data->streams = NULL; | |
308 if (spdy_decompress_headers) { | |
309 conv_data->rqst_decompressor = se_alloc0(sizeof(z_stream)); | |
310 conv_data->rply_decompressor = se_alloc0(sizeof(z_stream)); | |
311 retcode = inflateInit(conv_data->rqst_decompressor); | |
312 if (retcode == Z_OK) | |
313 retcode = inflateInit(conv_data->rply_decompressor); | |
314 if (retcode != Z_OK) | |
315 printf("frame #%d: inflateInit() failed: %d\n", pinfo->fd->num,
retcode); | |
316 else if (spdy_debug) | |
317 printf("created decompressor\n"); | |
318 conv_data->dictionary_id = adler32(0L, Z_NULL, 0); | |
319 conv_data->dictionary_id = adler32(conv_data->dictionary_id, | |
320 spdy_dictionary, | |
321 sizeof(spdy_dictionary)); | |
322 } | |
323 | |
324 conversation_add_proto_data(conversation, proto_spdy, conv_data); | |
325 register_postseq_cleanup_routine(reset_decompressors); | |
326 } | |
327 return conv_data; | |
328 } | |
329 | |
330 static void | |
331 spdy_save_stream_info(spdy_conv_t *conv_data, | |
332 guint32 stream_id, | |
333 gchar *content_type, | |
334 gchar *content_type_params, | |
335 gchar *content_encoding) | |
336 { | |
337 spdy_stream_info_t *si; | |
338 | |
339 if (conv_data->streams == NULL) | |
340 conv_data->streams = g_array_new(FALSE, TRUE, sizeof(spdy_stream_info_t
*)); | |
341 if (stream_id < conv_data->streams->len) | |
342 DISSECTOR_ASSERT(g_array_index(conv_data->streams, spdy_stream_info_t*,
stream_id) == NULL); | |
343 else | |
344 g_array_set_size(conv_data->streams, stream_id+1); | |
345 si = se_alloc(sizeof(spdy_stream_info_t)); | |
346 si->content_type = content_type; | |
347 si->content_type_parameters = content_type_params; | |
348 si->content_encoding = content_encoding; | |
349 si->data_frames = NULL; | |
350 si->num_data_frames = 0; | |
351 si->assembled_data = NULL; | |
352 g_array_index(conv_data->streams, spdy_stream_info_t*, stream_id) = si; | |
353 if (spdy_debug) | |
354 printf("Saved stream info for ID %u, content type %s\n", stream_id, cont
ent_type); | |
355 } | |
356 | |
357 static spdy_stream_info_t * | |
358 spdy_get_stream_info(spdy_conv_t *conv_data, guint32 stream_id) | |
359 { | |
360 if (conv_data->streams == NULL || stream_id >= conv_data->streams->len) | |
361 return NULL; | |
362 else | |
363 return g_array_index(conv_data->streams, spdy_stream_info_t*, stream_id)
; | |
364 } | |
365 | |
366 static void | |
367 spdy_add_data_chunk(spdy_conv_t *conv_data, guint32 stream_id, guint32 frame, | |
368 guint8 *data, guint32 length) | |
369 { | |
370 spdy_stream_info_t *si = spdy_get_stream_info(conv_data, stream_id); | |
371 | |
372 if (si == NULL) { | |
373 if (spdy_debug) printf("No stream_info found for stream %d\n", stream_id
); | |
374 } else { | |
375 spdy_data_frame_t *df = g_malloc(sizeof(spdy_data_frame_t)); | |
376 df->data = data; | |
377 df->length = length; | |
378 df->framenum = frame; | |
379 si->data_frames = g_slist_append(si->data_frames, df); | |
380 ++si->num_data_frames; | |
381 if (spdy_debug) | |
382 printf("Saved %u bytes of data for stream %u frame %u\n", | |
383 length, stream_id, df->framenum); | |
384 } | |
385 } | |
386 | |
387 static void | |
388 spdy_increment_data_chunk_count(spdy_conv_t *conv_data, guint32 stream_id) | |
389 { | |
390 spdy_stream_info_t *si = spdy_get_stream_info(conv_data, stream_id); | |
391 if (si != NULL) | |
392 ++si->num_data_frames; | |
393 } | |
394 | |
395 /* | |
396 * Return the number of data frames saved so far for the specified stream. | |
397 */ | |
398 static guint | |
399 spdy_get_num_data_frames(spdy_conv_t *conv_data, guint32 stream_id) | |
400 { | |
401 spdy_stream_info_t *si = spdy_get_stream_info(conv_data, stream_id); | |
402 | |
403 return si == NULL ? 0 : si->num_data_frames; | |
404 } | |
405 | |
406 static spdy_stream_info_t * | |
407 spdy_assemble_data_frames(spdy_conv_t *conv_data, guint32 stream_id) | |
408 { | |
409 spdy_stream_info_t *si = spdy_get_stream_info(conv_data, stream_id); | |
410 tvbuff_t *tvb; | |
411 | |
412 if (si == NULL) | |
413 return NULL; | |
414 | |
415 /* | |
416 * Compute the total amount of data and concatenate the | |
417 * data chunks, if it hasn't already been done. | |
418 */ | |
419 if (si->assembled_data == NULL) { | |
420 spdy_data_frame_t *df; | |
421 guint8 *data; | |
422 guint32 datalen; | |
423 guint32 offset; | |
424 guint32 framenum; | |
425 GSList *dflist = si->data_frames; | |
426 if (dflist == NULL) | |
427 return si; | |
428 dflist = si->data_frames; | |
429 datalen = 0; | |
430 /* | |
431 * I'd like to use a composite tvbuff here, but since | |
432 * only a real-data tvbuff can be the child of another | |
433 * tvb, I can't. It would be nice if this limitation | |
434 * could be fixed. | |
435 */ | |
436 while (dflist != NULL) { | |
437 df = dflist->data; | |
438 datalen += df->length; | |
439 dflist = g_slist_next(dflist); | |
440 } | |
441 if (datalen != 0) { | |
442 data = se_alloc(datalen); | |
443 dflist = si->data_frames; | |
444 offset = 0; | |
445 framenum = 0; | |
446 while (dflist != NULL) { | |
447 df = dflist->data; | |
448 memcpy(data+offset, df->data, df->length); | |
449 offset += df->length; | |
450 dflist = g_slist_next(dflist); | |
451 } | |
452 tvb = tvb_new_real_data(data, datalen, datalen); | |
453 si->assembled_data = tvb; | |
454 } | |
455 } | |
456 return si; | |
457 } | |
458 | |
459 static void | |
460 spdy_discard_data_frames(spdy_stream_info_t *si) | |
461 { | |
462 GSList *dflist = si->data_frames; | |
463 spdy_data_frame_t *df; | |
464 | |
465 if (dflist == NULL) | |
466 return; | |
467 while (dflist != NULL) { | |
468 df = dflist->data; | |
469 if (df->data != NULL) { | |
470 g_free(df->data); | |
471 df->data = NULL; | |
472 } | |
473 dflist = g_slist_next(dflist); | |
474 } | |
475 /*g_slist_free(si->data_frames); | |
476 si->data_frames = NULL; */ | |
477 } | |
478 | |
479 // TODO(cbentzel): tvb_child_uncompress should be exported by wireshark. | |
480 static tvbuff_t* spdy_tvb_child_uncompress(tvbuff_t *parent _U_, tvbuff_t *tvb, | |
481 int offset, int comprlen) | |
482 { | |
483 tvbuff_t *new_tvb = tvb_uncompress(tvb, offset, comprlen); | |
484 if (new_tvb) | |
485 tvb_set_child_real_data_tvbuff (parent, new_tvb); | |
486 return new_tvb; | |
487 } | |
488 | |
489 static int | |
490 dissect_spdy_data_frame(tvbuff_t *tvb, int offset, | |
491 packet_info *pinfo, | |
492 proto_tree *top_level_tree, | |
493 proto_tree *spdy_tree, | |
494 proto_item *spdy_proto, | |
495 spdy_conv_t *conv_data) | |
496 { | |
497 guint32 stream_id; | |
498 guint8 flags; | |
499 guint32 frame_length; | |
500 proto_item *ti; | |
501 proto_tree *flags_tree; | |
502 guint32 reported_datalen; | |
503 guint32 datalen; | |
504 dissector_table_t media_type_subdissector_table; | |
505 dissector_table_t port_subdissector_table; | |
506 dissector_handle_t handle; | |
507 guint num_data_frames; | |
508 gboolean dissected; | |
509 | |
510 stream_id = tvb_get_bits32(tvb, (offset << 3) + 1, 31, FALSE); | |
511 flags = tvb_get_guint8(tvb, offset+4); | |
512 frame_length = tvb_get_ntoh24(tvb, offset+5); | |
513 | |
514 if (spdy_debug) | |
515 printf("Data frame [stream_id=%u flags=0x%x length=%d]\n", | |
516 stream_id, flags, frame_length); | |
517 if (spdy_tree) proto_item_append_text(spdy_tree, ", data frame"); | |
518 col_add_fstr(pinfo->cinfo, COL_INFO, "DATA[%u] length=%d", | |
519 stream_id, frame_length); | |
520 | |
521 proto_item_append_text(spdy_proto, ":%s stream=%d length=%d", | |
522 flags & SPDY_FIN ? " [FIN]" : "", | |
523 stream_id, frame_length); | |
524 | |
525 proto_tree_add_boolean(spdy_tree, hf_spdy_control_bit, tvb, offset, 1, 0); | |
526 proto_tree_add_uint(spdy_tree, hf_spdy_streamid, tvb, offset, 4, stream_id); | |
527 ti = proto_tree_add_uint_format(spdy_tree, hf_spdy_flags, tvb, offset+4, 1,
flags, | |
528 "Flags: 0x%02x%s", flags, flags&SPDY_FIN ? " (FIN)" : ""); | |
529 | |
530 flags_tree = proto_item_add_subtree(ti, ett_spdy_flags); | |
531 proto_tree_add_boolean(flags_tree, hf_spdy_flags_fin, tvb, offset+4, 1, flag
s); | |
532 proto_tree_add_uint(spdy_tree, hf_spdy_length, tvb, offset+5, 3, frame_lengt
h); | |
533 | |
534 datalen = tvb_length_remaining(tvb, offset); | |
535 if (datalen > frame_length) | |
536 datalen = frame_length; | |
537 | |
538 reported_datalen = tvb_reported_length_remaining(tvb, offset); | |
539 if (reported_datalen > frame_length) | |
540 reported_datalen = frame_length; | |
541 | |
542 num_data_frames = spdy_get_num_data_frames(conv_data, stream_id); | |
543 if (datalen != 0 || num_data_frames != 0) { | |
544 /* | |
545 * There's stuff left over; process it. | |
546 */ | |
547 tvbuff_t *next_tvb = NULL; | |
548 tvbuff_t *data_tvb = NULL; | |
549 spdy_stream_info_t *si = NULL; | |
550 void *save_private_data = NULL; | |
551 guint8 *copied_data; | |
552 gboolean private_data_changed = FALSE; | |
553 gboolean is_single_chunk = FALSE; | |
554 gboolean have_entire_body; | |
555 | |
556 /* | |
557 * Create a tvbuff for the payload. | |
558 */ | |
559 if (datalen != 0) { | |
560 next_tvb = tvb_new_subset(tvb, offset+8, datalen, | |
561 reported_datalen); | |
562 is_single_chunk = num_data_frames == 0 && (flags & SPDY_FIN) != 0; | |
563 if (!pinfo->fd->flags.visited) { | |
564 if (!is_single_chunk) { | |
565 if (spdy_assemble_entity_bodies) { | |
566 copied_data = tvb_memdup(next_tvb, 0, datalen); | |
567 spdy_add_data_chunk(conv_data, stream_id, pinfo->fd->num
, | |
568 copied_data, datalen); | |
569 } else | |
570 spdy_increment_data_chunk_count(conv_data, stream_id); | |
571 } | |
572 } | |
573 } else | |
574 is_single_chunk = (num_data_frames == 1); | |
575 | |
576 if (!(flags & SPDY_FIN)) { | |
577 col_set_fence(pinfo->cinfo, COL_INFO); | |
578 col_add_fstr(pinfo->cinfo, COL_INFO, " (partial entity)"); | |
579 proto_item_append_text(spdy_proto, " (partial entity body)"); | |
580 /* would like the proto item to say */ | |
581 /* " (entity body fragment N of M)" */ | |
582 goto body_dissected; | |
583 } | |
584 have_entire_body = is_single_chunk; | |
585 /* | |
586 * On seeing the last data frame in a stream, we can | |
587 * reassemble the frames into one data block. | |
588 */ | |
589 si = spdy_assemble_data_frames(conv_data, stream_id); | |
590 if (si == NULL) | |
591 goto body_dissected; | |
592 data_tvb = si->assembled_data; | |
593 if (spdy_assemble_entity_bodies) | |
594 have_entire_body = TRUE; | |
595 | |
596 if (!have_entire_body) | |
597 goto body_dissected; | |
598 | |
599 if (data_tvb == NULL) | |
600 data_tvb = next_tvb; | |
601 else | |
602 add_new_data_source(pinfo, data_tvb, "Assembled entity body"); | |
603 | |
604 if (have_entire_body && si->content_encoding != NULL && | |
605 g_ascii_strcasecmp(si->content_encoding, "identity") != 0) { | |
606 /* | |
607 * We currently can't handle, for example, "compress"; | |
608 * just handle them as data for now. | |
609 * | |
610 * After July 7, 2004 the LZW patent expires, so support | |
611 * might be added then. However, I don't think that | |
612 * anybody ever really implemented "compress", due to | |
613 * the aforementioned patent. | |
614 */ | |
615 tvbuff_t *uncomp_tvb = NULL; | |
616 proto_item *e_ti = NULL; | |
617 proto_item *ce_ti = NULL; | |
618 proto_tree *e_tree = NULL; | |
619 | |
620 if (spdy_decompress_body && | |
621 (g_ascii_strcasecmp(si->content_encoding, "gzip") == 0 || | |
622 g_ascii_strcasecmp(si->content_encoding, "deflate") | |
623 == 0)) { | |
624 uncomp_tvb = spdy_tvb_child_uncompress(tvb, data_tvb, 0, | |
625 tvb_length(data_tvb)); | |
626 } | |
627 /* | |
628 * Add the encoded entity to the protocol tree | |
629 */ | |
630 e_ti = proto_tree_add_text(top_level_tree, data_tvb, | |
631 0, tvb_length(data_tvb), | |
632 "Content-encoded entity body (%s): %u bytes", | |
633 si->content_encoding, | |
634 tvb_length(data_tvb)); | |
635 e_tree = proto_item_add_subtree(e_ti, ett_spdy_encoded_entity); | |
636 if (si->num_data_frames > 1) { | |
637 GSList *dflist; | |
638 spdy_data_frame_t *df; | |
639 guint32 framenum; | |
640 ce_ti = proto_tree_add_text(e_tree, data_tvb, 0, | |
641 tvb_length(data_tvb), | |
642 "Assembled from %d frames in packet(s)", si->num_data_fr
ames); | |
643 dflist = si->data_frames; | |
644 framenum = 0; | |
645 while (dflist != NULL) { | |
646 df = dflist->data; | |
647 if (framenum != df->framenum) { | |
648 proto_item_append_text(ce_ti, " #%u", df->framenum); | |
649 framenum = df->framenum; | |
650 } | |
651 dflist = g_slist_next(dflist); | |
652 } | |
653 } | |
654 | |
655 if (uncomp_tvb != NULL) { | |
656 /* | |
657 * Decompression worked | |
658 */ | |
659 | |
660 /* XXX - Don't free this, since it's possible | |
661 * that the data was only partially | |
662 * decompressed, such as when desegmentation | |
663 * isn't enabled. | |
664 * | |
665 tvb_free(next_tvb); | |
666 */ | |
667 proto_item_append_text(e_ti, " -> %u bytes", tvb_length(uncomp_t
vb)); | |
668 data_tvb = uncomp_tvb; | |
669 add_new_data_source(pinfo, data_tvb, "Uncompressed entity body")
; | |
670 } else { | |
671 if (spdy_decompress_body) | |
672 proto_item_append_text(e_ti, " [Error: Decompression failed]
"); | |
673 call_dissector(data_handle, data_tvb, pinfo, e_tree); | |
674 | |
675 goto body_dissected; | |
676 } | |
677 } | |
678 if (si != NULL) | |
679 spdy_discard_data_frames(si); | |
680 /* | |
681 * Do subdissector checks. | |
682 * | |
683 * First, check whether some subdissector asked that they | |
684 * be called if something was on some particular port. | |
685 */ | |
686 | |
687 port_subdissector_table = find_dissector_table("http.port"); | |
688 media_type_subdissector_table = find_dissector_table("media_type"); | |
689 if (have_entire_body && port_subdissector_table != NULL) | |
690 handle = dissector_get_port_handle(port_subdissector_table, | |
691 pinfo->match_port); | |
692 else | |
693 handle = NULL; | |
694 if (handle == NULL && have_entire_body && si->content_type != NULL && | |
695 media_type_subdissector_table != NULL) { | |
696 /* | |
697 * We didn't find any subdissector that | |
698 * registered for the port, and we have a | |
699 * Content-Type value. Is there any subdissector | |
700 * for that content type? | |
701 */ | |
702 save_private_data = pinfo->private_data; | |
703 private_data_changed = TRUE; | |
704 | |
705 if (si->content_type_parameters) | |
706 pinfo->private_data = ep_strdup(si->content_type_parameters); | |
707 else | |
708 pinfo->private_data = NULL; | |
709 /* | |
710 * Calling the string handle for the media type | |
711 * dissector table will set pinfo->match_string | |
712 * to si->content_type for us. | |
713 */ | |
714 pinfo->match_string = si->content_type; | |
715 handle = dissector_get_string_handle( | |
716 media_type_subdissector_table, | |
717 si->content_type); | |
718 } | |
719 if (handle != NULL) { | |
720 /* | |
721 * We have a subdissector - call it. | |
722 */ | |
723 dissected = call_dissector(handle, data_tvb, pinfo, top_level_tree); | |
724 } else | |
725 dissected = FALSE; | |
726 | |
727 if (dissected) { | |
728 /* | |
729 * The subdissector dissected the body. | |
730 * Fix up the top-level item so that it doesn't | |
731 * include the stuff for that protocol. | |
732 */ | |
733 if (ti != NULL) | |
734 proto_item_set_len(ti, offset); | |
735 } else if (have_entire_body && si->content_type != NULL) { | |
736 /* | |
737 * Calling the default media handle if there is a content-type that | |
738 * wasn't handled above. | |
739 */ | |
740 call_dissector(media_handle, next_tvb, pinfo, top_level_tree); | |
741 } else { | |
742 /* Call the default data dissector */ | |
743 call_dissector(data_handle, next_tvb, pinfo, top_level_tree); | |
744 } | |
745 | |
746 body_dissected: | |
747 /* | |
748 * Do *not* attempt at freeing the private data; | |
749 * it may be in use by subdissectors. | |
750 */ | |
751 if (private_data_changed) /*restore even NULL value*/ | |
752 pinfo->private_data = save_private_data; | |
753 /* | |
754 * We've processed "datalen" bytes worth of data | |
755 * (which may be no data at all); advance the | |
756 * offset past whatever data we've processed. | |
757 */ | |
758 } | |
759 return frame_length + 8; | |
760 } | |
761 | |
762 static guint8 * | |
763 spdy_decompress_header_block(tvbuff_t *tvb, z_streamp decomp, | |
764 guint32 dictionary_id, int offset, | |
765 guint32 length, guint *uncomp_length) | |
766 { | |
767 int retcode; | |
768 size_t bufsize = 16384; | |
769 const guint8 *hptr = tvb_get_ptr(tvb, offset, length); | |
770 guint8 *uncomp_block = ep_alloc(bufsize); | |
771 decomp->next_in = (Bytef *)hptr; | |
772 decomp->avail_in = length; | |
773 decomp->next_out = uncomp_block; | |
774 decomp->avail_out = bufsize; | |
775 retcode = inflate(decomp, Z_SYNC_FLUSH); | |
776 if (retcode == Z_NEED_DICT) { | |
777 if (decomp->adler != dictionary_id) { | |
778 printf("decompressor wants dictionary %#x, but we have %#x\n", | |
779 (guint)decomp->adler, dictionary_id); | |
780 } else { | |
781 retcode = inflateSetDictionary(decomp, | |
782 spdy_dictionary, | |
783 sizeof(spdy_dictionary)); | |
784 if (retcode == Z_OK) | |
785 retcode = inflate(decomp, Z_SYNC_FLUSH); | |
786 } | |
787 } | |
788 | |
789 if (retcode != Z_OK) { | |
790 return NULL; | |
791 } else { | |
792 *uncomp_length = bufsize - decomp->avail_out; | |
793 if (spdy_debug) | |
794 printf("Inflation SUCCEEDED. uncompressed size=%d\n", *uncomp_length
); | |
795 if (decomp->avail_in != 0) | |
796 if (spdy_debug) | |
797 printf(" but there were %d input bytes left over\n", decomp->av
ail_in); | |
798 } | |
799 return se_memdup(uncomp_block, *uncomp_length); | |
800 } | |
801 | |
802 /* | |
803 * Try to determine heuristically whether the header block is | |
804 * compressed. For an uncompressed block, the first two bytes | |
805 * gives the number of headers. Each header name and value is | |
806 * a two-byte length followed by ASCII characters. | |
807 */ | |
808 static gboolean | |
809 spdy_check_header_compression(tvbuff_t *tvb, | |
810 int offset, | |
811 guint32 frame_length) | |
812 { | |
813 guint16 length; | |
814 if (!tvb_bytes_exist(tvb, offset, 6)) | |
815 return 1; | |
816 length = tvb_get_ntohs(tvb, offset); | |
817 if (length > frame_length) | |
818 return 1; | |
819 length = tvb_get_ntohs(tvb, offset+2); | |
820 if (length > frame_length) | |
821 return 1; | |
822 if (spdy_debug) printf("Looks like the header block is not compressed\n"); | |
823 return 0; | |
824 } | |
825 | |
826 // TODO(cbentzel): Change wireshark to export p_remove_proto_data, rather | |
827 // than duplicating code here. | |
828 typedef struct _spdy_frame_proto_data { | |
829 int proto; | |
830 void *proto_data; | |
831 } spdy_frame_proto_data; | |
832 | |
833 static gint spdy_p_compare(gconstpointer a, gconstpointer b) | |
834 { | |
835 const spdy_frame_proto_data *ap = (const spdy_frame_proto_data *)a; | |
836 const spdy_frame_proto_data *bp = (const spdy_frame_proto_data *)b; | |
837 | |
838 if (ap -> proto > bp -> proto) | |
839 return 1; | |
840 else if (ap -> proto == bp -> proto) | |
841 return 0; | |
842 else | |
843 return -1; | |
844 | |
845 } | |
846 | |
847 static void spdy_p_remove_proto_data(frame_data *fd, int proto) | |
848 { | |
849 spdy_frame_proto_data temp; | |
850 GSList *item; | |
851 | |
852 temp.proto = proto; | |
853 temp.proto_data = NULL; | |
854 | |
855 item = g_slist_find_custom(fd->pfd, (gpointer *)&temp, spdy_p_compare); | |
856 | |
857 if (item) { | |
858 fd->pfd = g_slist_remove(fd->pfd, item->data); | |
859 } | |
860 } | |
861 | |
862 static spdy_frame_info_t * | |
863 spdy_save_header_block(frame_data *fd, | |
864 guint32 stream_id, | |
865 guint frame_type, | |
866 guint8 *header, | |
867 guint length) | |
868 { | |
869 GSList *filist = p_get_proto_data(fd, proto_spdy); | |
870 spdy_frame_info_t *frame_info = se_alloc(sizeof(spdy_frame_info_t)); | |
871 if (filist != NULL) | |
872 spdy_p_remove_proto_data(fd, proto_spdy); | |
873 frame_info->stream_id = stream_id; | |
874 frame_info->header_block = header; | |
875 frame_info->header_block_len = length; | |
876 frame_info->frame_type = frame_type; | |
877 filist = g_slist_append(filist, frame_info); | |
878 p_add_proto_data(fd, proto_spdy, filist); | |
879 return frame_info; | |
880 /* TODO(ers) these need to get deleted when no longer needed */ | |
881 } | |
882 | |
883 static spdy_frame_info_t * | |
884 spdy_find_saved_header_block(frame_data *fd, | |
885 guint32 stream_id, | |
886 guint16 frame_type) | |
887 { | |
888 GSList *filist = p_get_proto_data(fd, proto_spdy); | |
889 while (filist != NULL) { | |
890 spdy_frame_info_t *fi = filist->data; | |
891 if (fi->stream_id == stream_id && fi->frame_type == frame_type) | |
892 return fi; | |
893 filist = g_slist_next(filist); | |
894 } | |
895 return NULL; | |
896 } | |
897 | |
898 /* | |
899 * Given a content type string that may contain optional parameters, | |
900 * return the parameter string, if any, otherwise return NULL. This | |
901 * also has the side effect of null terminating the content type | |
902 * part of the original string. | |
903 */ | |
904 static gchar * | |
905 spdy_parse_content_type(gchar *content_type) | |
906 { | |
907 gchar *cp = content_type; | |
908 | |
909 while (*cp != '\0' && *cp != ';' && !isspace(*cp)) { | |
910 *cp = tolower(*cp); | |
911 ++cp; | |
912 } | |
913 if (*cp == '\0') | |
914 cp = NULL; | |
915 | |
916 if (cp != NULL) { | |
917 *cp++ = '\0'; | |
918 while (*cp == ';' || isspace(*cp)) | |
919 ++cp; | |
920 if (*cp != '\0') | |
921 return cp; | |
922 } | |
923 return NULL; | |
924 } | |
925 | |
926 static int | |
927 dissect_spdy_message(tvbuff_t *tvb, int offset, packet_info *pinfo, | |
928 proto_tree *tree, spdy_conv_t *conv_data) | |
929 { | |
930 guint8 control_bit; | |
931 guint16 version; | |
932 guint16 frame_type; | |
933 guint8 flags; | |
934 guint32 frame_length; | |
935 guint32 stream_id; | |
936 guint32 associated_stream_id; | |
937 gint priority; | |
938 guint16 num_headers; | |
939 guint32 fin_status; | |
940 guint8 *frame_header; | |
941 const char *proto_tag; | |
942 const char *frame_type_name; | |
943 proto_tree *spdy_tree = NULL; | |
944 proto_item *ti = NULL; | |
945 proto_item *spdy_proto = NULL; | |
946 int orig_offset; | |
947 int hoffset; | |
948 int hdr_offset = 0; | |
949 spdy_frame_type_t spdy_type; | |
950 proto_tree *sub_tree; | |
951 proto_tree *flags_tree; | |
952 tvbuff_t *header_tvb = NULL; | |
953 gboolean headers_compressed; | |
954 gchar *hdr_verb = NULL; | |
955 gchar *hdr_url = NULL; | |
956 gchar *hdr_version = NULL; | |
957 gchar *content_type = NULL; | |
958 gchar *content_encoding = NULL; | |
959 | |
960 /* | |
961 * Minimum size for a SPDY frame is 8 bytes. | |
962 */ | |
963 if (tvb_reported_length_remaining(tvb, offset) < 8) | |
964 return -1; | |
965 | |
966 proto_tag = "SPDY"; | |
967 | |
968 if (check_col(pinfo->cinfo, COL_PROTOCOL)) | |
969 col_set_str(pinfo->cinfo, COL_PROTOCOL, proto_tag); | |
970 | |
971 /* | |
972 * Is this a control frame or a data frame? | |
973 */ | |
974 orig_offset = offset; | |
975 control_bit = tvb_get_bits8(tvb, offset << 3, 1); | |
976 if (control_bit) { | |
977 version = tvb_get_bits16(tvb, (offset << 3) + 1, 15, FALSE); | |
978 frame_type = tvb_get_ntohs(tvb, offset+2); | |
979 if (frame_type >= SPDY_INVALID) { | |
980 return -1; | |
981 } | |
982 frame_header = ep_tvb_memdup(tvb, offset, 16); | |
983 } else { | |
984 version = 1; /* avoid gcc warning */ | |
985 frame_type = SPDY_DATA; | |
986 frame_header = NULL; /* avoid gcc warning */ | |
987 } | |
988 frame_type_name = frame_type_names[frame_type]; | |
989 offset += 4; | |
990 flags = tvb_get_guint8(tvb, offset); | |
991 frame_length = tvb_get_ntoh24(tvb, offset+1); | |
992 offset += 4; | |
993 /* | |
994 * Make sure there's as much data as the frame header says there is. | |
995 */ | |
996 if ((guint)tvb_reported_length_remaining(tvb, offset) < frame_length) { | |
997 if (spdy_debug) | |
998 printf("Not enough header data: %d vs. %d\n", | |
999 frame_length, tvb_reported_length_remaining(tvb, offset)); | |
1000 return -1; | |
1001 } | |
1002 if (tree) { | |
1003 spdy_proto = proto_tree_add_item(tree, proto_spdy, tvb, orig_offset, fra
me_length+8, FALSE); | |
1004 spdy_tree = proto_item_add_subtree(spdy_proto, ett_spdy); | |
1005 } | |
1006 | |
1007 if (control_bit) { | |
1008 if (spdy_debug) | |
1009 printf("Control frame [version=%d type=%d flags=0x%x length=%d]\n", | |
1010 version, frame_type, flags, frame_length); | |
1011 if (tree) proto_item_append_text(spdy_tree, ", control frame"); | |
1012 } else { | |
1013 return dissect_spdy_data_frame(tvb, orig_offset, pinfo, tree, | |
1014 spdy_tree, spdy_proto, conv_data); | |
1015 } | |
1016 num_headers = 0; | |
1017 sub_tree = NULL; /* avoid gcc warning */ | |
1018 switch (frame_type) { | |
1019 case SPDY_SYN_STREAM: | |
1020 case SPDY_SYN_REPLY: | |
1021 if (tree) { | |
1022 int hf; | |
1023 hf = frame_type == SPDY_SYN_STREAM ? hf_spdy_syn_stream : hf_spd
y_syn_reply; | |
1024 ti = proto_tree_add_bytes(spdy_tree, hf, tvb, | |
1025 orig_offset, 16, frame_header); | |
1026 sub_tree = proto_item_add_subtree(ti, ett_spdy_syn_stream); | |
1027 } | |
1028 stream_id = tvb_get_bits32(tvb, (offset << 3) + 1, 31, FALSE); | |
1029 offset += 4; | |
1030 if (frame_type == SPDY_SYN_STREAM) { | |
1031 associated_stream_id = tvb_get_bits32(tvb, (offset << 3) + 1, 31
, FALSE); | |
1032 offset += 4; | |
1033 priority = tvb_get_bits8(tvb, offset << 3, 2); | |
1034 offset += 2; | |
1035 } else { | |
1036 // The next two bytes have no meaning in SYN_REPLY | |
1037 offset += 2; | |
1038 } | |
1039 if (tree) { | |
1040 proto_tree_add_boolean(sub_tree, hf_spdy_control_bit, tvb, orig_
offset, 1, control_bit); | |
1041 proto_tree_add_uint(sub_tree, hf_spdy_version, tvb, orig_offset,
2, version); | |
1042 proto_tree_add_uint(sub_tree, hf_spdy_type, tvb, orig_offset+2,
2, frame_type); | |
1043 ti = proto_tree_add_uint_format(sub_tree, hf_spdy_flags, tvb, or
ig_offset+4, 1, flags, | |
1044 "Flags: 0x%02x%s", flags, flags&
SPDY_FIN ? " (FIN)" : ""); | |
1045 flags_tree = proto_item_add_subtree(ti, ett_spdy_flags); | |
1046 proto_tree_add_boolean(flags_tree, hf_spdy_flags_fin, tvb, orig_
offset+4, 1, flags); | |
1047 proto_tree_add_uint(sub_tree, hf_spdy_length, tvb, orig_offset+5
, 3, frame_length); | |
1048 proto_tree_add_uint(sub_tree, hf_spdy_streamid, tvb, orig_offset
+8, 4, stream_id); | |
1049 if (frame_type == SPDY_SYN_STREAM) { | |
1050 proto_tree_add_uint(sub_tree, hf_spdy_associated_streamid,
tvb, orig_offset+12, 4, associated_stream_id); | |
1051 proto_tree_add_uint(sub_tree, hf_spdy_priority, tvb, orig_o
ffset+16, 1, priority); | |
1052 } | |
1053 proto_item_append_text(spdy_proto, ": %s%s stream=%d length=%d", | |
1054 frame_type_name, | |
1055 flags & SPDY_FIN ? " [FIN]" : "", | |
1056 stream_id, frame_length); | |
1057 if (spdy_debug) | |
1058 printf(" stream ID=%u priority=%d\n", stream_id, priority); | |
1059 } | |
1060 break; | |
1061 | |
1062 case SPDY_FIN_STREAM: | |
1063 stream_id = tvb_get_bits32(tvb, (offset << 3) + 1, 31, FALSE); | |
1064 fin_status = tvb_get_ntohl(tvb, offset); | |
1065 // TODO(ers) fill in tree and summary | |
1066 offset += 8; | |
1067 break; | |
1068 | |
1069 case SPDY_HELLO: | |
1070 // TODO(ers) fill in tree and summary | |
1071 stream_id = 0; /* avoid gcc warning */ | |
1072 break; | |
1073 | |
1074 default: | |
1075 stream_id = 0; /* avoid gcc warning */ | |
1076 return -1; | |
1077 break; | |
1078 } | |
1079 | |
1080 /* | |
1081 * Process the name-value pairs one at a time, after possibly | |
1082 * decompressing the header block. | |
1083 */ | |
1084 if (frame_type == SPDY_SYN_STREAM || frame_type == SPDY_SYN_REPLY) { | |
1085 headers_compressed = spdy_check_header_compression(tvb, offset, frame_le
ngth); | |
1086 if (!spdy_decompress_headers || !headers_compressed) { | |
1087 header_tvb = tvb; | |
1088 hdr_offset = offset; | |
1089 } else { | |
1090 spdy_frame_info_t *per_frame_info = | |
1091 spdy_find_saved_header_block(pinfo->fd, | |
1092 stream_id, | |
1093 frame_type == SPDY_SYN_REPLY); | |
1094 if (per_frame_info == NULL) { | |
1095 guint uncomp_length; | |
1096 z_streamp decomp = frame_type == SPDY_SYN_STREAM ? | |
1097 conv_data->rqst_decompressor : conv_data->rply_decompres
sor; | |
1098 guint8 *uncomp_ptr = | |
1099 spdy_decompress_header_block(tvb, decomp, | |
1100 conv_data->dictionary_id, | |
1101 offset, | |
1102 frame_length + 8 - (offset
- orig_offset), | |
1103 &uncomp_length); | |
1104 if (uncomp_ptr == NULL) { /* decompression failed */ | |
1105 if (spdy_debug) | |
1106 printf("Frame #%d: Inflation failed\n", pinfo->fd->num); | |
1107 proto_item_append_text(spdy_proto, " [Error: Header decompre
ssion failed]"); | |
1108 // Should we just bail here? | |
1109 } else { | |
1110 if (spdy_debug) | |
1111 printf("Saving %u bytes of uncomp hdr\n", uncomp_length)
; | |
1112 per_frame_info = | |
1113 spdy_save_header_block(pinfo->fd, stream_id, frame_type
== SPDY_SYN_REPLY, | |
1114 uncomp_ptr, uncomp_length); | |
1115 } | |
1116 } else if (spdy_debug) { | |
1117 printf("Found uncompressed header block len %u for stream %u fra
me_type=%d\n", | |
1118 per_frame_info->header_block_len, | |
1119 per_frame_info->stream_id, | |
1120 per_frame_info->frame_type); | |
1121 } | |
1122 if (per_frame_info != NULL) { | |
1123 header_tvb = tvb_new_child_real_data(tvb, | |
1124 per_frame_info->header_block, | |
1125 per_frame_info->header_block_le
n, | |
1126 per_frame_info->header_block_le
n); | |
1127 add_new_data_source(pinfo, header_tvb, "Uncompressed headers"); | |
1128 hdr_offset = 0; | |
1129 } | |
1130 } | |
1131 offset = orig_offset + 8 + frame_length; | |
1132 num_headers = tvb_get_ntohs(header_tvb, hdr_offset); | |
1133 hdr_offset += 2; | |
1134 if (header_tvb == NULL || | |
1135 (headers_compressed && !spdy_decompress_headers)) { | |
1136 num_headers = 0; | |
1137 ti = proto_tree_add_string(sub_tree, hf_spdy_num_headers_string, | |
1138 tvb, | |
1139 frame_type == SPDY_SYN_STREAM ? orig_offset+18
: orig_offset + 14, | |
1140 2, | |
1141 "Unknown (header block is compressed)"); | |
1142 } else | |
1143 ti = proto_tree_add_uint(sub_tree, hf_spdy_num_headers, | |
1144 tvb, | |
1145 frame_type == SPDY_SYN_STREAM ? orig_offset+18 :
orig_offset +14, | |
1146 2, num_headers); | |
1147 } | |
1148 spdy_type = SPDY_INVALID; /* type not known yet */ | |
1149 if (spdy_debug) | |
1150 printf(" %d Headers:\n", num_headers); | |
1151 if (num_headers > frame_length) { | |
1152 printf("Number of headers is greater than frame length!\n"); | |
1153 proto_item_append_text(ti, " [Error: Number of headers is larger than fr
ame length]"); | |
1154 col_add_fstr(pinfo->cinfo, COL_INFO, "%s[%d]", frame_type_name, stream_i
d); | |
1155 return frame_length+8; | |
1156 } | |
1157 hdr_verb = hdr_url = hdr_version = content_type = content_encoding = NULL; | |
1158 while (num_headers-- && tvb_reported_length_remaining(header_tvb, hdr_offset
) != 0) { | |
1159 gchar *header_name; | |
1160 gchar *header_value; | |
1161 proto_tree *header_tree; | |
1162 proto_tree *name_tree; | |
1163 proto_tree *value_tree; | |
1164 proto_item *header; | |
1165 gint16 length; | |
1166 gint header_length = 0; | |
1167 | |
1168 hoffset = hdr_offset; | |
1169 | |
1170 header = proto_tree_add_item(spdy_tree, hf_spdy_header, header_tvb, | |
1171 hdr_offset, frame_length, FALSE); | |
1172 header_tree = proto_item_add_subtree(header, ett_spdy_header); | |
1173 | |
1174 length = tvb_get_ntohs(header_tvb, hdr_offset); | |
1175 hdr_offset += 2; | |
1176 header_name = (gchar *)tvb_get_ephemeral_string(header_tvb, hdr_offset,
length); | |
1177 hdr_offset += length; | |
1178 header_length += hdr_offset - hoffset; | |
1179 if (tree) { | |
1180 ti = proto_tree_add_text(header_tree, header_tvb, hoffset, length+2,
"Name: %s", | |
1181 header_name); | |
1182 name_tree = proto_item_add_subtree(ti, ett_spdy_header_name); | |
1183 proto_tree_add_uint(name_tree, hf_spdy_length, header_tvb, hoffset,
2, length); | |
1184 proto_tree_add_string_format(name_tree, hf_spdy_header_name_text, he
ader_tvb, hoffset+2, length, | |
1185 header_name, "Text: %s", format_text(he
ader_name, length)); | |
1186 } | |
1187 | |
1188 hoffset = hdr_offset; | |
1189 length = tvb_get_ntohs(header_tvb, hdr_offset); | |
1190 hdr_offset += 2; | |
1191 header_value = (gchar *)tvb_get_ephemeral_string(header_tvb, hdr_offset,
length); | |
1192 hdr_offset += length; | |
1193 header_length += hdr_offset - hoffset; | |
1194 if (tree) { | |
1195 ti = proto_tree_add_text(header_tree, header_tvb, hoffset, length+2,
"Value: %s", | |
1196 header_value); | |
1197 value_tree = proto_item_add_subtree(ti, ett_spdy_header_value); | |
1198 proto_tree_add_uint(value_tree, hf_spdy_length, header_tvb, hoffset,
2, length); | |
1199 proto_tree_add_string_format(value_tree, hf_spdy_header_value_text,
header_tvb, hoffset+2, length, | |
1200 header_value, "Text: %s", format_text(h
eader_value, length)); | |
1201 proto_item_append_text(header, ": %s: %s", header_name, header_value
); | |
1202 proto_item_set_len(header, header_length); | |
1203 } | |
1204 if (spdy_debug) printf(" %s: %s\n", header_name, header_value); | |
1205 /* | |
1206 * TODO(ers) check that the header name contains only legal characters. | |
1207 */ | |
1208 if (g_ascii_strcasecmp(header_name, "method") == 0 || | |
1209 g_ascii_strcasecmp(header_name, "status") == 0) { | |
1210 hdr_verb = header_value; | |
1211 } else if (g_ascii_strcasecmp(header_name, "url") == 0) { | |
1212 hdr_url = header_value; | |
1213 } else if (g_ascii_strcasecmp(header_name, "version") == 0) { | |
1214 hdr_version = header_value; | |
1215 } else if (g_ascii_strcasecmp(header_name, "content-type") == 0) { | |
1216 content_type = se_strdup(header_value); | |
1217 } else if (g_ascii_strcasecmp(header_name, "content-encoding") == 0) { | |
1218 content_encoding = se_strdup(header_value); | |
1219 } | |
1220 } | |
1221 if (hdr_version != NULL) { | |
1222 if (hdr_url != NULL) { | |
1223 col_add_fstr(pinfo->cinfo, COL_INFO, "%s[%d]: %s %s %s", | |
1224 frame_type_name, stream_id, hdr_verb, hdr_url, hdr_vers
ion); | |
1225 } else { | |
1226 col_add_fstr(pinfo->cinfo, COL_INFO, "%s[%d]: %s %s", | |
1227 frame_type_name, stream_id, hdr_verb, hdr_version); | |
1228 } | |
1229 } else { | |
1230 col_add_fstr(pinfo->cinfo, COL_INFO, "%s[%d]", frame_type_name, stream_i
d); | |
1231 } | |
1232 /* | |
1233 * If we expect data on this stream, we need to remember the content | |
1234 * type and content encoding. | |
1235 */ | |
1236 if (content_type != NULL && !pinfo->fd->flags.visited) { | |
1237 gchar *content_type_params = spdy_parse_content_type(content_type); | |
1238 spdy_save_stream_info(conv_data, stream_id, content_type, | |
1239 content_type_params, content_encoding); | |
1240 } | |
1241 | |
1242 return offset - orig_offset; | |
1243 } | |
1244 | |
1245 static int | |
1246 dissect_spdy(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) | |
1247 { | |
1248 spdy_conv_t *conv_data; | |
1249 int offset = 0; | |
1250 int len; | |
1251 int firstpkt = 1; | |
1252 | |
1253 /* | |
1254 * The first byte of a SPDY packet must be either 0 or | |
1255 * 0x80. If it's not, assume that this is not SPDY. | |
1256 * (In theory, a data frame could have a stream ID | |
1257 * >= 2^24, in which case it won't have 0 for a first | |
1258 * byte, but this is a pretty reliable heuristic for | |
1259 * now.) | |
1260 */ | |
1261 guint8 first_byte = tvb_get_guint8(tvb, 0); | |
1262 if (first_byte != 0x80 && first_byte != 0x0) | |
1263 return 0; | |
1264 | |
1265 conv_data = get_spdy_conversation_data(pinfo); | |
1266 | |
1267 while (tvb_reported_length_remaining(tvb, offset) != 0) { | |
1268 if (!firstpkt) { | |
1269 col_add_fstr(pinfo->cinfo, COL_INFO, " >> "); | |
1270 col_set_fence(pinfo->cinfo, COL_INFO); | |
1271 } | |
1272 len = dissect_spdy_message(tvb, offset, pinfo, tree, conv_data); | |
1273 if (len <= 0) | |
1274 return 0; | |
1275 offset += len; | |
1276 /* | |
1277 * OK, we've set the Protocol and Info columns for the | |
1278 * first SPDY message; set a fence so that subsequent | |
1279 * SPDY messages don't overwrite the Info column. | |
1280 */ | |
1281 col_set_fence(pinfo->cinfo, COL_INFO); | |
1282 firstpkt = 0; | |
1283 } | |
1284 return 1; | |
1285 } | |
1286 | |
1287 static gboolean | |
1288 dissect_spdy_heur(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) | |
1289 { | |
1290 if (!value_is_in_range(global_spdy_tcp_range, pinfo->destport) && | |
1291 !value_is_in_range(global_spdy_tcp_range, pinfo->srcport)) | |
1292 return FALSE; | |
1293 return dissect_spdy(tvb, pinfo, tree) != 0; | |
1294 } | |
1295 | |
1296 static void reinit_spdy(void) | |
1297 { | |
1298 } | |
1299 | |
1300 // NMAKE complains about flags_set_truth not being constant. Duplicate | |
1301 // the values inside of it. | |
1302 static const true_false_string tfs_spdy_set_notset = { "Set", "Not set" }; | |
1303 | |
1304 void | |
1305 proto_register_spdy(void) | |
1306 { | |
1307 static hf_register_info hf[] = { | |
1308 { &hf_spdy_syn_stream, | |
1309 { "Syn Stream", "spdy.syn_stream", | |
1310 FT_BYTES, BASE_NONE, NULL, 0x0, | |
1311 "", HFILL }}, | |
1312 { &hf_spdy_syn_reply, | |
1313 { "Syn Reply", "spdy.syn_reply", | |
1314 FT_BYTES, BASE_NONE, NULL, 0x0, | |
1315 "", HFILL }}, | |
1316 { &hf_spdy_control_bit, | |
1317 { "Control bit", "spdy.control_bit", | |
1318 FT_BOOLEAN, BASE_NONE, NULL, 0x0, | |
1319 "TRUE if SPDY control frame", HFILL }}, | |
1320 { &hf_spdy_version, | |
1321 { "Version", "spdy.version", | |
1322 FT_UINT16, BASE_DEC, NULL, 0x0, | |
1323 "", HFILL }}, | |
1324 { &hf_spdy_type, | |
1325 { "Type", "spdy.type", | |
1326 FT_UINT16, BASE_DEC, NULL, 0x0, | |
1327 "", HFILL }}, | |
1328 { &hf_spdy_flags, | |
1329 { "Flags", "spdy.flags", | |
1330 FT_UINT8, BASE_HEX, NULL, 0x0, | |
1331 "", HFILL }}, | |
1332 { &hf_spdy_flags_fin, | |
1333 { "Fin", "spdy.flags.fin", | |
1334 FT_BOOLEAN, 8, TFS(&tfs_spdy_set_notset), | |
1335 SPDY_FIN, "", HFILL }}, | |
1336 { &hf_spdy_length, | |
1337 { "Length", "spdy.length", | |
1338 FT_UINT24, BASE_DEC, NULL, 0x0, | |
1339 "", HFILL }}, | |
1340 { &hf_spdy_header, | |
1341 { "Header", "spdy.header", | |
1342 FT_NONE, BASE_NONE, NULL, 0x0, | |
1343 "", HFILL }}, | |
1344 { &hf_spdy_header_name, | |
1345 { "Name", "spdy.header.name", | |
1346 FT_NONE, BASE_NONE, NULL, 0x0, | |
1347 "", HFILL }}, | |
1348 { &hf_spdy_header_name_text, | |
1349 { "Text", "spdy.header.name.text", | |
1350 FT_STRING, BASE_NONE, NULL, 0x0, | |
1351 "", HFILL }}, | |
1352 { &hf_spdy_header_value, | |
1353 { "Value", "spdy.header.value", | |
1354 FT_NONE, BASE_NONE, NULL, 0x0, | |
1355 "", HFILL }}, | |
1356 { &hf_spdy_header_value_text, | |
1357 { "Text", "spdy.header.value.text", | |
1358 FT_STRING, BASE_NONE, NULL, 0x0, | |
1359 "", HFILL }}, | |
1360 { &hf_spdy_streamid, | |
1361 { "Stream ID", "spdy.streamid", | |
1362 FT_UINT32, BASE_DEC, NULL, 0x0, | |
1363 "", HFILL }}, | |
1364 { &hf_spdy_associated_streamid, | |
1365 { "Associated Stream ID", "spdy.associated.streamid", | |
1366 FT_UINT32, BASE_DEC, NULL, 0x0, | |
1367 "", HFILL }}, | |
1368 { &hf_spdy_priority, | |
1369 { "Priority", "spdy.priority", | |
1370 FT_UINT8, BASE_DEC, NULL, 0x0, | |
1371 "", HFILL }}, | |
1372 { &hf_spdy_num_headers, | |
1373 { "Number of headers", "spdy.numheaders", | |
1374 FT_UINT16, BASE_DEC, NULL, 0x0, | |
1375 "", HFILL }}, | |
1376 { &hf_spdy_num_headers_string, | |
1377 { "Number of headers", "spdy.numheaders", | |
1378 FT_STRING, BASE_NONE, NULL, 0x0, | |
1379 "", HFILL }}, | |
1380 }; | |
1381 static gint *ett[] = { | |
1382 &ett_spdy, | |
1383 &ett_spdy_syn_stream, | |
1384 &ett_spdy_syn_reply, | |
1385 &ett_spdy_fin_stream, | |
1386 &ett_spdy_flags, | |
1387 &ett_spdy_header, | |
1388 &ett_spdy_header_name, | |
1389 &ett_spdy_header_value, | |
1390 &ett_spdy_encoded_entity, | |
1391 }; | |
1392 | |
1393 module_t *spdy_module; | |
1394 | |
1395 proto_spdy = proto_register_protocol("SPDY", "SPDY", "spdy"); | |
1396 proto_register_field_array(proto_spdy, hf, array_length(hf)); | |
1397 proto_register_subtree_array(ett, array_length(ett)); | |
1398 new_register_dissector("spdy", dissect_spdy, proto_spdy); | |
1399 spdy_module = prefs_register_protocol(proto_spdy, reinit_spdy); | |
1400 prefs_register_bool_preference(spdy_module, "desegment_headers", | |
1401 "Reassemble SPDY control frames spanning mult
iple TCP segments", | |
1402 "Whether the SPDY dissector should reassemble
control frames " | |
1403 "spanning multiple TCP segments. " | |
1404 "To use this option, you must also enable " | |
1405 "\"Allow subdissectors to reassemble TCP stre
ams\" in the TCP protocol settings.", | |
1406 &spdy_desegment_control_frames); | |
1407 prefs_register_bool_preference(spdy_module, "desegment_body", | |
1408 "Reassemble SPDY bodies spanning multiple TCP
segments", | |
1409 "Whether the SPDY dissector should reassemble
" | |
1410 "data frames spanning multiple TCP segments.
" | |
1411 "To use this option, you must also enable " | |
1412 "\"Allow subdissectors to reassemble TCP stre
ams\" in the TCP protocol settings.", | |
1413 &spdy_desegment_data_frames); | |
1414 prefs_register_bool_preference(spdy_module, "assemble_data_frames", | |
1415 "Assemble SPDY bodies that consist of multipl
e DATA frames", | |
1416 "Whether the SPDY dissector should reassemble
multiple " | |
1417 "data frames into an entity body.", | |
1418 &spdy_assemble_entity_bodies); | |
1419 #ifdef HAVE_LIBZ | |
1420 prefs_register_bool_preference(spdy_module, "decompress_headers", | |
1421 "Uncompress SPDY headers", | |
1422 "Whether to uncompress SPDY headers.", | |
1423 &spdy_decompress_headers); | |
1424 prefs_register_bool_preference(spdy_module, "decompress_body", | |
1425 "Uncompress entity bodies", | |
1426 "Whether to uncompress entity bodies that are
compressed " | |
1427 "using \"Content-Encoding: \"", | |
1428 &spdy_decompress_body); | |
1429 #endif | |
1430 prefs_register_bool_preference(spdy_module, "debug_output", | |
1431 "Print debug info on stdout", | |
1432 "Print debug info on stdout", | |
1433 &spdy_debug); | |
1434 #if 0 | |
1435 prefs_register_string_preference(ssl_module, "debug_file", "SPDY debug file"
, | |
1436 "Redirect SPDY debug to file name; " | |
1437 "leave empty to disable debugging, " | |
1438 "or use \"" SPDY_DEBUG_USE_STDOUT "\"" | |
1439 " to redirect output to stdout\n", | |
1440 (const gchar **)&sdpy_debug_file_name); | |
1441 #endif | |
1442 prefs_register_obsolete_preference(spdy_module, "tcp_alternate_port"); | |
1443 | |
1444 range_convert_str(&global_spdy_tcp_range, TCP_DEFAULT_RANGE, 65535); | |
1445 spdy_tcp_range = range_empty(); | |
1446 prefs_register_range_preference(spdy_module, "tcp.port", "TCP Ports", | |
1447 "TCP Ports range", | |
1448 &global_spdy_tcp_range, 65535); | |
1449 | |
1450 range_convert_str(&global_spdy_ssl_range, SSL_DEFAULT_RANGE, 65535); | |
1451 spdy_ssl_range = range_empty(); | |
1452 prefs_register_range_preference(spdy_module, "ssl.port", "SSL/TLS Ports", | |
1453 "SSL/TLS Ports range", | |
1454 &global_spdy_ssl_range, 65535); | |
1455 | |
1456 spdy_handle = new_create_dissector_handle(dissect_spdy, proto_spdy); | |
1457 /* | |
1458 * Register for tapping | |
1459 */ | |
1460 spdy_tap = register_tap("spdy"); /* SPDY statistics tap */ | |
1461 spdy_eo_tap = register_tap("spdy_eo"); /* SPDY Export Object tap */ | |
1462 } | |
1463 | |
1464 void | |
1465 proto_reg_handoff_spdy(void) | |
1466 { | |
1467 data_handle = find_dissector("data"); | |
1468 media_handle = find_dissector("media"); | |
1469 heur_dissector_add("tcp", dissect_spdy_heur, proto_spdy); | |
1470 } | |
1471 | |
1472 /* | |
1473 * Content-Type: message/http | |
1474 */ | |
1475 | |
1476 static gint proto_message_spdy = -1; | |
1477 static gint ett_message_spdy = -1; | |
1478 | |
1479 static void | |
1480 dissect_message_spdy(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) | |
1481 { | |
1482 proto_tree *subtree; | |
1483 proto_item *ti; | |
1484 gint offset = 0, next_offset; | |
1485 gint len; | |
1486 | |
1487 if (check_col(pinfo->cinfo, COL_INFO)) | |
1488 col_append_str(pinfo->cinfo, COL_INFO, " (message/spdy)"); | |
1489 if (tree) { | |
1490 ti = proto_tree_add_item(tree, proto_message_spdy, | |
1491 tvb, 0, -1, FALSE); | |
1492 subtree = proto_item_add_subtree(ti, ett_message_spdy); | |
1493 while (tvb_reported_length_remaining(tvb, offset) != 0) { | |
1494 len = tvb_find_line_end(tvb, offset, | |
1495 tvb_ensure_length_remaining(tvb, offset)
, | |
1496 &next_offset, FALSE); | |
1497 if (len == -1) | |
1498 break; | |
1499 proto_tree_add_text(subtree, tvb, offset, next_offset -
offset, | |
1500 "%s", tvb_format_text(tvb, offset, len))
; | |
1501 offset = next_offset; | |
1502 } | |
1503 } | |
1504 } | |
1505 | |
1506 void | |
1507 proto_register_message_spdy(void) | |
1508 { | |
1509 static gint *ett[] = { | |
1510 &ett_message_spdy, | |
1511 }; | |
1512 | |
1513 proto_message_spdy = proto_register_protocol( | |
1514 "Media Type: message/spdy", | |
1515 "message/spdy", | |
1516 "message-spdy" | |
1517 ); | |
1518 proto_register_subtree_array(ett, array_length(ett)); | |
1519 } | |
1520 | |
1521 void | |
1522 proto_reg_handoff_message_spdy(void) | |
1523 { | |
1524 dissector_handle_t message_spdy_handle; | |
1525 | |
1526 message_spdy_handle = create_dissector_handle(dissect_message_spdy, | |
1527 proto_message_spdy); | |
1528 | |
1529 dissector_add_string("media_type", "message/spdy", message_spdy_handle); | |
1530 | |
1531 reinit_spdy(); | |
1532 } | |
OLD | NEW |