diff options
author | Gilbert Ramirez <gram@alumni.rice.edu> | 2004-07-18 18:06:47 +0000 |
---|---|---|
committer | Gilbert Ramirez <gram@alumni.rice.edu> | 2004-07-18 18:06:47 +0000 |
commit | 669db206cb1f270046ad400fff7655e20c63e723 (patch) | |
tree | 4eff24a2e16c8963e497e1fc575f35e6af59bd26 /epan/dissectors/packet-rtp.c | |
parent | ae46c27a38700af669ef907491081f09df6f6b2c (diff) |
Move dissectors to epan/dissectors directory.
Also move ncp222.py, x11-fields, process-x11-fields.pl,
make-reg-dotc, and make-reg-dotc.py.
Adjust #include lines in files that include packet-*.h
files.
svn path=/trunk/; revision=11410
Diffstat (limited to 'epan/dissectors/packet-rtp.c')
-rw-r--r-- | epan/dissectors/packet-rtp.c | 979 |
1 files changed, 979 insertions, 0 deletions
diff --git a/epan/dissectors/packet-rtp.c b/epan/dissectors/packet-rtp.c new file mode 100644 index 0000000000..900b01c2be --- /dev/null +++ b/epan/dissectors/packet-rtp.c @@ -0,0 +1,979 @@ +/* packet-rtp.c + * + * Routines for RTP dissection + * RTP = Real time Transport Protocol + * + * Copyright 2000, Philips Electronics N.V. + * Written by Andreas Sikkema <h323@ramdyne.nl> + * + * $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. + */ + +/* + * This dissector tries to dissect the RTP protocol according to Annex A + * of ITU-T Recommendation H.225.0 (02/98) or RFC 1889 + * + * RTP traffic is handled by an even UDP portnumber. This can be any + * port number, but there is a registered port available, port 5004 + * See Annex B of ITU-T Recommendation H.225.0, section B.7 + * + * This doesn't dissect older versions of RTP, such as: + * + * the vat protocol ("version 0") - see + * + * ftp://ftp.ee.lbl.gov/conferencing/vat/alpha-test/vatsrc-4.0b2.tar.gz + * + * and look in "session-vat.cc" if you want to write a dissector + * (have fun - there aren't any nice header files showing the packet + * format); + * + * version 1, as documented in + * + * ftp://gaia.cs.umass.edu/pub/hgschulz/rtp/draft-ietf-avt-rtp-04.txt + */ + + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include <glib.h> +#include <epan/packet.h> + +#include <stdio.h> +#include <string.h> + +#include "packet-rtp.h" +#include "rtp_pt.h" +#include <epan/conversation.h> +#include "tap.h" + +#include "prefs.h" + +static dissector_handle_t rtp_handle; + +static int rtp_tap = -1; + +static dissector_table_t rtp_pt_dissector_table; + +/* RTP header fields */ +static int proto_rtp = -1; +static int hf_rtp_version = -1; +static int hf_rtp_padding = -1; +static int hf_rtp_extension = -1; +static int hf_rtp_csrc_count = -1; +static int hf_rtp_marker = -1; +static int hf_rtp_payload_type = -1; +static int hf_rtp_seq_nr = -1; +static int hf_rtp_timestamp = -1; +static int hf_rtp_ssrc = -1; +static int hf_rtp_csrc_item = -1; +static int hf_rtp_data = -1; +static int hf_rtp_padding_data = -1; +static int hf_rtp_padding_count= -1; + +/* RTP header extension fields */ +static int hf_rtp_prof_define = -1; +static int hf_rtp_length = -1; +static int hf_rtp_hdr_ext = -1; + +/* RTP setup fields */ +static int hf_rtp_setup = -1; +static int hf_rtp_setup_frame = -1; +static int hf_rtp_setup_method = -1; + +/* RTP fields defining a sub tree */ +static gint ett_rtp = -1; +static gint ett_csrc_list = -1; +static gint ett_hdr_ext = -1; +static gint ett_rtp_setup = -1; + +static dissector_handle_t data_handle; + +static gboolean dissect_rtp_heur( tvbuff_t *tvb, packet_info *pinfo, + proto_tree *tree ); +static void dissect_rtp( tvbuff_t *tvb, packet_info *pinfo, + proto_tree *tree ); +static void show_setup_info(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree); + +/* Preferences bool to control whether or not setup info should be shown */ +static gboolean global_rtp_show_setup_info = TRUE; + +/* Try heuristic RTP decode */ +static gboolean global_rtp_heur = FALSE; + +/* Memory chunk for storing conversation and per-packet info */ +static GMemChunk *rtp_conversations = NULL; + +/* + * Fields in the first octet of the RTP header. + */ + +/* Version is the first 2 bits of the first octet*/ +#define RTP_VERSION(octet) ((octet) >> 6) + +/* Padding is the third bit; No need to shift, because true is any value + other than 0! */ +#define RTP_PADDING(octet) ((octet) & 0x20) + +/* Extension bit is the fourth bit */ +#define RTP_EXTENSION(octet) ((octet) & 0x10) + +/* CSRC count is the last four bits */ +#define RTP_CSRC_COUNT(octet) ((octet) & 0xF) + +static const value_string rtp_version_vals[] = +{ + { 0, "Old VAT Version" }, + { 1, "First Draft Version" }, + { 2, "RFC 1889 Version" }, + { 0, NULL }, +}; + +/* + * Fields in the second octet of the RTP header. + */ + +/* Marker is the first bit of the second octet */ +#define RTP_MARKER(octet) ((octet) & 0x80) + +/* Payload type is the last 7 bits */ +#define RTP_PAYLOAD_TYPE(octet) ((octet) & 0x7F) + +const value_string rtp_payload_type_vals[] = +{ + { PT_PCMU, "ITU-T G.711 PCMU" }, + { PT_1016, "USA Federal Standard FS-1016" }, + { PT_G721, "ITU-T G.721" }, + { PT_GSM, "GSM 06.10" }, + { PT_G723, "ITU-T G.723" }, + { PT_DVI4_8000, "DVI4 8000 samples/s" }, + { PT_DVI4_16000, "DVI4 16000 samples/s" }, + { PT_LPC, "Experimental linear predictive encoding from Xerox PARC" }, + { PT_PCMA, "ITU-T G.711 PCMA" }, + { PT_G722, "ITU-T G.722" }, + { PT_L16_STEREO, "16-bit uncompressed audio, stereo" }, + { PT_L16_MONO, "16-bit uncompressed audio, monaural" }, + { PT_QCELP, "Qualcomm Code Excited Linear Predictive coding" }, + { PT_CN, "Comfort noise" }, + { PT_MPA, "MPEG-I/II Audio"}, + { PT_G728, "ITU-T G.728" }, + { PT_DVI4_11025, "DVI4 11025 samples/s" }, + { PT_DVI4_22050, "DVI4 22050 samples/s" }, + { PT_G729, "ITU-T G.729" }, + { PT_CN_OLD, "Comfort noise (old)" }, + { PT_CELB, "Sun CellB video encoding" }, + { PT_JPEG, "JPEG-compressed video" }, + { PT_NV, "'nv' program" }, + { PT_H261, "ITU-T H.261" }, + { PT_MPV, "MPEG-I/II Video"}, + { PT_MP2T, "MPEG-II transport streams"}, + { PT_H263, "ITU-T H.263" }, + { 0, NULL }, +}; + +static address fake_addr; + +/* Set up an RTP conversation */ +void rtp_add_address(packet_info *pinfo, + const unsigned char* ip_addr, int port, + int other_port, + gchar *setup_method, guint32 setup_frame_number) +{ + address src_addr; + conversation_t* p_conv = NULL; + struct _rtp_conversation_info *p_conv_data = NULL; + + /* + * If this isn't the first time this packet has been processed, + * we've already done this work, so we don't need to do it + * again. + */ + if (pinfo->fd->flags.visited) + { + return; + } + + src_addr.type = pinfo->net_src.type; + src_addr.len = pinfo->net_src.len; + src_addr.data = ip_addr; + + /* + * Check if the ip address and port combination is not + * already registered + */ + p_conv = find_conversation( &src_addr, &src_addr, PT_UDP, port, other_port, + NO_ADDR_B | (!other_port ? NO_PORT_B : 0)); + + /* + * If not, add + */ + if ( ! p_conv ) { + /* Create conversation data */ + p_conv_data = g_mem_chunk_alloc(rtp_conversations); + + /* Check length first time we look at method string */ + strncpy(p_conv_data->method, setup_method, + (strlen(setup_method)+1 <= MAX_RTP_SETUP_METHOD_SIZE) ? + strlen(setup_method)+1 : + MAX_RTP_SETUP_METHOD_SIZE); + p_conv_data->method[MAX_RTP_SETUP_METHOD_SIZE] = '\0'; + p_conv_data->frame_number = setup_frame_number; + + /* Create conversation with this data */ + p_conv = conversation_new( &src_addr, &src_addr, PT_UDP, + (guint32)port, (guint32)other_port, + NO_ADDR2 | (!other_port ? NO_PORT2 : 0)); + conversation_add_proto_data(p_conv, proto_rtp, p_conv_data); + + /* Set dissector */ + conversation_set_dissector(p_conv, rtp_handle); + } + else + { + /* Update existing conversation data */ + p_conv_data = conversation_get_proto_data(p_conv, proto_rtp); + strcpy(p_conv_data->method, setup_method); + p_conv_data->frame_number = setup_frame_number; + } +} + +static void rtp_init( void ) +{ + unsigned char* tmp_data; + int i; + + /* (Re)allocate mem chunk for conversations */ + if (rtp_conversations) + { + g_mem_chunk_destroy(rtp_conversations); + } + rtp_conversations = g_mem_chunk_new("rtp_conversations", + sizeof(struct _rtp_conversation_info), + 20 * sizeof(struct _rtp_conversation_info), + G_ALLOC_ONLY); + + /* Create a fake adddress... */ + fake_addr.type = AT_IPv4; + fake_addr.len = 4; + + tmp_data = malloc( fake_addr.len ); + for ( i = 0; i < fake_addr.len; i++) { + tmp_data[i] = 0; + } + fake_addr.data = tmp_data; +} + +static gboolean +dissect_rtp_heur( tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree ) +{ + guint8 octet1, octet2; + unsigned int version; + unsigned int payload_type; + unsigned int offset = 0; + + /* This is a heuristic dissector, which means we get all the UDP + * traffic not sent to a known dissector and not claimed by + * a heuristic dissector called before us! + */ + + if (! global_rtp_heur) + return FALSE; + + /* Get the fields in the first octet */ + octet1 = tvb_get_guint8( tvb, offset ); + version = RTP_VERSION( octet1 ); + + if (version != 2) { + /* Unknown or unsupported version */ + return FALSE; + } + + /* Get the fields in the second octet */ + octet2 = tvb_get_guint8( tvb, offset + 1 ); + payload_type = RTP_PAYLOAD_TYPE( octet2 ); + /* if (payload_type == PT_PCMU || + * payload_type == PT_PCMA) + * payload_type == PT_G729) + * */ + if (payload_type <= PT_H263) { + dissect_rtp( tvb, pinfo, tree ); + return TRUE; + } + else { + return FALSE; + } +} + +static void +dissect_rtp_data( tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, + proto_tree *rtp_tree, int offset, unsigned int data_len, + unsigned int data_reported_len, unsigned int payload_type ) +{ + tvbuff_t *newtvb; + + newtvb = tvb_new_subset( tvb, offset, data_len, data_reported_len ); + if (!dissector_try_port(rtp_pt_dissector_table, payload_type, newtvb, + pinfo, tree)) + proto_tree_add_item( rtp_tree, hf_rtp_data, newtvb, 0, -1, FALSE ); +} + +static void +dissect_rtp( tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree ) +{ + proto_item *ti = NULL; + proto_tree *rtp_tree = NULL; + proto_tree *rtp_csrc_tree = NULL; + guint8 octet1, octet2; + unsigned int version; + gboolean padding_set; + gboolean extension_set; + unsigned int csrc_count; + gboolean marker_set; + unsigned int payload_type; + unsigned int i = 0; + unsigned int hdr_extension= 0; + unsigned int padding_count; + gint length, reported_length; + int data_len; + unsigned int offset = 0; + guint16 seq_num; + guint32 timestamp; + guint32 sync_src; + guint32 csrc_item; + + static struct _rtp_info rtp_info; + + /* Get the fields in the first octet */ + octet1 = tvb_get_guint8( tvb, offset ); + version = RTP_VERSION( octet1 ); + + if (version != 2) { + /* + * Unknown or unsupported version. + */ + if ( check_col( pinfo->cinfo, COL_PROTOCOL ) ) { + col_set_str( pinfo->cinfo, COL_PROTOCOL, "RTP" ); + } + + if ( check_col( pinfo->cinfo, COL_INFO) ) { + col_add_fstr( pinfo->cinfo, COL_INFO, + "Unknown RTP version %u", version); + } + + if ( tree ) { + ti = proto_tree_add_item( tree, proto_rtp, tvb, offset, -1, FALSE ); + rtp_tree = proto_item_add_subtree( ti, ett_rtp ); + + proto_tree_add_uint( rtp_tree, hf_rtp_version, tvb, + offset, 1, octet1); + } + return; + } + + padding_set = RTP_PADDING( octet1 ); + extension_set = RTP_EXTENSION( octet1 ); + csrc_count = RTP_CSRC_COUNT( octet1 ); + + /* Get the fields in the second octet */ + octet2 = tvb_get_guint8( tvb, offset + 1 ); + marker_set = RTP_MARKER( octet2 ); + payload_type = RTP_PAYLOAD_TYPE( octet2 ); + + /* Get the subsequent fields */ + seq_num = tvb_get_ntohs( tvb, offset + 2 ); + timestamp = tvb_get_ntohl( tvb, offset + 4 ); + sync_src = tvb_get_ntohl( tvb, offset + 8 ); + + /* fill in the rtp_info structure */ + rtp_info.info_padding_set = padding_set; + rtp_info.info_padding_count = 0; + rtp_info.info_marker_set = marker_set; + rtp_info.info_payload_type = payload_type; + rtp_info.info_seq_num = seq_num; + rtp_info.info_timestamp = timestamp; + rtp_info.info_sync_src = sync_src; + + /* + * Do we have all the data? + */ + length = tvb_length_remaining(tvb, offset); + reported_length = tvb_reported_length_remaining(tvb, offset); + if (reported_length >= 0 && length >= reported_length) { + /* + * Yes. + */ + rtp_info.info_all_data_present = TRUE; + rtp_info.info_data_len = reported_length; + + /* + * Save the pointer to raw rtp data (header + payload incl. + * padding). + * That should be safe because the "epan_dissect_t" + * constructed for the packet has not yet been freed when + * the taps are called. + * (Destroying the "epan_dissect_t" will end up freeing + * all the tvbuffs and hence invalidating pointers to + * their data.) + * See "add_packet_to_packet_list()" for details. + */ + rtp_info.info_data = tvb_get_ptr(tvb, 0, -1); + } else { + /* + * No - packet was cut short at capture time. + */ + rtp_info.info_all_data_present = FALSE; + rtp_info.info_data_len = 0; + rtp_info.info_data = NULL; + } + + if ( check_col( pinfo->cinfo, COL_PROTOCOL ) ) { + col_set_str( pinfo->cinfo, COL_PROTOCOL, "RTP" ); + } + + if ( check_col( pinfo->cinfo, COL_INFO) ) { + col_add_fstr( pinfo->cinfo, COL_INFO, + "Payload type=%s, SSRC=%u, Seq=%u, Time=%u%s", + val_to_str( payload_type, rtp_payload_type_vals, + "Unknown (%u)" ), + sync_src, + seq_num, + timestamp, + marker_set ? ", Mark" : ""); + } + if ( tree ) { + /* Create RTP protocol tree */ + ti = proto_tree_add_item(tree, proto_rtp, tvb, offset, -1, FALSE ); + rtp_tree = proto_item_add_subtree(ti, ett_rtp ); + + /* Conversation setup info */ + if (global_rtp_show_setup_info) + { + show_setup_info(tvb, pinfo, rtp_tree); + } + + proto_tree_add_uint( rtp_tree, hf_rtp_version, tvb, + offset, 1, octet1 ); + proto_tree_add_boolean( rtp_tree, hf_rtp_padding, tvb, + offset, 1, octet1 ); + proto_tree_add_boolean( rtp_tree, hf_rtp_extension, tvb, + offset, 1, octet1 ); + proto_tree_add_uint( rtp_tree, hf_rtp_csrc_count, tvb, + offset, 1, octet1 ); + offset++; + + proto_tree_add_boolean( rtp_tree, hf_rtp_marker, tvb, offset, + 1, octet2 ); + proto_tree_add_uint( rtp_tree, hf_rtp_payload_type, tvb, + offset, 1, octet2 ); + offset++; + + /* Sequence number 16 bits (2 octets) */ + proto_tree_add_uint( rtp_tree, hf_rtp_seq_nr, tvb, offset, 2, seq_num ); + offset += 2; + + /* Timestamp 32 bits (4 octets) */ + proto_tree_add_uint( rtp_tree, hf_rtp_timestamp, tvb, offset, 4, timestamp ); + offset += 4; + + /* Synchronization source identifier 32 bits (4 octets) */ + proto_tree_add_uint( rtp_tree, hf_rtp_ssrc, tvb, offset, 4, sync_src ); + offset += 4; + } else { + offset += 12; + } + /* CSRC list*/ + if ( csrc_count > 0 ) { + if ( tree ) { + ti = proto_tree_add_text(rtp_tree, tvb, offset, csrc_count * 4, "Contributing Source identifiers"); + rtp_csrc_tree = proto_item_add_subtree( ti, ett_csrc_list ); + } + for (i = 0; i < csrc_count; i++ ) { + csrc_item = tvb_get_ntohl( tvb, offset ); + if ( tree ) proto_tree_add_uint_format( rtp_csrc_tree, + hf_rtp_csrc_item, tvb, offset, 4, + csrc_item, + "CSRC item %d: %u", + i, csrc_item ); + offset += 4; + } + } + + /* Optional RTP header extension */ + if ( extension_set ) { + /* Defined by profile field is 16 bits (2 octets) */ + if ( tree ) proto_tree_add_uint( rtp_tree, hf_rtp_prof_define, tvb, offset, 2, tvb_get_ntohs( tvb, offset ) ); + offset += 2; + + hdr_extension = tvb_get_ntohs( tvb, offset ); + if ( tree ) proto_tree_add_uint( rtp_tree, hf_rtp_length, tvb, + offset, 2, hdr_extension); + offset += 2; + if ( hdr_extension > 0 ) { + if ( tree ) { + ti = proto_tree_add_text(rtp_tree, tvb, offset, csrc_count * 4, "Header extensions"); + /* I'm re-using the old tree variable here + from the CSRC list!*/ + rtp_csrc_tree = proto_item_add_subtree( ti, + ett_hdr_ext ); + } + for (i = 0; i < hdr_extension; i++ ) { + if ( tree ) proto_tree_add_uint( rtp_csrc_tree, hf_rtp_hdr_ext, tvb, offset, 4, tvb_get_ntohl( tvb, offset ) ); + offset += 4; + } + } + } + + if ( padding_set ) { + /* + * This RTP frame has padding - find it. + * + * The padding count is found in the LAST octet of + * the packet; it contains the number of octets + * that can be ignored at the end of the packet. + */ + if (tvb_length(tvb) < tvb_reported_length(tvb)) { + /* + * We don't *have* the last octet of the + * packet, so we can't get the padding + * count. + * + * Put an indication of that into the + * tree, and just put in a raw data + * item. + */ + if ( tree ) proto_tree_add_text(rtp_tree, tvb, 0, 0, + "Frame has padding, but not all the frame data was captured"); + call_dissector(data_handle, + tvb_new_subset(tvb, offset, -1, -1), + pinfo, rtp_tree); + return; + } + + padding_count = tvb_get_guint8( tvb, + tvb_reported_length( tvb ) - 1 ); + data_len = + tvb_reported_length_remaining( tvb, offset ) - padding_count; + + rtp_info.info_payload_offset = offset; + rtp_info.info_payload_len = tvb_length_remaining(tvb, offset); + rtp_info.info_padding_count = padding_count; + + if (data_len > 0) { + /* + * There's data left over when you take out + * the padding; dissect it. + */ + dissect_rtp_data( tvb, pinfo, tree, rtp_tree, + offset, + data_len, + data_len, + payload_type ); + offset += data_len; + } else if (data_len < 0) { + /* + * The padding count is bigger than the + * amount of RTP payload in the packet! + * Clip the padding count. + * + * XXX - put an item in the tree to indicate + * that the padding count is bogus? + */ + padding_count = + tvb_reported_length_remaining(tvb, offset); + } + if (padding_count > 1) { + /* + * There's more than one byte of padding; + * show all but the last byte as padding + * data. + */ + if ( tree ) proto_tree_add_item( rtp_tree, hf_rtp_padding_data, + tvb, offset, padding_count - 1, FALSE ); + offset += padding_count - 1; + } + /* + * Show the last byte in the PDU as the padding + * count. + */ + if ( tree ) proto_tree_add_item( rtp_tree, hf_rtp_padding_count, + tvb, offset, 1, FALSE ); + } + else { + /* + * No padding. + */ + dissect_rtp_data( tvb, pinfo, tree, rtp_tree, offset, + tvb_length_remaining( tvb, offset ), + tvb_reported_length_remaining( tvb, offset ), + payload_type ); + rtp_info.info_payload_offset = offset; + rtp_info.info_payload_len = tvb_length_remaining(tvb, offset); + } + if (!pinfo->in_error_pkt) + tap_queue_packet(rtp_tap, pinfo, &rtp_info); +} + + +/* Look for conversation info and display any setup info found */ +void show_setup_info(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree) +{ + /* Conversation and current data */ + conversation_t *p_conv = NULL; + struct _rtp_conversation_info *p_conv_data = NULL; + + /* Use existing packet info if available */ + p_conv_data = p_get_proto_data(pinfo->fd, proto_rtp); + + if (!p_conv_data) + { + /* First time, get info from conversation */ + p_conv = find_conversation(&pinfo->net_dst, &pinfo->net_src, + pinfo->ptype, + pinfo->destport, pinfo->srcport, NO_ADDR_B); + if (p_conv) + { + /* Create space for packet info */ + struct _rtp_conversation_info *p_conv_packet_data; + p_conv_data = conversation_get_proto_data(p_conv, proto_rtp); + + /* Save this conversation info into packet info */ + p_conv_packet_data = g_mem_chunk_alloc(rtp_conversations); + strcpy(p_conv_packet_data->method, p_conv_data->method); + p_conv_packet_data->frame_number = p_conv_data->frame_number; + p_add_proto_data(pinfo->fd, proto_rtp, p_conv_packet_data); + } + } + + /* Create setup info subtree with summary info. */ + if (p_conv_data) + { + proto_tree *rtp_setup_tree; + proto_item *ti = proto_tree_add_string_format(tree, hf_rtp_setup, tvb, 0, 0, + "", + "Stream setup by %s (frame %d)", + p_conv_data->method, + p_conv_data->frame_number); + PROTO_ITEM_SET_GENERATED(ti); + rtp_setup_tree = proto_item_add_subtree(ti, ett_rtp_setup); + if (rtp_setup_tree) + { + /* Add details into subtree */ + proto_item* item = proto_tree_add_uint(rtp_setup_tree, hf_rtp_setup_frame, + tvb, 0, 0, p_conv_data->frame_number); + PROTO_ITEM_SET_GENERATED(item); + item = proto_tree_add_string(rtp_setup_tree, hf_rtp_setup_method, + tvb, 0, 0, p_conv_data->method); + PROTO_ITEM_SET_GENERATED(item); + } + } +} + + +void +proto_register_rtp(void) +{ + static hf_register_info hf[] = + { + { + &hf_rtp_version, + { + "Version", + "rtp.version", + FT_UINT8, + BASE_DEC, + VALS(rtp_version_vals), + 0xC0, + "", HFILL + } + }, + { + &hf_rtp_padding, + { + "Padding", + "rtp.padding", + FT_BOOLEAN, + 8, + NULL, + 0x20, + "", HFILL + } + }, + { + &hf_rtp_extension, + { + "Extension", + "rtp.ext", + FT_BOOLEAN, + 8, + NULL, + 0x10, + "", HFILL + } + }, + { + &hf_rtp_csrc_count, + { + "Contributing source identifiers count", + "rtp.cc", + FT_UINT8, + BASE_DEC, + NULL, + 0x0F, + "", HFILL + } + }, + { + &hf_rtp_marker, + { + "Marker", + "rtp.marker", + FT_BOOLEAN, + 8, + NULL, + 0x80, + "", HFILL + } + }, + { + &hf_rtp_payload_type, + { + "Payload type", + "rtp.p_type", + FT_UINT8, + BASE_DEC, + VALS(rtp_payload_type_vals), + 0x7F, + "", HFILL + } + }, + { + &hf_rtp_seq_nr, + { + "Sequence number", + "rtp.seq", + FT_UINT16, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_timestamp, + { + "Timestamp", + "rtp.timestamp", + FT_UINT32, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_ssrc, + { + "Synchronization Source identifier", + "rtp.ssrc", + FT_UINT32, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_prof_define, + { + "Defined by profile", + "rtp.ext.profile", + FT_UINT16, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_length, + { + "Extension length", + "rtp.ext.len", + FT_UINT16, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_csrc_item, + { + "CSRC item", + "rtp.csrc.item", + FT_UINT32, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_hdr_ext, + { + "Header extension", + "rtp.hdr_ext", + FT_UINT32, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_data, + { + "Payload", + "rtp.payload", + FT_BYTES, + BASE_HEX, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_padding_data, + { + "Padding data", + "rtp.padding.data", + FT_BYTES, + BASE_HEX, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_padding_count, + { + "Padding count", + "rtp.padding.count", + FT_UINT8, + BASE_DEC, + NULL, + 0x0, + "", HFILL + } + }, + { + &hf_rtp_setup, + { + "Stream setup", + "rtp.setup", + FT_STRING, + BASE_NONE, + NULL, + 0x0, + "Stream setup, method and frame number", HFILL + } + }, + { + &hf_rtp_setup_frame, + { + "Setup frame", + "rtp.setup-frame", + FT_FRAMENUM, + BASE_NONE, + NULL, + 0x0, + "Frame that set up this stream", HFILL + } + }, + { + &hf_rtp_setup_method, + { + "Setup Method", + "rtp.setup-method", + FT_STRING, + BASE_NONE, + NULL, + 0x0, + "Method used to set up this stream", HFILL + } + } + + }; + + static gint *ett[] = + { + &ett_rtp, + &ett_csrc_list, + &ett_hdr_ext, + &ett_rtp_setup + }; + + module_t *rtp_module; + + + proto_rtp = proto_register_protocol("Real-Time Transport Protocol", + "RTP", "rtp"); + proto_register_field_array(proto_rtp, hf, array_length(hf)); + proto_register_subtree_array(ett, array_length(ett)); + + register_dissector("rtp", dissect_rtp, proto_rtp); + rtp_tap = register_tap("rtp"); + + rtp_pt_dissector_table = register_dissector_table("rtp.pt", + "RTP payload type", FT_UINT8, BASE_DEC); + + rtp_module = prefs_register_protocol(proto_rtp, NULL); + + prefs_register_bool_preference(rtp_module, "show_setup_info", + "Show stream setup information", + "Where available, show which protocol and frame caused " + "this RTP stream to be created", + &global_rtp_show_setup_info); + + prefs_register_bool_preference(rtp_module, "heuristic_rtp", + "Try to decode RTP outside of conversations ", + "If call control SIP/H323/RTSP/.. messages are missing in the trace, " + "RTP isn't decoded without this", + &global_rtp_heur); + + register_init_routine( &rtp_init ); +} + +void +proto_reg_handoff_rtp(void) +{ + data_handle = find_dissector("data"); + + /* + * Register this dissector as one that can be selected by a + * UDP port number. + */ + rtp_handle = find_dissector("rtp"); + dissector_add_handle("udp.port", rtp_handle); + + heur_dissector_add( "udp", dissect_rtp_heur, proto_rtp); +} |