diff options
Diffstat (limited to 'packet-rtsp.c')
-rw-r--r-- | packet-rtsp.c | 1217 |
1 files changed, 0 insertions, 1217 deletions
diff --git a/packet-rtsp.c b/packet-rtsp.c deleted file mode 100644 index 1a1453494e..0000000000 --- a/packet-rtsp.c +++ /dev/null @@ -1,1217 +0,0 @@ -/* packet-rtsp.c - * Routines for RTSP packet disassembly (RFC 2326) - * - * Jason Lango <jal@netapp.com> - * Liberally copied from packet-http.c, by Guy Harris <guy@alum.mit.edu> - * - * $Id$ - * - * Ethereal - Network traffic analyzer - * By Gerald Combs <gerald@ethereal.com> - * Copyright 1998 Gerald Combs - * - * This program is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * of the License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. - * - * References: - * RTSP is defined in RFC 2326, http://www.ietf.org/rfc/rfc2326.txt?number=2326 - * http://www.iana.org/assignments/rsvp-parameters - */ - -#include "config.h" - -#include <string.h> -#include <ctype.h> -#include <stdlib.h> - -#include "prefs.h" - -#include <glib.h> -#include <epan/packet.h> -#include "req_resp_hdrs.h" -#include "packet-rtp.h" -#include "packet-rtcp.h" -#include <epan/conversation.h> -#include <epan/strutil.h> -#include "packet-e164.h" - -static int proto_rtsp = -1; - -static gint ett_rtsp = -1; -static gint ett_rtspframe = -1; -static gint ett_rtsp_method = -1; - -static int hf_rtsp_method = -1; -static int hf_rtsp_url = -1; -static int hf_rtsp_status = -1; -static int hf_rtsp_session = -1; -static int hf_rtsp_X_Vig_Msisdn = -1; - -static dissector_handle_t sdp_handle; -static dissector_handle_t rtp_handle; -static dissector_handle_t rtcp_handle; - -void proto_reg_handoff_rtsp(void); - -static GMemChunk *rtsp_vals = NULL; -#define rtsp_hash_init_count 20 - -/* - * desegmentation of RTSP headers - * (when we are over TCP or another protocol providing the desegmentation API) - */ -static gboolean rtsp_desegment_headers = FALSE; - -/* - * desegmentation of RTSP bodies - * (when we are over TCP or another protocol providing the desegmentation API) - * TODO let the user filter on content-type the bodies he wants desegmented - */ -static gboolean rtsp_desegment_body = FALSE; - -/* http://www.iana.org/assignments/port-numberslists two rtsp ports */ -#define TCP_PORT_RTSP 554 -#define TCP_ALTERNATE_PORT_RTSP 8554 -static guint global_rtsp_tcp_port = TCP_PORT_RTSP; -static guint global_rtsp_tcp_alternate_port = TCP_ALTERNATE_PORT_RTSP; -/* -* Variables to allow for proper deletion of dissector registration when -* the user changes port from the gui. -*/ -static guint tcp_port = 0; -static guint tcp_alternate_port = 0; - -/* - * Takes an array of bytes, assumed to contain a null-terminated - * string, as an argument, and returns the length of the string - - * i.e., the size of the array, minus 1 for the null terminator. - */ -#define STRLEN_CONST(str) (sizeof (str) - 1) - -#define RTSP_FRAMEHDR ('$') - -typedef struct { - dissector_handle_t dissector; -} rtsp_interleaved_t; - -#define RTSP_MAX_INTERLEAVED (8) - -/* - * Careful about dynamically allocating memory in this structure (say - * for dynamically increasing the size of the 'interleaved' array) - - * the containing structure is garbage collected and contained - * pointers will not be freed. - */ -typedef struct { - rtsp_interleaved_t interleaved[RTSP_MAX_INTERLEAVED]; -} rtsp_conversation_data_t; - -static int -dissect_rtspinterleaved(tvbuff_t *tvb, int offset, packet_info *pinfo, - proto_tree *tree) -{ - guint length_remaining; - proto_item *ti; - proto_tree *rtspframe_tree = NULL; - int orig_offset; - guint8 rf_start; /* always RTSP_FRAMEHDR */ - guint8 rf_chan; /* interleaved channel id */ - guint16 rf_len; /* packet length */ - tvbuff_t *next_tvb; - conversation_t *conv; - rtsp_conversation_data_t *data; - dissector_handle_t dissector; - - /* - * This will throw an exception if we don't have any data left. - * That's what we want. (See "tcp_dissect_pdus()", which is - * similar.) - */ - length_remaining = tvb_ensure_length_remaining(tvb, offset); - - /* - * Can we do reassembly? - */ - if (rtsp_desegment_headers && pinfo->can_desegment) { - /* - * Yes - would an RTSP multiplexed header starting at - * this offset be split across segment boundaries? - */ - if (length_remaining < 4) { - /* - * Yes. Tell the TCP dissector where the data - * for this message starts in the data it handed - * us, and how many more bytes we need, and return. - */ - pinfo->desegment_offset = offset; - pinfo->desegment_len = 4 - length_remaining; - return -1; - } - } - - /* - * Get the "$", channel, and length from the header. - */ - orig_offset = offset; - rf_start = tvb_get_guint8(tvb, offset); - rf_chan = tvb_get_guint8(tvb, offset+1); - rf_len = tvb_get_ntohs(tvb, offset+2); - - /* - * Can we do reassembly? - */ - if (rtsp_desegment_body && pinfo->can_desegment) { - /* - * Yes - is the header + encapsulated packet split - * across segment boundaries? - */ - if (length_remaining < 4U + rf_len) { - /* - * Yes. Tell the TCP dissector where the data - * for this message starts in the data it handed - * us, and how many more bytes we need, and return. - */ - pinfo->desegment_offset = offset; - pinfo->desegment_len = 4U + rf_len - length_remaining; - return -1; - } - } - - if (check_col(pinfo->cinfo, COL_INFO)) - col_add_fstr(pinfo->cinfo, COL_INFO, - "Interleaved channel 0x%02x, %u bytes", - rf_chan, rf_len); - - if (tree != NULL) { - ti = proto_tree_add_protocol_format(tree, proto_rtsp, tvb, - offset, 4, - "RTSP Interleaved Frame, Channel: 0x%02x, %u bytes", - rf_chan, rf_len); - rtspframe_tree = proto_item_add_subtree(ti, ett_rtspframe); - - proto_tree_add_text(rtspframe_tree, tvb, offset, 1, - "Magic: 0x%02x", - rf_start); - } - offset += 1; - - if (tree != NULL) { - proto_tree_add_text(rtspframe_tree, tvb, offset, 1, - "Channel: 0x%02x", - rf_chan); - } - offset += 1; - - if (tree != NULL) { - proto_tree_add_text(rtspframe_tree, tvb, offset, 2, - "Length: %u bytes", - rf_len); - } - offset += 2; - - /* - * We set the actual length of the tvbuff for the interleaved - * stuff to the minimum of what's left in the tvbuff and the - * length in the header. - * - * XXX - what if there's nothing left in the tvbuff? - * We'd want a BoundsError exception to be thrown, so - * that a Short Frame would be reported. - */ - if (length_remaining > rf_len) - length_remaining = rf_len; - next_tvb = tvb_new_subset(tvb, offset, length_remaining, rf_len); - - conv = find_conversation(&pinfo->src, &pinfo->dst, pinfo->ptype, - pinfo->srcport, pinfo->destport, 0); - - if (conv && - (data = conversation_get_proto_data(conv, proto_rtsp)) && - rf_chan < RTSP_MAX_INTERLEAVED && - (dissector = data->interleaved[rf_chan].dissector)) { - call_dissector(dissector, next_tvb, pinfo, tree); - } else { - proto_tree_add_text(rtspframe_tree, tvb, offset, rf_len, - "Data (%u bytes)", rf_len); - } - - offset += rf_len; - - return offset - orig_offset; -} - -static void process_rtsp_request(tvbuff_t *tvb, int offset, const guchar *data, - size_t linelen, proto_tree *tree); - -static void process_rtsp_reply(tvbuff_t *tvb, int offset, const guchar *data, - size_t linelen, proto_tree *tree); - -typedef enum { - RTSP_REQUEST, - RTSP_REPLY, - NOT_RTSP -} rtsp_type_t; - -static const char *rtsp_methods[] = { - "DESCRIBE", - "ANNOUNCE", - "GET_PARAMETER", - "OPTIONS", - "PAUSE", - "PLAY", - "RECORD", - "REDIRECT", - "SETUP", - "SET_PARAMETER", - "TEARDOWN" -}; - -#define RTSP_NMETHODS (sizeof rtsp_methods / sizeof rtsp_methods[0]) - -static gboolean -is_rtsp_request_or_reply(const guchar *line, size_t linelen, rtsp_type_t *type) -{ - unsigned ii; - - /* Is this an RTSP reply? */ - if (linelen >= 5 && strncasecmp("RTSP/", line, 5) == 0) { - /* - * Yes. - */ - *type = RTSP_REPLY; - return TRUE; - } - - /* - * Is this an RTSP request? - * Check whether the line begins with one of the RTSP request - * methods. - */ - for (ii = 0; ii < RTSP_NMETHODS; ii++) { - size_t len = strlen(rtsp_methods[ii]); - if (linelen >= len && - strncasecmp(rtsp_methods[ii], line, len) == 0 && - (len == linelen || isspace(line[len]))) { - *type = RTSP_REQUEST; - return TRUE; - } - } - *type = NOT_RTSP; - return FALSE; -} - -static const char rtsp_content_type[] = "Content-Type:"; - -static int -is_content_sdp(const guchar *line, size_t linelen) -{ - static const char type[] = "application/sdp"; - size_t typelen = STRLEN_CONST(type); - - line += STRLEN_CONST(rtsp_content_type); - linelen -= STRLEN_CONST(rtsp_content_type); - while (linelen > 0 && (*line == ' ' || *line == '\t')) { - line++; - linelen--; - } - - if (linelen < typelen || strncasecmp(type, line, typelen)) - return FALSE; - - line += typelen; - linelen -= typelen; - if (linelen > 0 && !isspace(*line)) - return FALSE; - - return TRUE; -} - -static const char rtsp_transport[] = "Transport:"; -static const char rtsp_sps[] = "server_port="; -static const char rtsp_cps[] = "client_port="; -static const char rtsp_rtp[] = "rtp/"; -static const char rtsp_inter[] = "interleaved="; - -static void -rtsp_create_conversation(packet_info *pinfo, const guchar *line_begin, - size_t line_len) -{ - conversation_t *conv; - guchar buf[256]; - guchar *tmp; - guint c_data_port, c_mon_port; - guint s_data_port, s_mon_port; - address null_addr; - - if (line_len > sizeof(buf) - 1) { - /* - * Don't overflow the buffer. - */ - line_len = sizeof(buf) - 1; - } - memcpy(buf, line_begin, line_len); - buf[line_len] = '\0'; - - tmp = buf + STRLEN_CONST(rtsp_transport); - while (*tmp && isspace(*tmp)) - tmp++; - if (strncasecmp(tmp, rtsp_rtp, strlen(rtsp_rtp)) != 0) { - g_warning("Frame %u: rtsp: unknown transport", pinfo->fd->num); - return; - } - - c_data_port = c_mon_port = 0; - s_data_port = s_mon_port = 0; - if ((tmp = strstr(buf, rtsp_sps))) { - tmp += strlen(rtsp_sps); - if (sscanf(tmp, "%u-%u", &s_data_port, &s_mon_port) < 1) { - g_warning("Frame %u: rtsp: bad server_port", - pinfo->fd->num); - return; - } - } - if ((tmp = strstr(buf, rtsp_cps))) { - tmp += strlen(rtsp_cps); - if (sscanf(tmp, "%u-%u", &c_data_port, &c_mon_port) < 1) { - g_warning("Frame %u: rtsp: bad client_port", - pinfo->fd->num); - return; - } - } - if (!c_data_port) { - rtsp_conversation_data_t *data; - guint s_data_chan, s_mon_chan; - int i; - - /* - * Deal with RTSP TCP-interleaved conversations. - */ - if ((tmp = strstr(buf, rtsp_inter)) == NULL) { - /* - * No interleaved or server_port - probably a - * SETUP request, rather than reply. - */ - return; - } - tmp += strlen(rtsp_inter); - i = sscanf(tmp, "%u-%u", &s_data_chan, &s_mon_chan); - if (i < 1) { - g_warning("Frame %u: rtsp: bad interleaved", - pinfo->fd->num); - return; - } - conv = find_conversation(&pinfo->src, &pinfo->dst, pinfo->ptype, - pinfo->srcport, pinfo->destport, 0); - if (!conv) { - conv = conversation_new(&pinfo->src, &pinfo->dst, - pinfo->ptype, pinfo->srcport, pinfo->destport, - 0); - } - data = conversation_get_proto_data(conv, proto_rtsp); - if (!data) { - data = g_mem_chunk_alloc(rtsp_vals); - conversation_add_proto_data(conv, proto_rtsp, data); - } - if (s_data_chan < RTSP_MAX_INTERLEAVED) { - data->interleaved[s_data_chan].dissector = - rtp_handle; - } - if (i > 1 && s_mon_chan < RTSP_MAX_INTERLEAVED) { - data->interleaved[s_mon_chan].dissector = - rtcp_handle; - } - return; - } - - /* - * We only want to match on the destination address, not the - * source address, because the server might send back a packet - * from an address other than the address to which its client - * sent the packet, so we construct a conversation with no - * second address. - */ - SET_ADDRESS(&null_addr, pinfo->src.type, 0, NULL); - - rtp_add_address(pinfo, (char *)pinfo->dst.data, c_data_port, s_data_port, - "RTSP", pinfo->fd->num); - - if (!c_mon_port) - return; - - rtcp_add_address(pinfo, (char *)pinfo->dst.data, c_mon_port, s_mon_port, - "RTSP", pinfo->fd->num); -} - -static const char rtsp_content_length[] = "Content-Length:"; - -static int -rtsp_get_content_length(const guchar *line_begin, size_t line_len) -{ - guchar buf[256]; - guchar *tmp; - long content_length; - char *p; - guchar *up; - - if (line_len > sizeof(buf) - 1) { - /* - * Don't overflow the buffer. - */ - line_len = sizeof(buf) - 1; - } - memcpy(buf, line_begin, line_len); - buf[line_len] = '\0'; - - tmp = buf + STRLEN_CONST(rtsp_content_length); - while (*tmp && isspace(*tmp)) - tmp++; - content_length = strtol(tmp, &p, 10); - up = p; - if (up == tmp || (*up != '\0' && !isspace(*up))) - return -1; /* not a valid number */ - return content_length; -} - -static const char rtsp_Session[] = "Session:"; -static const char rtsp_X_Vig_Msisdn[] = "X-Vig-Msisdn"; - -static int -dissect_rtspmessage(tvbuff_t *tvb, int offset, packet_info *pinfo, - proto_tree *tree) -{ - proto_tree *rtsp_tree = NULL; - proto_tree *sub_tree = NULL; - proto_item *ti = NULL; - const guchar *line; - gint next_offset; - const guchar *linep, *lineend; - int orig_offset; - int first_linelen, linelen; - int line_end_offset; - int colon_offset; - gboolean is_request_or_reply; - gboolean body_requires_content_len; - gboolean saw_req_resp_or_header; - guchar c; - rtsp_type_t rtsp_type; - gboolean is_header; - int is_sdp = FALSE; - int datalen; - int content_length; - int reported_datalen; - int value_offset; - int value_len; - e164_info_t e164_info; - /* - * Is this a request or response? - * - * Note that "tvb_find_line_end()" will return a value that - * is not longer than what's in the buffer, so the - * "tvb_get_ptr()" call won't throw an exception. - */ - first_linelen = tvb_find_line_end(tvb, offset, - tvb_ensure_length_remaining(tvb, offset), &next_offset, - FALSE); - /* - * Is the first line a request or response? - */ - line = tvb_get_ptr(tvb, offset, first_linelen); - is_request_or_reply = is_rtsp_request_or_reply(line, first_linelen, - &rtsp_type); - if (is_request_or_reply) { - /* - * Yes, it's a request or response. - * Do header desegmentation if we've been told to, - * and do body desegmentation if we've been told to and - * we find a Content-Length header. - */ - if (!req_resp_hdrs_do_reassembly(tvb, pinfo, - rtsp_desegment_headers, rtsp_desegment_body)) { - /* - * More data needed for desegmentation. - */ - return -1; - } - } - - /* - * RFC 2326 says that a content length must be specified - * in requests that have a body, although section 4.4 speaks - * of a server closing the connection indicating the end of - * a reply body. - * - * We assume that an absent content length in a request means - * that we don't have a body, and that an absent content length - * in a reply means that the reply body runs to the end of - * the connection. If the first line is neither, we assume - * that whatever follows a blank line should be treated as a - * body; there's not much else we can do, as we're jumping - * into the message in the middle. - * - * XXX - if there was no Content-Length entity header, we should - * accumulate all data until the end of the connection. - * That'd require that the TCP dissector call subdissectors - * for all frames with FIN, even if they contain no data, - * which would require subdissectors to deal intelligently - * with empty segments. - */ - if (rtsp_type == RTSP_REQUEST) - body_requires_content_len = TRUE; - else - body_requires_content_len = FALSE; - - if (check_col(pinfo->cinfo, COL_PROTOCOL)) - col_set_str(pinfo->cinfo, COL_PROTOCOL, "RTSP"); - if (check_col(pinfo->cinfo, COL_INFO)) { - /* - * Put the first line from the buffer into the summary - * if it's an RTSP request or reply (but leave out the - * line terminator). - * Otherwise, just call it a continuation. - * - * Note that "tvb_find_line_end()" will return a value that - * is not longer than what's in the buffer, so the - * "tvb_get_ptr()" call won't throw an exception. - */ - line = tvb_get_ptr(tvb, offset, first_linelen); - if (is_request_or_reply) - if ( rtsp_type == RTSP_REPLY ) { - col_add_str(pinfo->cinfo, COL_INFO, "Reply: "); - col_append_str(pinfo->cinfo, COL_INFO, - format_text(line, first_linelen)); - } - else - col_add_str(pinfo->cinfo, COL_INFO, - format_text(line, first_linelen)); - - else - col_set_str(pinfo->cinfo, COL_INFO, "Continuation"); - - } - - orig_offset = offset; - if (tree) { - ti = proto_tree_add_item(tree, proto_rtsp, tvb, offset, -1, - FALSE); - rtsp_tree = proto_item_add_subtree(ti, ett_rtsp); - } - - /* - * We haven't yet seen a Content-Length header. - */ - content_length = -1; - - /* - * Process the packet data, a line at a time. - */ - saw_req_resp_or_header = FALSE; /* haven't seen anything yet */ - while (tvb_reported_length_remaining(tvb, offset) != 0) { - /* - * We haven't yet concluded that this is a header. - */ - is_header = FALSE; - - /* - * Find the end of the line. - */ - linelen = tvb_find_line_end(tvb, offset, - tvb_ensure_length_remaining(tvb, offset), &next_offset, - FALSE); - if (linelen < 0) - return -1; - line_end_offset = offset + linelen; - /* - * colon_offset may be -1 - */ - colon_offset = tvb_find_guint8(tvb, offset, linelen, ':'); - - - /* - * Get a buffer that refers to the line. - */ - line = tvb_get_ptr(tvb, offset, linelen); - lineend = line + linelen; - - /* - * OK, does it look like an RTSP request or response? - */ - is_request_or_reply = is_rtsp_request_or_reply(line, linelen, - &rtsp_type); - if (is_request_or_reply) - goto is_rtsp; - - /* - * No. Does it look like a blank line (as would appear - * at the end of an RTSP request)? - */ - if (linelen == 0) - goto is_rtsp; /* Yes. */ - - /* - * No. Does it look like a header? - */ - linep = line; - while (linep < lineend) { - c = *linep++; - - /* - * This must be a CHAR to be part of a token; that - * means it must be ASCII. - */ - if (!isascii(c)) - break; /* not ASCII, thus not a CHAR */ - - /* - * This mustn't be a CTL to be part of a token. - * - * XXX - what about leading LWS on continuation - * lines of a header? - */ - if (iscntrl(c)) - break; /* CTL, not part of a header */ - - switch (c) { - - case '(': - case ')': - case '<': - case '>': - case '@': - case ',': - case ';': - case '\\': - case '"': - case '/': - case '[': - case ']': - case '?': - case '=': - case '{': - case '}': - /* - * It's a tspecial, so it's not - * part of a token, so it's not - * a field name for the beginning - * of a header. - */ - goto not_rtsp; - - case ':': - /* - * This ends the token; we consider - * this to be a header. - */ - is_header = TRUE; - goto is_rtsp; - - case ' ': - case '\t': - /* - * LWS (RFC-2616, 4.2); continue the previous - * header. - */ - goto is_rtsp; - } - } - - /* - * We haven't seen the colon, but everything else looks - * OK for a header line. - * - * If we've already seen an RTSP request or response - * line, or a header line, and we're at the end of - * the tvbuff, we assume this is an incomplete header - * line. (We quit this loop after seeing a blank line, - * so if we've seen a request or response line, or a - * header line, this is probably more of the request - * or response we're presumably seeing. There is some - * risk of false positives, but the same applies for - * full request or response lines or header lines, - * although that's less likely.) - * - * We throw an exception in that case, by checking for - * the existence of the next byte after the last one - * in the line. If it exists, "tvb_ensure_bytes_exist()" - * throws no exception, and we fall through to the - * "not RTSP" case. If it doesn't exist, - * "tvb_ensure_bytes_exist()" will throw the appropriate - * exception. - */ - if (saw_req_resp_or_header) - tvb_ensure_bytes_exist(tvb, offset, linelen + 1); - - not_rtsp: - /* - * We don't consider this part of an RTSP request or - * reply, so we don't display it. - */ - break; - - is_rtsp: - /* - * Process this line. - */ - if (linelen == 0) { - /* - * This is a blank line, which means that - * whatever follows it isn't part of this - * request or reply. - */ - proto_tree_add_text(rtsp_tree, tvb, offset, - next_offset - offset, "%s", - tvb_format_text(tvb, offset, next_offset - offset)); - offset = next_offset; - break; - } - - /* - * Not a blank line - either a request, a reply, or a header - * line. - */ - saw_req_resp_or_header = TRUE; - if (rtsp_tree) { - ti = proto_tree_add_text(rtsp_tree, tvb, offset, - next_offset - offset, "%s", - tvb_format_text(tvb, offset, next_offset - offset)); - - sub_tree = proto_item_add_subtree(ti, ett_rtsp_method); - - switch (rtsp_type) { - - case RTSP_REQUEST: - process_rtsp_request(tvb, offset, line, linelen, - sub_tree); - break; - - case RTSP_REPLY: - process_rtsp_reply(tvb, offset, line, linelen, - sub_tree); - break; - - case NOT_RTSP: - break; - } - } - if (is_header) { - /* - * Process some headers specially. - */ -#define HDR_MATCHES(header) \ - (linelen > STRLEN_CONST(header) && \ - strncasecmp(line, (header), STRLEN_CONST(header)) == 0) - - if (HDR_MATCHES(rtsp_transport)) { - /* - * Based on the port numbers specified - * in the Transport: header, set up - * a conversation that will be dissected - * with the appropriate dissector. - */ - rtsp_create_conversation(pinfo, line, linelen); - } else if (HDR_MATCHES(rtsp_content_type)) { - /* - * If the Content-Type: header says this - * is SDP, dissect the payload as SDP. - * - * XXX - we should just do the same - * sort of header processing - * that HTTP does, and use the - * "media_type" dissector table on - * the content type. - * - * We should use those for Transport: - * and Content-Length: as well (and - * should process Content-Length: in - * HTTP). - */ - if (is_content_sdp(line, linelen)) - is_sdp = TRUE; - } else if (HDR_MATCHES(rtsp_content_length)) { - /* - * Only the amount specified by the - * Content-Length: header should be treated - * as payload. - */ - content_length = rtsp_get_content_length(line, - linelen); - - } else if (HDR_MATCHES(rtsp_Session)) { - /* - * Extract the session string - - */ - - if ( colon_offset != -1 ){ - /* - * Skip whitespace after the colon. - * (Code from SIP dissector ) - */ - value_offset = colon_offset + 1; - while (value_offset < line_end_offset - && ((c = tvb_get_guint8(tvb, - value_offset)) == ' ' - || c == '\t')) - value_offset++; - /* - * Put the value into the protocol tree - */ - value_len = line_end_offset - value_offset; - proto_tree_add_string(sub_tree, hf_rtsp_session,tvb, - value_offset, value_len , - tvb_format_text(tvb, value_offset, value_len)); - } - - } else if (HDR_MATCHES(rtsp_X_Vig_Msisdn)) { - /* - * Extract the X_Vig_Msisdn string - */ - if ( colon_offset != -1 ){ - /* - * Skip whitespace after the colon. - * (Code from SIP dissector ) - */ - value_offset = colon_offset + 1; - while (value_offset < line_end_offset - && ((c = tvb_get_guint8(tvb, - value_offset)) == ' ' - || c == '\t')) - value_offset++; - /* - * Put the value into the protocol tree - */ - value_len = line_end_offset - value_offset; - proto_tree_add_string(sub_tree, hf_rtsp_X_Vig_Msisdn,tvb, - value_offset, value_len , - tvb_format_text(tvb, value_offset, value_len)); - - e164_info.e164_number_type = CALLING_PARTY_NUMBER; - e164_info.nature_of_address = 0; - - e164_info.E164_number_str = tvb_get_string(tvb, value_offset, - value_len); - e164_info.E164_number_length = value_len; - dissect_e164_number(tvb, sub_tree, value_offset, - value_len, e164_info); - g_free(e164_info.E164_number_str); - - - } - } - - } - offset = next_offset; - } - - /* - * If a content length was supplied, the amount of data to be - * processed as RTSP payload is the minimum of the content - * length and the amount of data remaining in the frame. - * - * If no content length was supplied (or if a bad content length - * was supplied), the amount of data to be processed is the amount - * of data remaining in the frame. - */ - datalen = tvb_length_remaining(tvb, offset); - reported_datalen = tvb_reported_length_remaining(tvb, offset); - if (content_length != -1) { - /* - * Content length specified; display only that amount - * as payload. - */ - if (datalen > content_length) - datalen = content_length; - - /* - * XXX - limit the reported length in the tvbuff we'll - * hand to a subdissector to be no greater than the - * content length. - * - * We really need both unreassembled and "how long it'd - * be if it were reassembled" lengths for tvbuffs, so - * that we throw the appropriate exceptions for - * "not enough data captured" (running past the length), - * "packet needed reassembly" (within the length but - * running past the unreassembled length), and - * "packet is malformed" (running past the reassembled - * length). - */ - if (reported_datalen > content_length) - reported_datalen = content_length; - } else { - /* - * No content length specified; if this message doesn't - * have a body if no content length is specified, process - * nothing as payload. - */ - if (body_requires_content_len) - datalen = 0; - } - - if (datalen > 0) { - /* - * There's stuff left over; process it. - */ - if (is_sdp) { - tvbuff_t *new_tvb; - - /* - * Fix up the top-level item so that it doesn't - * include the SDP stuff. - */ - if (ti != NULL) - proto_item_set_len(ti, offset); - - /* - * Now create a tvbuff for the SDP stuff and - * dissect it. - * - * The amount of data to be processed that's - * available in the tvbuff is "datalen", which - * is the minimum of the amount of data left in - * the tvbuff and any specified content length. - * - * The amount of data to be processed that's in - * this frame, regardless of whether it was - * captured or not, is "reported_datalen", - * which, if no content length was specified, - * is -1, i.e. "to the end of the frame. - */ - new_tvb = tvb_new_subset(tvb, offset, datalen, - reported_datalen); - call_dissector(sdp_handle, new_tvb, pinfo, tree); - } else { - if (tvb_get_guint8(tvb, offset) == RTSP_FRAMEHDR) { - /* - * This is interleaved stuff; don't - * treat it as raw data - set "datalen" - * to 0, so we won't skip the offset - * past it, which will cause our - * caller to process that stuff itself. - */ - datalen = 0; - } else { - proto_tree_add_text(rtsp_tree, tvb, offset, - datalen, "Data (%d bytes)", - reported_datalen); - } - } - - /* - * We've processed "datalen" bytes worth of data - * (which may be no data at all); advance the - * offset past whatever data we've processed. - */ - offset += datalen; - } - return offset - orig_offset; -} - -static void -process_rtsp_request(tvbuff_t *tvb, int offset, const guchar *data, - size_t linelen, proto_tree *tree) -{ - const guchar *lineend = data + linelen; - unsigned ii; - const guchar *url; - const guchar *url_start; - guchar *tmp_url; - - /* Request Methods */ - for (ii = 0; ii < RTSP_NMETHODS; ii++) { - size_t len = strlen(rtsp_methods[ii]); - if (linelen >= len && - strncasecmp(rtsp_methods[ii], data, len) == 0 && - (len == linelen || isspace(data[len]))) - break; - } - if (ii == RTSP_NMETHODS) { - /* - * We got here because "is_rtsp_request_or_reply()" returned - * RTSP_REQUEST, so we know one of the request methods - * matched, so we "can't get here". - */ - g_assert_not_reached(); - } - - /* Method name */ - proto_tree_add_string(tree, hf_rtsp_method, tvb, offset, - strlen(rtsp_methods[ii]), rtsp_methods[ii]); - - - /* URL */ - url = data; - while (url < lineend && !isspace(*url)) - url++; - while (url < lineend && isspace(*url)) - url++; - url_start = url; - while (url < lineend && !isspace(*url)) - url++; - tmp_url = g_malloc(url - url_start + 1); - memcpy(tmp_url, url_start, url - url_start); - tmp_url[url - url_start] = 0; - proto_tree_add_string(tree, hf_rtsp_url, tvb, - offset + (url_start - data), url - url_start, tmp_url); - g_free(tmp_url); -} - -static void -process_rtsp_reply(tvbuff_t *tvb, int offset, const guchar *data, - size_t linelen, proto_tree *tree) -{ - const guchar *lineend = data + linelen; - const guchar *status = data; - const guchar *status_start; - unsigned int status_i; - - /* status code */ - while (status < lineend && !isspace(*status)) - status++; - while (status < lineend && isspace(*status)) - status++; - status_start = status; - status_i = 0; - while (status < lineend && isdigit(*status)) - status_i = status_i * 10 + *status++ - '0'; - proto_tree_add_uint(tree, hf_rtsp_status, tvb, - offset + (status_start - data), - status - status_start, status_i); -} - -static void -dissect_rtsp(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) -{ - int offset = 0; - int len; - - while (tvb_reported_length_remaining(tvb, offset) != 0) { - len = (tvb_get_guint8(tvb, offset) == RTSP_FRAMEHDR) - ? dissect_rtspinterleaved(tvb, offset, pinfo, tree) - : dissect_rtspmessage(tvb, offset, pinfo, tree); - if (len == -1) - break; - offset += len; - - /* - * OK, we've set the Protocol and Info columns for the - * first RTSP message; make the columns non-writable, - * so that we don't change it for subsequent RTSP messages. - */ - col_set_writable(pinfo->cinfo, FALSE); - } -} - -static void -rtsp_init(void) -{ -/* Routine to initialize rtsp protocol before each capture or filter pass. */ -/* Release any memory if needed. Then setup the memory chunks. */ - - if (rtsp_vals) - g_mem_chunk_destroy(rtsp_vals); - - rtsp_vals = g_mem_chunk_new("rtsp_vals", - sizeof(rtsp_conversation_data_t), - rtsp_hash_init_count * sizeof(rtsp_conversation_data_t), - G_ALLOC_AND_FREE); -} - -void -proto_register_rtsp(void) -{ - static gint *ett[] = { - &ett_rtspframe, - &ett_rtsp, - &ett_rtsp_method, - }; - static hf_register_info hf[] = { - { &hf_rtsp_method, - { "Method", "rtsp.method", FT_STRING, BASE_NONE, NULL, 0, - "", HFILL }}, - { &hf_rtsp_url, - { "URL", "rtsp.url", FT_STRING, BASE_NONE, NULL, 0, - "", HFILL }}, - { &hf_rtsp_status, - { "Status", "rtsp.status", FT_UINT32, BASE_DEC, NULL, 0, - "", HFILL }}, - { &hf_rtsp_session, - { "Session", "rtsp.session", FT_STRING, BASE_NONE, NULL, 0, - "", HFILL }}, - { &hf_rtsp_X_Vig_Msisdn, - { "X-Vig-Msisdn", "X_Vig_Msisdn", FT_STRING, BASE_NONE, NULL, 0, - "", HFILL }}, - - - }; - module_t *rtsp_module; - - proto_rtsp = proto_register_protocol("Real Time Streaming Protocol", - "RTSP", "rtsp"); - proto_register_field_array(proto_rtsp, hf, array_length(hf)); - proto_register_subtree_array(ett, array_length(ett)); - - /* Register our configuration options, particularly our ports */ - - rtsp_module = prefs_register_protocol(proto_rtsp, proto_reg_handoff_rtsp); - prefs_register_uint_preference(rtsp_module, "tcp.port", - "RTSP TCP Port", - "Set the TCP port for RTSP messages", - 10, &global_rtsp_tcp_port); - prefs_register_uint_preference(rtsp_module, "tcp.alternate_port", - "Alternate RTSP TCP Port", - "Set the alternate TCP port for RTSP messages", - 10, &global_rtsp_tcp_alternate_port); - prefs_register_bool_preference(rtsp_module, "desegment_headers", - "Desegment all RTSP headers\nspanning multiple TCP segments", - "Whether the RTSP dissector should desegment all headers " - "of a request spanning multiple TCP segments", - &rtsp_desegment_headers); - prefs_register_bool_preference(rtsp_module, "desegment_body", - "Trust the \"Content-length:\" header and\ndesegment RTSP " - "bodies\nspanning multiple TCP segments", - "Whether the RTSP dissector should use the " - "\"Content-length:\" value to desegment the body " - "of a request spanning multiple TCP segments", - &rtsp_desegment_body); - - register_init_routine(rtsp_init); /* register re-init routine */ -} - -void -proto_reg_handoff_rtsp(void) -{ - dissector_handle_t rtsp_handle; - static int rtsp_prefs_initialized = FALSE; - - rtsp_handle = create_dissector_handle(dissect_rtsp, proto_rtsp); - - if (!rtsp_prefs_initialized) { - rtsp_prefs_initialized = TRUE; - } - else { - dissector_delete("tcp.port", tcp_port, rtsp_handle); - dissector_delete("tcp.port", tcp_alternate_port, rtsp_handle); - } - /* Set our port number for future use */ - - tcp_port = global_rtsp_tcp_port; - tcp_alternate_port = global_rtsp_tcp_alternate_port; - - dissector_add("tcp.port", tcp_port, rtsp_handle); - dissector_add("tcp.port", tcp_alternate_port, rtsp_handle); - - sdp_handle = find_dissector("sdp"); - rtp_handle = find_dissector("rtp"); - rtcp_handle = find_dissector("rtcp"); -} |