2003-09-11 13:36:04 +00:00
/* packet-t38.c
* Routines for T .38 packet dissection
* 2003 Hans Viens
2004-10-22 06:19:00 +00:00
* 2004 Alejandro Vaquero , add support Conversations for SDP
2006-01-15 15:01:14 +00:00
* 2006 Alejandro Vaquero , add T30 reassemble and dissection
2003-09-11 13:36:04 +00:00
*
2004-07-18 00:24:25 +00:00
* $ Id $
2003-09-11 13:36:04 +00:00
*
2006-05-21 04:49:01 +00:00
* Wireshark - Network traffic analyzer
* By Gerald Combs < gerald @ wireshark . org >
2003-09-11 13:36:04 +00:00
* 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 .
*/
2004-01-26 22:16:43 +00:00
/* Depending on what ASN.1 specification is used you may have to change
* the preference setting regarding Pre - Corrigendum ASN .1 specification :
* http : //www.itu.int/ITU-T/asn1/database/itu-t/t/t38/1998/T38.html (Pre-Corrigendum=TRUE)
* http : //www.itu.int/ITU-T/asn1/database/itu-t/t/t38/2003/T38(1998).html (Pre-Corrigendum=TRUE)
*
* http : //www.itu.int/ITU-T/asn1/database/itu-t/t/t38/2003/T38(2002).html (Pre-Corrigendum=FALSE)
* http : //www.itu.int/ITU-T/asn1/database/itu-t/t/t38/2002/t38.html (Pre-Corrigendum=FALSE)
* http : //www.itu.int/ITU-T/asn1/database/itu-t/t/t38/2002-Amd1/T38.html (Pre-Corrigendum=FALSE)
*/
/* TO DO:
* - TCP desegmentation is currently not supported for T .38 IFP directly over TCP .
2004-10-22 06:19:00 +00:00
* - H .245 dissectors should be updated to start conversations for T .38 similar to RTP .
2004-01-09 23:24:54 +00:00
* - Sometimes the last octet is not high - lighted when selecting something in the tree . Bug in PER dissector ?
2004-01-26 22:16:43 +00:00
* - Add support for RTP payload audio / t38 ( draft - jones - avt - audio - t38 - 03. txt ) , i . e . T38 in RTP packets .
2004-01-09 23:24:54 +00:00
*/
2003-09-11 13:36:04 +00:00
# ifdef HAVE_CONFIG_H
# include "config.h"
# endif
# include <glib.h>
# include <epan/packet.h>
2006-01-15 15:01:14 +00:00
# include <epan/reassemble.h>
2003-09-11 13:36:04 +00:00
# include <epan/conversation.h>
2005-10-03 05:55:26 +00:00
# include <epan/tap.h>
2006-01-15 15:01:14 +00:00
# include <epan/expert.h>
2003-09-11 13:36:04 +00:00
# include <stdio.h>
# include <string.h>
2004-10-22 06:19:00 +00:00
# include "packet-t38.h"
2004-09-27 22:55:15 +00:00
# include <epan/prefs.h>
2004-09-29 00:52:45 +00:00
# include <epan/ipproto.h>
2003-09-11 13:36:04 +00:00
# include "packet-per.h"
2004-09-27 22:55:15 +00:00
# include <epan/prefs.h>
2004-01-26 22:16:43 +00:00
# include "packet-tpkt.h"
2005-08-12 22:12:32 +00:00
# include <epan/emem.h>
2003-09-11 13:36:04 +00:00
2004-01-09 23:24:54 +00:00
# define PORT_T38 6004
static guint global_t38_tcp_port = PORT_T38 ;
static guint global_t38_udp_port = PORT_T38 ;
2005-10-03 05:55:26 +00:00
static int t38_tap = - 1 ;
2004-01-09 23:24:54 +00:00
/*
* Variables to allow for proper deletion of dissector registration when
* the user changes port from the gui .
*/
static guint tcp_port = 0 ;
static guint udp_port = 0 ;
/* dissect using the Pre Corrigendum T.38 ASN.1 specification (1998) */
static gboolean use_pre_corrigendum_asn1_specification = TRUE ;
/* dissect packets that looks like RTP version 2 packets as RTP */
/* instead of as T.38. This may result in that some T.38 UPTL */
/* packets with sequence number values higher than 32767 may be */
/* shown as RTP packets. */
static gboolean dissect_possible_rtpv2_packets_as_rtp = FALSE ;
2003-09-11 13:36:04 +00:00
2004-01-26 22:16:43 +00:00
/* Reassembly of T.38 PDUs over TPKT over TCP */
static gboolean t38_tpkt_reassembly = TRUE ;
/* Preference setting whether TPKT header is used when sending T.38 over TCP.
* The default setting is Maybe where the dissector will look on the first
* bytes to try to determine whether TPKT header is used or not . This may not
* work so well in some cases . You may want to change the setting to Always or
* Newer .
*/
# define T38_TPKT_NEVER 0 /* Assume that there is never a TPKT header */
# define T38_TPKT_ALWAYS 1 /* Assume that there is always a TPKT header */
# define T38_TPKT_MAYBE 2 /* Assume TPKT if first octets are 03-00-xx-xx */
static gint t38_tpkt_usage = T38_TPKT_MAYBE ;
static const enum_val_t t38_tpkt_options [ ] = {
2004-05-24 02:25:21 +00:00
{ " never " , " Never " , T38_TPKT_NEVER } ,
{ " always " , " Always " , T38_TPKT_ALWAYS } ,
{ " maybe " , " Maybe " , T38_TPKT_MAYBE } ,
{ NULL , NULL , - 1 }
2004-01-26 22:16:43 +00:00
} ;
2006-01-15 15:01:14 +00:00
/* T30 */
static int proto_t30 = - 1 ;
static int hf_t30_Address = - 1 ;
static int hf_t30_Control = - 1 ;
static int hf_t30_Facsimile_Control = - 1 ;
static int hf_t30_fif_sm = - 1 ;
static int hf_t30_fif_rtif = - 1 ;
static int hf_t30_fif_3gmn = - 1 ;
static int hf_t30_fif_v8c = - 1 ;
static int hf_t30_fif_op = - 1 ;
static int hf_t30_fif_rtfc = - 1 ;
static int hf_t30_fif_rfo = - 1 ;
static int hf_t30_fif_dsr = - 1 ;
static int hf_t30_fif_dsr_dcs = - 1 ;
static int hf_t30_fif_res = - 1 ;
static int hf_t30_fif_tdcc = - 1 ;
static int hf_t30_fif_rwc = - 1 ;
static int hf_t30_fif_rw_dcs = - 1 ;
static int hf_t30_fif_rlc = - 1 ;
static int hf_t30_fif_rl_dcs = - 1 ;
static int hf_t30_fif_msltcr = - 1 ;
static int hf_t30_fif_mslt_dcs = - 1 ;
static int hf_t30_fif_ext = - 1 ;
static int hf_t30_fif_cm = - 1 ;
static int hf_t30_fif_ecm = - 1 ;
static int hf_t30_fif_fs_dcs = - 1 ;
static int hf_t30_fif_t6 = - 1 ;
static int hf_t30_fif_fvc = - 1 ;
static int hf_t30_fif_mspc = - 1 ;
static int hf_t30_fif_ps = - 1 ;
static int hf_t30_fif_t43 = - 1 ;
static int hf_t30_fif_pi = - 1 ;
static int hf_t30_fif_vc32k = - 1 ;
static int hf_t30_fif_r8x15 = - 1 ;
static int hf_t30_fif_300x300 = - 1 ;
static int hf_t30_fif_r16x15 = - 1 ;
static int hf_t30_fif_ibrp = - 1 ;
static int hf_t30_fif_mbrp = - 1 ;
static int hf_t30_fif_msltchr = - 1 ;
static int hf_t30_fif_rts = - 1 ;
static int hf_t30_fif_sp = - 1 ;
static int hf_t30_fif_sc = - 1 ;
static int hf_t30_fif_passw = - 1 ;
static int hf_t30_fif_sit = - 1 ;
static int hf_t30_fif_rttd = - 1 ;
static int hf_t30_fif_bft = - 1 ;
static int hf_t30_fif_dtm = - 1 ;
static int hf_t30_fif_edi = - 1 ;
static int hf_t30_fif_btm = - 1 ;
static int hf_t30_fif_rttcmmd = - 1 ;
static int hf_t30_fif_chrm = - 1 ;
static int hf_t30_fif_mm = - 1 ;
static int hf_t30_fif_pm26 = - 1 ;
static int hf_t30_fif_dnc = - 1 ;
static int hf_t30_fif_do = - 1 ;
static int hf_t30_fif_jpeg = - 1 ;
static int hf_t30_fif_fcm = - 1 ;
static int hf_t30_fif_pht = - 1 ;
static int hf_t30_fif_12c = - 1 ;
static int hf_t30_fif_ns = - 1 ;
static int hf_t30_fif_ci = - 1 ;
static int hf_t30_fif_cgr = - 1 ;
static int hf_t30_fif_nalet = - 1 ;
static int hf_t30_fif_naleg = - 1 ;
static int hf_t30_fif_spscb = - 1 ;
static int hf_t30_fif_spsco = - 1 ;
static int hf_t30_fif_hkm = - 1 ;
static int hf_t30_fif_rsa = - 1 ;
static int hf_t30_fif_oc = - 1 ;
static int hf_t30_fif_hfx40 = - 1 ;
static int hf_t30_fif_acn2c = - 1 ;
static int hf_t30_fif_acn3c = - 1 ;
static int hf_t30_fif_hfx40i = - 1 ;
static int hf_t30_fif_ahsn2 = - 1 ;
static int hf_t30_fif_ahsn3 = - 1 ;
static int hf_t30_fif_t441 = - 1 ;
static int hf_t30_fif_t442 = - 1 ;
static int hf_t30_fif_t443 = - 1 ;
static int hf_t30_fif_plmss = - 1 ;
static int hf_t30_fif_cg300 = - 1 ;
static int hf_t30_fif_100x100cg = - 1 ;
static int hf_t30_fif_spcbft = - 1 ;
static int hf_t30_fif_ebft = - 1 ;
static int hf_t30_fif_isp = - 1 ;
static int hf_t30_fif_ira = - 1 ;
static int hf_t30_fif_600x600 = - 1 ;
static int hf_t30_fif_1200x1200 = - 1 ;
static int hf_t30_fif_300x600 = - 1 ;
static int hf_t30_fif_400x800 = - 1 ;
static int hf_t30_fif_600x1200 = - 1 ;
static int hf_t30_fif_cg600x600 = - 1 ;
static int hf_t30_fif_cg1200x1200 = - 1 ;
static int hf_t30_fif_dspcam = - 1 ;
static int hf_t30_fif_dspccm = - 1 ;
static int hf_t30_fif_bwmrcp = - 1 ;
static int hf_t30_fif_t45 = - 1 ;
static int hf_t30_fif_sdmc = - 1 ;
static int hf_t30_fif_number = - 1 ;
static int hf_t30_fif_country_code = - 1 ;
static int hf_t30_fif_non_stand_bytes = - 1 ;
static int hf_t30_t4_frame_num = - 1 ;
static int hf_t30_t4_data = - 1 ;
static int hf_t30_partial_page_fcf2 = - 1 ;
static int hf_t30_partial_page_i1 = - 1 ;
static int hf_t30_partial_page_i2 = - 1 ;
static int hf_t30_partial_page_i3 = - 1 ;
static gint ett_t30 = - 1 ;
static gint ett_t30_fif = - 1 ;
/* T38 */
2004-01-26 22:52:22 +00:00
static dissector_handle_t t38_udp_handle ;
static dissector_handle_t t38_tcp_handle ;
2004-01-26 22:16:43 +00:00
static dissector_handle_t t38_tcp_pdu_handle ;
2004-01-09 23:24:54 +00:00
static dissector_handle_t rtp_handle ;
2003-09-11 13:36:04 +00:00
static guint32 Type_of_msg_value ;
static guint32 Data_Field_field_type_value ;
2004-01-09 23:24:54 +00:00
static guint32 Data_value ;
static guint32 T30ind_value ;
2006-01-15 15:01:14 +00:00
static guint32 Data_Field_item_num ;
2003-09-11 13:36:04 +00:00
static int proto_t38 = - 1 ;
2006-06-29 15:26:41 +00:00
static int hf_t38_null = - 1 ;
static int hf_t38_dummy = - 1 ;
2003-09-11 13:36:04 +00:00
static int hf_t38_IFPPacket = - 1 ;
static int hf_t38_Type_of_msg = - 1 ;
static int hf_t38_t30_indicator = - 1 ;
static int hf_t38_data = - 1 ;
static int hf_t38_Data_Field = - 1 ;
static int hf_t38_Data_Field_item = - 1 ;
static int hf_t38_Data_Field_field_type = - 1 ;
static int hf_t38_Data_Field_field_data = - 1 ;
static int hf_t38_UDPTLPacket = - 1 ;
static int hf_t38_seq_number = - 1 ;
static int hf_t38_primary_ifp_packet = - 1 ;
static int hf_t38_primary_ifp_packet_length = - 1 ;
static int hf_t38_error_recovery = - 1 ;
static int hf_t38_secondary_ifp_packets = - 1 ;
static int hf_t38_secondary_ifp_packets_item = - 1 ;
static int hf_t38_secondary_ifp_packets_item_length = - 1 ;
static int hf_t38_fec_info = - 1 ;
static int hf_t38_fec_npackets = - 1 ;
static int hf_t38_fec_data = - 1 ;
static int hf_t38_fec_data_item = - 1 ;
2004-10-22 06:19:00 +00:00
/* T38 setup fields */
static int hf_t38_setup = - 1 ;
static int hf_t38_setup_frame = - 1 ;
static int hf_t38_setup_method = - 1 ;
2006-01-15 15:01:14 +00:00
/* T38 Data reassemble fields */
static int hf_data_fragments = - 1 ;
static int hf_data_fragment = - 1 ;
static int hf_data_fragment_overlap = - 1 ;
static int hf_data_fragment_overlap_conflicts = - 1 ;
static int hf_data_fragment_multiple_tails = - 1 ;
static int hf_data_fragment_too_long_fragment = - 1 ;
static int hf_data_fragment_error = - 1 ;
static int hf_data_reassembled_in = - 1 ;
2003-09-11 13:36:04 +00:00
static gint ett_t38 = - 1 ;
static gint ett_t38_IFPPacket = - 1 ;
static gint ett_t38_Type_of_msg = - 1 ;
static gint ett_t38_t30_indicator = - 1 ;
static gint ett_t38_data = - 1 ;
static gint ett_t38_Data_Field = - 1 ;
static gint ett_t38_Data_Field_item = - 1 ;
static gint ett_t38_Data_Field_field_type = - 1 ;
static gint ett_t38_UDPTLPacket = - 1 ;
static gint ett_t38_error_recovery = - 1 ;
static gint ett_t38_secondary_ifp_packets = - 1 ;
static gint ett_t38_fec_info = - 1 ;
static gint ett_t38_fec_data = - 1 ;
2004-10-22 06:19:00 +00:00
static gint ett_t38_setup = - 1 ;
2003-09-11 13:36:04 +00:00
2006-01-15 15:01:14 +00:00
static gint ett_data_fragment = - 1 ;
static gint ett_data_fragments = - 1 ;
2004-01-09 23:24:54 +00:00
static gboolean primary_part = TRUE ;
static guint32 seq_number = 0 ;
2006-01-15 15:01:14 +00:00
/* Tables for reassembly of Data fragments. */
static GHashTable * data_fragment_table = NULL ;
static GHashTable * data_reassembled_table = NULL ;
static const fragment_items data_frag_items = {
/* Fragment subtrees */
& ett_data_fragment ,
& ett_data_fragments ,
/* Fragment fields */
& hf_data_fragments ,
& hf_data_fragment ,
& hf_data_fragment_overlap ,
& hf_data_fragment_overlap_conflicts ,
& hf_data_fragment_multiple_tails ,
& hf_data_fragment_too_long_fragment ,
& hf_data_fragment_error ,
/* Reassembled in field */
& hf_data_reassembled_in ,
/* Tag */
" Data fragments "
} ;
typedef struct _fragment_key {
address src ;
address dst ;
guint32 id ;
} fragment_key ;
static conversation_t * p_conv = NULL ;
static t38_conv * p_t38_conv = NULL ;
static t38_conv * p_t38_packet_conv = NULL ;
static t38_conv_info * p_t38_conv_info = NULL ;
static t38_conv_info * p_t38_packet_conv_info = NULL ;
2004-01-09 23:24:54 +00:00
/* RTP Version is the first 2 bits of the first octet in the UDP payload*/
# define RTP_VERSION(octet) ((octet) >> 6)
void proto_reg_handoff_t38 ( void ) ;
2006-01-15 15:01:14 +00:00
static void show_setup_info ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree , conversation_t * p_conv , t38_conv * p_t38_conv ) ;
2004-10-22 06:19:00 +00:00
/* Preferences bool to control whether or not setup info should be shown */
static gboolean global_t38_show_setup_info = TRUE ;
2006-01-15 15:01:14 +00:00
/* Can tap up to 4 T38 packets within same packet */
/* We only tap the primary part, not the redundancy */
# define MAX_T38_MESSAGES_IN_PACKET 4
static t38_packet_info t38_info_arr [ MAX_T38_MESSAGES_IN_PACKET ] ;
static int t38_info_current = 0 ;
static t38_packet_info * t38_info = NULL ;
static void t38_defragment_init ( void )
{
/* Init reassemble tables */
fragment_table_init ( & data_fragment_table ) ;
reassembled_table_init ( & data_reassembled_table ) ;
}
2004-10-22 06:19:00 +00:00
/* Set up an T38 conversation */
void t38_add_address ( packet_info * pinfo ,
address * addr , int port ,
int other_port ,
2005-08-05 00:23:22 +00:00
const gchar * setup_method , guint32 setup_frame_number )
2004-10-22 06:19:00 +00:00
{
address null_addr ;
conversation_t * p_conv ;
2006-01-15 15:01:14 +00:00
t38_conv * p_conv_data = NULL ;
2004-10-22 06:19:00 +00:00
/*
* 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 ;
}
SET_ADDRESS ( & null_addr , AT_NONE , 0 , NULL ) ;
/*
* Check if the ip address and port combination is not
* already registered as a conversation .
*/
2005-02-02 20:07:03 +00:00
p_conv = find_conversation ( setup_frame_number , addr , & null_addr , PT_UDP , port , other_port ,
2004-10-22 06:19:00 +00:00
NO_ADDR_B | ( ! other_port ? NO_PORT_B : 0 ) ) ;
/*
* If not , create a new conversation .
*/
2005-02-02 20:07:03 +00:00
if ( ! p_conv | | p_conv - > setup_frame ! = setup_frame_number ) {
p_conv = conversation_new ( setup_frame_number , addr , & null_addr , PT_UDP ,
2004-10-22 06:19:00 +00:00
( guint32 ) port , ( guint32 ) other_port ,
NO_ADDR2 | ( ! other_port ? NO_PORT2 : 0 ) ) ;
}
/* Set dissector */
conversation_set_dissector ( p_conv , t38_udp_handle ) ;
/*
* Check if the conversation has data associated with it .
*/
p_conv_data = conversation_get_proto_data ( p_conv , proto_t38 ) ;
/*
* If not , add a new data item .
*/
if ( ! p_conv_data ) {
/* Create conversation data */
2006-01-15 15:01:14 +00:00
p_conv_data = se_alloc ( sizeof ( t38_conv ) ) ;
2004-10-22 06:19:00 +00:00
conversation_add_proto_data ( p_conv , proto_t38 , p_conv_data ) ;
}
/*
* Update the conversation data .
*/
2006-01-15 15:01:14 +00:00
strncpy ( p_conv_data - > setup_method , setup_method , MAX_T38_SETUP_METHOD_SIZE ) ;
p_conv_data - > setup_method [ MAX_T38_SETUP_METHOD_SIZE ] = ' \0 ' ;
p_conv_data - > setup_frame_number = setup_frame_number ;
p_conv_data - > src_t38_info . reass_ID = 0 ;
p_conv_data - > src_t38_info . reass_start_seqnum = - 1 ;
p_conv_data - > src_t38_info . reass_data_type = 0 ;
p_conv_data - > src_t38_info . last_seqnum = - 1 ;
p_conv_data - > src_t38_info . packet_lost = 0 ;
p_conv_data - > src_t38_info . burst_lost = 0 ;
p_conv_data - > src_t38_info . time_first_t4_data = 0 ;
p_conv_data - > dst_t38_info . reass_ID = 0 ;
p_conv_data - > dst_t38_info . reass_start_seqnum = - 1 ;
p_conv_data - > dst_t38_info . reass_data_type = 0 ;
p_conv_data - > dst_t38_info . last_seqnum = - 1 ;
p_conv_data - > dst_t38_info . packet_lost = 0 ;
p_conv_data - > dst_t38_info . burst_lost = 0 ;
p_conv_data - > dst_t38_info . time_first_t4_data = 0 ;
2004-10-22 06:19:00 +00:00
}
2003-09-11 13:36:04 +00:00
2006-01-15 15:01:14 +00:00
/* T30 Routines */
static int
dissect_t30_NULL ( tvbuff_t * tvb _U_ , int offset , packet_info * pinfo _U_ , proto_tree * tree _U_ )
{
return offset ;
}
static const value_string t30_control_vals [ ] = {
{ 0xC0 , " non-final frames within the procedure " } ,
{ 0xC8 , " final frames within the procedure " } ,
{ 0 , NULL }
} ;
# define T30_FC_DIS 0x01
# define T30_FC_CSI 0x02
# define T30_FC_NSF 0x04
# define T30_FC_DTC 0x81
# define T30_FC_CIG 0x82
# define T30_FC_NSC 0x84
# define T30_FC_PWD 0x83
# define T30_FC_SEP 0x85
# define T30_FC_PSA 0x86
# define T30_FC_CIA 0x87
# define T30_FC_ISP 0x88
# define T30_FC_DCS 0x41
# define T30_FC_TSI 0x42
# define T30_FC_NSS 0x44
# define T30_FC_SUB 0x43
# define T30_FC_SID 0x45
# define T30_FC_TSA 0x46
# define T30_FC_IRA 0x47
# define T30_FC_CFR 0x21
# define T30_FC_FTT 0x22
# define T30_FC_CSA 0x24
# define T30_FC_EOM 0x71
# define T30_FC_MPS 0x72
# define T30_FC_EOP 0x74
# define T30_FC_PRI_EOM 0x79
# define T30_FC_PRI_MPS 0x7A
# define T30_FC_PRI_EOP 0x7C
# define T30_FC_PRI_EOP2 0x78
# define T30_FC_MCF 0x31
# define T30_FC_RTP 0x33
# define T30_FC_RTN 0x32
# define T30_FC_PIP 0x35
# define T30_FC_PIN 0x34
# define T30_FC_FDM 0x3F
# define T30_FC_DCN 0x5F
# define T30_FC_CRP 0x58
# define T30_FC_FNV 0x53
# define T30_FC_TNR 0x57
# define T30_FC_TR 0x56
# define T30_FC_MCF 0x31
# define T30_FC_PID 0x36
# define T30_FC_PPR 0x3D
# define T30_FC_RNR 0x37
# define T30_FC_CRP 0x58
# define T30_FC_CTC 0x48
# define T30_FC_CTR 0x23
# define T30_FC_PPS 0x7D
# define T30_FC_EOR 0x73
# define T30_FC_RR 0x76
# define T30_FC_ERR 0x38
# define T30_FC_FCD 0x60
# define T30_FC_RCP 0x61
const value_string t30_facsimile_control_field_vals [ ] = {
{ T30_FC_DIS , " Digital Identification Signal " } ,
{ T30_FC_CSI , " Called Subscriber Identification " } ,
{ T30_FC_NSF , " Non-Standard Facilities " } ,
{ T30_FC_DTC , " Digital Transmit Command " } ,
{ T30_FC_CIG , " Calling Subscriber Identification " } ,
{ T30_FC_NSC , " Non-Standard facilities Command " } ,
{ T30_FC_PWD , " Password " } ,
{ T30_FC_SEP , " Selective Polling " } ,
{ T30_FC_PSA , " Polled Subaddress " } ,
{ T30_FC_CIA , " Calling subscriber Internet Address " } ,
{ T30_FC_ISP , " Internet Selective Polling Address " } ,
{ T30_FC_DCS , " Digital Command Signal " } ,
{ T30_FC_TSI , " Transmitting Subscriber Identification " } ,
{ T30_FC_NSS , " Non-Standard facilities Set-up " } ,
{ T30_FC_SUB , " Subaddress " } ,
{ T30_FC_SID , " Sender Identification " } ,
{ T30_FC_TSA , " Transmitting Subscriber Internet address " } ,
{ T30_FC_IRA , " Internet Routing Address " } ,
{ T30_FC_CFR , " Confirmation To Receive " } ,
{ T30_FC_FTT , " Failure To Train " } ,
{ T30_FC_CSA , " Called Subscriber Internet Address " } ,
{ T30_FC_EOM , " End Of Message " } ,
{ T30_FC_MPS , " MultiPage Signal " } ,
{ T30_FC_EOP , " End Of Procedure " } ,
{ T30_FC_PRI_EOM , " Procedure Interrupt-End Of Message " } ,
{ T30_FC_PRI_MPS , " Procedure Interrupt-MultiPage Signal " } ,
{ T30_FC_PRI_EOP , " Procedure Interrupt-End Of Procedure " } ,
{ T30_FC_PRI_EOP2 , " Procedure Interrupt-End Of Procedure " } ,
{ T30_FC_MCF , " Message Confirmation " } ,
{ T30_FC_RTP , " Retrain Positive " } ,
{ T30_FC_RTN , " Retrain Negative " } ,
{ T30_FC_PIP , " Procedure Interrupt Positive " } ,
{ T30_FC_PIN , " Procedure Interrupt Negative " } ,
{ T30_FC_FDM , " File Diagnostics Message " } ,
{ T30_FC_DCN , " Disconnect " } ,
{ T30_FC_CRP , " Command Repeat " } ,
{ T30_FC_FNV , " Field Not Valid " } ,
{ T30_FC_TNR , " Transmit not ready " } ,
{ T30_FC_TR , " Transmit ready " } ,
{ T30_FC_MCF , " Message Confirmation " } ,
{ T30_FC_PID , " Procedure Interrupt Disconnect " } ,
{ T30_FC_PPR , " Partial Page Request " } ,
{ T30_FC_RNR , " Receive Not Ready " } ,
{ T30_FC_CRP , " Command Repeat " } ,
{ T30_FC_CTC , " Continue To Correct " } ,
{ T30_FC_CTR , " Response for Continue To Correct " } ,
{ T30_FC_PPS , " Partial Page Signal " } ,
{ T30_FC_EOR , " End Of Retransmission " } ,
{ T30_FC_RR , " Receive Ready " } ,
{ T30_FC_ERR , " Response for End of Retransmission " } ,
{ T30_FC_FCD , " Facsimile coded data " } ,
{ T30_FC_RCP , " Return to control for partial page " } ,
{ 0 , NULL }
} ;
const value_string t30_facsimile_control_field_vals_short [ ] = {
{ T30_FC_DIS , " DIS " } ,
{ T30_FC_CSI , " CSI " } ,
{ T30_FC_NSF , " NSF " } ,
{ T30_FC_DTC , " DTC " } ,
{ T30_FC_CIG , " CIG " } ,
{ T30_FC_NSC , " NSC " } ,
{ T30_FC_PWD , " PWD " } ,
{ T30_FC_SEP , " SEP " } ,
{ T30_FC_PSA , " PSA " } ,
{ T30_FC_CIA , " CIA " } ,
{ T30_FC_ISP , " ISP " } ,
{ T30_FC_DCS , " DCS " } ,
{ T30_FC_TSI , " TSI " } ,
{ T30_FC_NSS , " NSS " } ,
{ T30_FC_SUB , " SUB " } ,
{ T30_FC_SID , " SID " } ,
{ T30_FC_TSA , " TSA " } ,
{ T30_FC_IRA , " IRA " } ,
{ T30_FC_CFR , " CFR " } ,
{ T30_FC_FTT , " FTT " } ,
{ T30_FC_CSA , " CSA " } ,
{ T30_FC_EOM , " EOM " } ,
{ T30_FC_MPS , " MPS " } ,
{ T30_FC_EOP , " EOP " } ,
{ T30_FC_PRI_EOM , " PRI_EOM " } ,
{ T30_FC_PRI_MPS , " PRI_MPS " } ,
{ T30_FC_PRI_EOP , " EOP " } ,
{ T30_FC_PRI_EOP2 , " EOP2 " } ,
{ T30_FC_MCF , " MCF " } ,
{ T30_FC_RTP , " RTP " } ,
{ T30_FC_RTN , " RTN " } ,
{ T30_FC_PIP , " PIP " } ,
{ T30_FC_PIN , " PIN " } ,
{ T30_FC_FDM , " FDM " } ,
{ T30_FC_DCN , " DCN " } ,
{ T30_FC_CRP , " CRP " } ,
{ T30_FC_FNV , " FNV " } ,
{ T30_FC_TNR , " TNR " } ,
{ T30_FC_TR , " TR " } ,
{ T30_FC_MCF , " MCF " } ,
{ T30_FC_PID , " PID " } ,
{ T30_FC_PPR , " PPR " } ,
{ T30_FC_RNR , " RNR " } ,
{ T30_FC_CRP , " CRP " } ,
{ T30_FC_CTC , " CTC " } ,
{ T30_FC_CTR , " CTR " } ,
{ T30_FC_PPS , " PPS " } ,
{ T30_FC_EOR , " EOR " } ,
{ T30_FC_RR , " RR " } ,
{ T30_FC_ERR , " ERR " } ,
{ T30_FC_FCD , " FCD " } ,
{ T30_FC_RCP , " RCP " } ,
{ 0 , NULL }
} ;
static const value_string t30_data_signalling_rate_vals [ ] = {
{ 0x00 , " ITU-T V.27 ter fall-back mode " } ,
{ 0x04 , " ITU-T V.27 ter " } ,
{ 0x08 , " ITU-T V.29 " } ,
{ 0x0C , " ITU-T V.27 ter and V.29 " } ,
{ 0x02 , " Not used " } ,
{ 0x06 , " Reserved " } ,
{ 0x0A , " Not used " } ,
{ 0x0E , " Invalid " } ,
{ 0x01 , " Not used " } ,
{ 0x05 , " Reserved " } ,
{ 0x09 , " Not used " } ,
{ 0x0D , " ITU-T V.27 ter, V.29, and V.17 " } ,
{ 0x03 , " Not used " } ,
{ 0x07 , " Reserved " } ,
{ 0x0B , " Not used " } ,
{ 0x0F , " Reserved " } ,
} ;
static const value_string t30_data_signalling_rate_dcs_vals [ ] = {
{ 0x00 , " 2400 bit/s, ITU-T V.27 ter " } ,
{ 0x04 , " 4800 bit/s, ITU-T V.27 ter " } ,
{ 0x08 , " 9600 bit/s, ITU-T V.29 " } ,
{ 0x0C , " 7200 bit/s, ITU-T V.29 " } ,
{ 0x02 , " Invalid " } ,
{ 0x06 , " Invalid " } ,
{ 0x0A , " Reserved " } ,
{ 0x0E , " Reserved " } ,
{ 0x01 , " 14 400 bit/s, ITU-T V.17 " } ,
{ 0x05 , " 12 000 bit/s, ITU-T V.17 " } ,
{ 0x09 , " 9600 bit/s, ITU-T V.17 " } ,
{ 0x0D , " 7200 bit/s, ITU-T V.17 " } ,
{ 0x03 , " Reserved " } ,
{ 0x07 , " Reserved " } ,
{ 0x0B , " Reserved " } ,
{ 0x0F , " Reserved " } ,
} ;
static const value_string t30_recording_width_capabilities_vals [ ] = {
{ 0x00 , " Scan line length 215 mm +- 1% " } ,
{ 0x01 , " Scan line length 215 mm +- 1% and Scan line length 255 mm +- 1% and Scan line length 303 mm +- 1% " } ,
{ 0x02 , " Scan line length 215 mm +- 1% and Scan line length 255 mm +- 1% " } ,
{ 0x03 , " Invalid " } ,
} ;
static const value_string t30_recording_width_dcs_vals [ ] = {
{ 0x00 , " Scan line length 215 mm +- 1% " } ,
{ 0x01 , " Scan line length 303 mm +- 1% " } ,
{ 0x02 , " Scan line length 255 mm +- 1% " } ,
{ 0x03 , " Invalid " } ,
} ;
static const value_string t30_recording_length_capability_vals [ ] = {
{ 0x00 , " A4 (297 mm) " } ,
{ 0x01 , " Unlimited " } ,
{ 0x02 , " A4 (297 mm) and B4 (364 mm) " } ,
{ 0x03 , " Invalid " } ,
} ;
static const value_string t30_recording_length_dcs_vals [ ] = {
{ 0x00 , " A4 (297 mm) " } ,
{ 0x01 , " Unlimited " } ,
{ 0x02 , " B4 (364 mm) " } ,
{ 0x03 , " Invalid " } ,
} ;
static const value_string t30_minimum_scan_line_time_rec_vals [ ] = {
{ 0x00 , " 20 ms at 3.85 l/mm: T7.7 = T3.85 " } ,
{ 0x01 , " 40 ms at 3.85 l/mm: T7.7 = T3.85 " } ,
{ 0x02 , " 10 ms at 3.85 l/mm: T7.7 = T3.85 " } ,
{ 0x04 , " 05 ms at 3.85 l/mm: T7.7 = T3.85 " } ,
{ 0x03 , " 10 ms at 3.85 l/mm: T7.7 = 1/2 T3.85 " } ,
{ 0x06 , " 20 ms at 3.85 l/mm: T7.7 = 1/2 T3.85 " } ,
{ 0x05 , " 40 ms at 3.85 l/mm: T7.7 = 1/2 T3.85 " } ,
{ 0x07 , " 00 ms at 3.85 l/mm: T7.7 = T3.85 " } ,
} ;
static const value_string t30_partial_page_fcf2_vals [ ] = {
{ 0x00 , " NULL code which indicates the partial page boundary " } ,
{ 0xF1 , " EOM in optional T.4 error correction mode " } ,
{ 0xF2 , " MPS in optional T.4 error correction mode " } ,
{ 0xF4 , " EOP in optional T.4 error correction mode " } ,
{ 0xF8 , " EOS in optional T.4 error correction mode " } ,
{ 0xF9 , " PRI-EOM in optional T.4 error correction mode " } ,
{ 0xFA , " PRI-MPS in optional T.4 error correction mode " } ,
{ 0xFC , " PRI-EOP in optional T.4 error correction mode " } ,
} ;
static const value_string t30_minimum_scan_line_time_dcs_vals [ ] = {
{ 0x00 , " 20 ms " } ,
{ 0x01 , " 40 ms " } ,
{ 0x02 , " 10 ms " } ,
{ 0x04 , " 05 ms " } ,
{ 0x07 , " 00 ms " } ,
} ;
static const value_string t30_SharedDataMemory_capacity_vals [ ] = {
{ 0x00 , " Not available " } ,
{ 0x01 , " Level 1 = 1.0 Mbytes " } ,
{ 0x02 , " Level 2 = 2.0 Mbytes " } ,
{ 0x03 , " Level 3 = unlimited (i.e. >= 32 Mbytes) " } ,
} ;
static const true_false_string t30_octets_preferred_value = {
" 64 octets preferred " ,
" 256 octets preferred " ,
} ;
static const true_false_string t30_extension_ind_value = {
" information continues through the next octet " ,
" last octet " ,
} ;
static const true_false_string t30_compress_value = {
" Uncompressed mode " ,
" Compressed mode " ,
} ;
static const true_false_string t30_minimum_scan_value = {
" T15.4 = 1/2 T7.7 " ,
" T15.4 = T7.7 " ,
} ;
static const true_false_string t30_duplex_operation_value = {
" Duplex and half duplex operation " ,
" Half duplex operation only " ,
} ;
static const true_false_string t30_frame_size_dcs_value = {
" 64 octets " ,
" 256 octets " ,
} ;
static const true_false_string t30_res_type_sel_value = {
" inch based resolution " ,
" metric based resolution " ,
} ;
guint8 reverse_byte ( guint8 val )
{
return ( ( ( val & 0x80 ) > > 7 ) | ( ( val & 0x40 ) > > 5 ) |
( ( val & 0x20 ) > > 3 ) | ( ( val & 0x10 ) > > 1 ) |
( ( val & 0x08 ) < < 1 ) | ( ( val & 0x04 ) < < 3 ) |
( ( val & 0x02 ) < < 5 ) | ( ( val & 0x01 ) < < 7 ) ) ;
}
# define LENGTH_T30_NUM 20
gchar *
t30_get_string_numbers ( tvbuff_t * tvb , int offset , int len )
{
gchar * buf ;
int i ;
/* the lenght must be 20 bytes per T30 rec*/
if ( len ! = LENGTH_T30_NUM ) return NULL ;
buf = ep_alloc ( LENGTH_T30_NUM + 1 ) ;
for ( i = 0 ; i < LENGTH_T30_NUM ; i + + )
buf [ LENGTH_T30_NUM - i - 1 ] = reverse_byte ( tvb_get_guint8 ( tvb , offset + i ) ) ;
/* add end of string */
buf [ LENGTH_T30_NUM ] = ' \0 ' ;
return g_strstrip ( buf ) ;
}
static void
dissect_t30_numbers ( tvbuff_t * tvb , int offset , packet_info * pinfo , int len , proto_tree * tree )
{
gchar * str_num = NULL ;
str_num = t30_get_string_numbers ( tvb , offset , len ) ;
if ( str_num ) {
proto_tree_add_string_format ( tree , hf_t30_fif_number , tvb , offset , LENGTH_T30_NUM , str_num , " Number: %s " , str_num ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " - Number:%s " , str_num ) ;
g_snprintf ( t38_info - > desc , MAX_T38_DESC , " Num: %s " , str_num ) ;
}
else {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: number of digits must be 20] " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " [MALFORMED OR SHORT PACKET: number of digits must be 20] " ) ;
}
}
static void
dissect_t30_facsimile_coded_data ( tvbuff_t * tvb , int offset , packet_info * pinfo , int len , proto_tree * tree )
{
guint8 octet ;
gchar * t4_data ;
if ( len < 2 ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: FCD length must be at least 2 bytes] " ) ;
expert_add_info_format ( pinfo , NULL , PI_MALFORMED , PI_ERROR , " T30 FCD length must be at least 2 bytes " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " [MALFORMED OR SHORT PACKET] " ) ;
return ;
}
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_uint ( tree , hf_t30_t4_frame_num , tvb , offset , 1 , reverse_byte ( octet ) ) ;
offset + + ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " - Frame num:%d " , reverse_byte ( octet ) ) ;
g_snprintf ( t38_info - > desc , MAX_T38_DESC , " Frm num: %d " , reverse_byte ( octet ) ) ;
t4_data = ep_alloc ( len - 1 ) ;
tvb_memcpy ( tvb , t4_data , offset , len - 1 ) ;
proto_tree_add_bytes ( tree , hf_t30_t4_data , tvb , offset , len - 1 , t4_data ) ;
}
static void
dissect_t30_non_standard_cap ( tvbuff_t * tvb , int offset , packet_info * pinfo , int len , proto_tree * tree )
{
guint8 octet ;
gchar * non_standard_bytes ;
if ( len < 2 ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: NSC length must be at least 2 bytes] " ) ;
expert_add_info_format ( pinfo , NULL , PI_MALFORMED , PI_ERROR , " T30 NSC length must be at least 2 bytes " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " [MALFORMED OR SHORT PACKET] " ) ;
return ;
}
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_uint ( tree , hf_t30_fif_country_code , tvb , offset , 1 , octet ) ;
offset + + ;
non_standard_bytes = ep_alloc ( len - 1 ) ;
tvb_memcpy ( tvb , non_standard_bytes , offset , len - 1 ) ;
proto_tree_add_bytes ( tree , hf_t30_fif_non_stand_bytes , tvb , offset , len - 1 , non_standard_bytes ) ;
}
static void
dissect_t30_partial_page_signal ( tvbuff_t * tvb , int offset , packet_info * pinfo , int len , proto_tree * tree )
{
guint8 octet , page_count , block_count , frame_count ;
if ( len ! = 4 ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: PPS length must be 4 bytes] " ) ;
expert_add_info_format ( pinfo , NULL , PI_MALFORMED , PI_ERROR , " T30 PPS length must be 4 bytes " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " [MALFORMED OR SHORT PACKET] " ) ;
return ;
}
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_uint ( tree , hf_t30_partial_page_fcf2 , tvb , offset , 1 , octet ) ;
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
page_count = reverse_byte ( octet ) ;
proto_tree_add_uint ( tree , hf_t30_partial_page_i1 , tvb , offset , 1 , page_count ) ;
offset + + ;
octet = tvb_get_guint8 ( tvb , offset ) ;
block_count = reverse_byte ( octet ) ;
proto_tree_add_uint ( tree , hf_t30_partial_page_i2 , tvb , offset , 1 , block_count ) ;
offset + + ;
octet = tvb_get_guint8 ( tvb , offset ) ;
frame_count = reverse_byte ( octet ) ;
proto_tree_add_uint ( tree , hf_t30_partial_page_i3 , tvb , offset , 1 , frame_count ) ;
offset + + ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " - PC:%d BC:%d FC:%d " , page_count , block_count , frame_count ) ;
g_snprintf ( t38_info - > desc , MAX_T38_DESC , " PC:%d BC:%d FC:%d " , page_count , block_count , frame_count ) ;
}
static void
dissect_t30_dis_dtc ( tvbuff_t * tvb , int offset , packet_info * pinfo , int len , proto_tree * tree , gboolean dis_dtc )
{
guint8 octet ;
if ( len < 3 ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: DIS length must be at least 4 bytes] " ) ;
expert_add_info_format ( pinfo , NULL , PI_MALFORMED , PI_ERROR , " T30 DIS length must be at least 4 bytes " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " [MALFORMED OR SHORT PACKET] " ) ;
return ;
}
/* bits 1 to 8 */
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_sm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_rtif , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_3gmn , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_boolean ( tree , hf_t30_fif_v8c , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_op , tvb , offset , 1 , octet ) ;
}
/* bits 9 to 16 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
if ( dis_dtc ) proto_tree_add_boolean ( tree , hf_t30_fif_rtfc , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_rfo , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_uint ( tree , hf_t30_fif_dsr , tvb , offset , 1 , octet ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " - DSR:%s " , val_to_str ( ( octet & 0x3C ) > > 2 , t30_data_signalling_rate_vals , " <unknown> " ) ) ;
g_snprintf ( t38_info - > desc , MAX_T38_DESC , " DSR:%s " , val_to_str ( ( octet & 0x3C ) > > 2 , t30_data_signalling_rate_vals , " <unknown> " ) ) ;
}
else {
proto_tree_add_uint ( tree , hf_t30_fif_dsr_dcs , tvb , offset , 1 , octet ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " - DSR:%s " , val_to_str ( ( octet & 0x3C ) > > 2 , t30_data_signalling_rate_dcs_vals , " <unknown> " ) ) ;
g_snprintf ( t38_info - > desc , MAX_T38_DESC , " DSR:%s " , val_to_str ( ( octet & 0x3C ) > > 2 , t30_data_signalling_rate_dcs_vals , " <unknown> " ) ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_res , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_tdcc , tvb , offset , 1 , octet ) ;
/* bits 17 to 24 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
if ( dis_dtc ) {
proto_tree_add_uint ( tree , hf_t30_fif_rwc , tvb , offset , 1 , octet ) ;
proto_tree_add_uint ( tree , hf_t30_fif_rlc , tvb , offset , 1 , octet ) ;
proto_tree_add_uint ( tree , hf_t30_fif_msltcr , tvb , offset , 1 , octet ) ;
} else {
proto_tree_add_uint ( tree , hf_t30_fif_rw_dcs , tvb , offset , 1 , octet ) ;
proto_tree_add_uint ( tree , hf_t30_fif_rl_dcs , tvb , offset , 1 , octet ) ;
proto_tree_add_uint ( tree , hf_t30_fif_mslt_dcs , tvb , offset , 1 , octet ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 4 ) ) return ; /* no extension */
/* bits 25 to 32 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_cm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ecm , tvb , offset , 1 , octet ) ;
if ( ! dis_dtc ) proto_tree_add_boolean ( tree , hf_t30_fif_fs_dcs , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_t6 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 5 ) ) return ; /* no extension */
/* bits 33 to 40 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_fvc , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_boolean ( tree , hf_t30_fif_mspc , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ps , tvb , offset , 1 , octet ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_t43 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_pi , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_vc32k , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 6 ) ) return ; /* no extension */
/* bits 41 to 48 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_r8x15 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_300x300 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_r16x15 , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_boolean ( tree , hf_t30_fif_ibrp , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_mbrp , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_msltchr , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_sp , tvb , offset , 1 , octet ) ;
} else {
proto_tree_add_boolean ( tree , hf_t30_fif_rts , tvb , offset , 1 , octet ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 7 ) ) return ; /* no extension */
/* bits 49 to 56 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_sc , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_boolean ( tree , hf_t30_fif_passw , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_rttd , tvb , offset , 1 , octet ) ;
} else {
proto_tree_add_boolean ( tree , hf_t30_fif_sit , tvb , offset , 1 , octet ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_bft , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_dtm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_edi , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 8 ) ) return ; /* no extension */
/* bits 57 to 64 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_btm , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) proto_tree_add_boolean ( tree , hf_t30_fif_rttcmmd , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_chrm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_mm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 9 ) ) return ; /* no extension */
/* bits 65 to 72 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_pm26 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_dnc , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_do , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_jpeg , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_fcm , tvb , offset , 1 , octet ) ;
if ( ! dis_dtc ) proto_tree_add_boolean ( tree , hf_t30_fif_pht , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_12c , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 10 ) ) return ; /* no extension */
/* bits 73 to 80 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ns , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ci , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_cgr , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_nalet , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_naleg , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_spscb , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_spsco , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 11 ) ) return ; /* no extension */
/* bits 81 to 88 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_hkm , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_rsa , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_oc , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_hfx40 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_acn2c , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_acn3c , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_hfx40i , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 12 ) ) return ; /* no extension */
/* bits 89 to 96 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ahsn2 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ahsn3 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_t441 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_t442 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_t443 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_plmss , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 13 ) ) return ; /* no extension */
/* bits 97 to 104 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_cg300 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_100x100cg , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_spcbft , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) {
proto_tree_add_boolean ( tree , hf_t30_fif_ebft , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_isp , tvb , offset , 1 , octet ) ;
}
proto_tree_add_boolean ( tree , hf_t30_fif_ira , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 14 ) ) return ; /* no extension */
/* bits 105 to 112 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_600x600 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_1200x1200 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_300x600 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_400x800 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_600x1200 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_cg600x600 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_cg1200x1200 , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) | | ( len < 15 ) ) return ; /* no extension */
/* bits 113 to 120 */
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_dspcam , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_dspccm , tvb , offset , 1 , octet ) ;
if ( dis_dtc ) proto_tree_add_boolean ( tree , hf_t30_fif_bwmrcp , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_t45 , tvb , offset , 1 , octet ) ;
proto_tree_add_uint ( tree , hf_t30_fif_sdmc , tvb , offset , 1 , octet ) ;
proto_tree_add_boolean ( tree , hf_t30_fif_ext , tvb , offset , 1 , octet ) ;
if ( ! ( octet & 0x01 ) ) return ; /* no extension */
}
static int
dissect_t30_hdlc ( tvbuff_t * tvb , int offset , packet_info * pinfo , proto_tree * tree )
{
proto_item * it ;
proto_tree * tr ;
proto_tree * tr_fif ;
proto_item * it_fcf ;
guint8 octet ;
guint32 frag_len ;
proto_item * item ;
if ( tvb_reported_length_remaining ( tvb , offset ) < 3 ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) , " [MALFORMED OR SHORT PACKET: hdlc T30 length must be at least 4 bytes] " ) ;
expert_add_info_format ( pinfo , NULL , PI_MALFORMED , PI_ERROR , " T30 length must be at least 4 bytes " ) ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " (HDLC Reassembled: [MALFORMED OR SHORT PACKET]) " ) ;
return offset ;
}
/* if (tree) {
proto_item * item ; */
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " (HDLC Reassembled: " ) ;
it = proto_tree_add_protocol_format ( tree , proto_t30 , tvb , offset , - 1 ,
" ITU-T Recommendation T.30 " ) ;
tr = proto_item_add_subtree ( it , ett_t30 ) ;
octet = tvb_get_guint8 ( tvb , offset ) ;
item = proto_tree_add_uint ( tr , hf_t30_Address , tvb , offset , 1 , octet ) ;
if ( octet ! = 0xFF ) expert_add_info_format ( pinfo , item , PI_REASSEMBLE , PI_WARN , " T30 Address must be 0xFF " ) ;
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
item = proto_tree_add_uint ( tr , hf_t30_Control , tvb , offset , 1 , octet ) ;
if ( ( octet ! = 0xC0 ) & & ( octet ! = 0xC8 ) ) expert_add_info_format ( pinfo , item , PI_REASSEMBLE , PI_WARN , " T30 Control Field must be 0xC0 or 0xC8 " ) ;
offset + = 1 ;
octet = tvb_get_guint8 ( tvb , offset ) ;
it_fcf = proto_tree_add_uint ( tr , hf_t30_Facsimile_Control , tvb , offset , 1 , octet & 0x7F ) ;
offset + = 1 ;
tr_fif = proto_item_add_subtree ( it_fcf , ett_t30_fif ) ;
frag_len = tvb_length_remaining ( tvb , offset ) ;
t38_info - > t30_Facsimile_Control = octet ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " %s - %s " , val_to_str ( octet & 0x7F , t30_facsimile_control_field_vals_short , " <unknown> " ) ,
val_to_str ( octet & 0x7F , t30_facsimile_control_field_vals , " <unknown> " ) ) ;
switch ( octet & 0x7F ) {
case T30_FC_DIS :
case T30_FC_DTC :
dissect_t30_dis_dtc ( tvb , offset , pinfo , frag_len , tr_fif , TRUE ) ;
break ;
case T30_FC_DCS :
dissect_t30_dis_dtc ( tvb , offset , pinfo , frag_len , tr_fif , FALSE ) ;
break ;
case T30_FC_CSI :
case T30_FC_CIG :
case T30_FC_TSI :
case T30_FC_PWD :
case T30_FC_SEP :
case T30_FC_SUB :
case T30_FC_SID :
case T30_FC_PSA :
dissect_t30_numbers ( tvb , offset , pinfo , frag_len , tr_fif ) ;
break ;
case T30_FC_NSF :
case T30_FC_NSC :
case T30_FC_NSS :
dissect_t30_non_standard_cap ( tvb , offset , pinfo , frag_len , tr_fif ) ;
break ;
case T30_FC_FCD :
dissect_t30_facsimile_coded_data ( tvb , offset , pinfo , frag_len , tr_fif ) ;
break ;
case T30_FC_PPS :
dissect_t30_partial_page_signal ( tvb , offset , pinfo , frag_len , tr_fif ) ;
break ;
}
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_str ( pinfo - > cinfo , COL_INFO , " ) " ) ;
/* }*/
return offset ;
}
/* T38 Routines */
2003-09-11 13:36:04 +00:00
static int
2006-06-29 15:26:41 +00:00
dissect_t38_NULL ( tvbuff_t * tvb _U_ , int offset , asn_ctx_t * actx _U_ , proto_tree * tree _U_ , int hf_index _U_ )
2003-09-11 13:36:04 +00:00
{
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_choice_t t30_indicator_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 1 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 2 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 3 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 4 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 5 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 6 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 7 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 8 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 9 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 10 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 11 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 12 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 13 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 14 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 15 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 16 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 17 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 18 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 19 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 20 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 21 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 22 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
{ 0 , NULL , 0 , NULL }
} ;
2005-10-03 05:55:26 +00:00
const value_string t30_indicator_vals [ ] = {
2003-09-11 13:36:04 +00:00
{ 0 , " no-signal " } ,
{ 1 , " cng " } ,
{ 2 , " ced " } ,
{ 3 , " v21-preamble " } ,
{ 4 , " v27-2400-training " } ,
{ 5 , " v27-4800-training " } ,
{ 6 , " v29-7200-training " } ,
{ 7 , " v29-9600-training " } ,
{ 8 , " v17-7200-short-training " } ,
{ 9 , " v17-7200-long-training " } ,
{ 10 , " v17-9600-short-training " } ,
{ 11 , " v17-9600-long-training " } ,
{ 12 , " v17-12000-short-training " } ,
{ 13 , " v17-12000-long-training " } ,
{ 14 , " v17-14400-short-training " } ,
{ 15 , " v17-14400-long-training " } ,
{ 16 , " v8-ansam " } ,
{ 17 , " v8-signal " } ,
{ 18 , " v34-cntl-channel-1200 " } ,
{ 19 , " v34-pri-channel " } ,
{ 20 , " v34-CC-retrain " } ,
{ 21 , " v33-12000-training " } ,
{ 22 , " v33-14400-training " } ,
{ 0 , NULL } ,
} ;
static int
2006-06-29 15:26:41 +00:00
dissect_t38_T30_Indicator ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_choice ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_t30_indicator ,
2005-09-19 07:35:05 +00:00
t30_indicator_choice , & T30ind_value ) ;
2004-01-09 23:24:54 +00:00
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) & & primary_part ) {
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " t30ind: %s " ,
2004-01-09 23:24:54 +00:00
val_to_str ( T30ind_value , t30_indicator_vals , " <unknown> " ) ) ;
}
2005-10-03 05:55:26 +00:00
/* info for tap */
if ( primary_part )
t38_info - > t30ind_value = T30ind_value ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_choice_t data_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 1 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 2 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 3 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 4 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 5 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 6 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 7 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 8 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 9 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 10 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 11 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 12 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 13 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 14 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
{ 0 , NULL , 0 , NULL }
} ;
2006-01-15 15:01:14 +00:00
const value_string t30_data_vals [ ] = {
2003-09-11 13:36:04 +00:00
{ 0 , " v21 " } ,
{ 1 , " v27-2400 " } ,
{ 2 , " v27-4800 " } ,
{ 3 , " v29-7200 " } ,
{ 4 , " v29-9600 " } ,
{ 5 , " v17-7200 " } ,
{ 6 , " v17-9600 " } ,
{ 7 , " v17-12000 " } ,
2003-10-09 22:35:07 +00:00
{ 8 , " v17-14400 " } ,
{ 9 , " v8 " } ,
{ 10 , " v34-pri-rate " } ,
{ 11 , " v34-CC-1200 " } ,
{ 12 , " v34-pri-ch " } ,
{ 13 , " v33-12000 " } ,
{ 14 , " v33-14400 " } ,
2003-09-11 13:36:04 +00:00
{ 0 , NULL } ,
} ;
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Data ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_choice ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_data ,
2005-09-19 07:35:05 +00:00
data_choice , & Data_value ) ;
2004-01-09 23:24:54 +00:00
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) & & primary_part ) {
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " data:%s: " ,
2006-01-15 15:01:14 +00:00
val_to_str ( Data_value , t30_data_vals , " <unknown> " ) ) ;
2004-01-09 23:24:54 +00:00
}
2005-10-03 05:55:26 +00:00
/* info for tap */
if ( primary_part )
t38_info - > data_value = Data_value ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_choice_t Type_of_msg_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_t30_indicator , ASN1_NO_EXTENSIONS ,
dissect_t38_T30_Indicator } ,
{ 1 , & hf_t38_data , ASN1_NO_EXTENSIONS ,
dissect_t38_Data } ,
2003-09-11 13:36:04 +00:00
{ 0 , NULL , 0 , NULL }
} ;
static const value_string Type_of_msg_vals [ ] = {
{ 0 , " t30-indicator " } ,
{ 1 , " data " } ,
{ 0 , NULL }
} ;
static int
2006-05-22 11:49:50 +00:00
dissect_t38_Type_of_msg ( tvbuff_t * tvb , int offset , asn_ctx_t * actx _U_ , proto_tree * tree , int hf_index ) {
offset = dissect_per_choice ( tvb , offset , actx , tree , hf_index ,
2005-09-19 07:35:05 +00:00
ett_t38_Type_of_msg , Type_of_msg_choice ,
2005-07-16 21:34:19 +00:00
& Type_of_msg_value ) ;
2005-10-03 05:55:26 +00:00
/* info for tap */
if ( primary_part )
t38_info - > type_msg = Type_of_msg_value ;
2005-07-16 21:34:19 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_choice_t Data_Field_field_type_PreCorrigendum_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 1 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 2 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 3 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 4 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 5 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 6 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 7 , & hf_t38_null , ASN1_NO_EXTENSIONS ,
2003-09-11 13:36:04 +00:00
dissect_t38_NULL } ,
{ 0 , NULL , 0 , NULL }
} ;
2004-01-09 23:24:54 +00:00
2004-10-11 06:51:19 +00:00
static const per_choice_t Data_Field_field_type_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 1 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 2 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 3 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 4 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 5 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 6 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 7 , & hf_t38_null , ASN1_EXTENSION_ROOT ,
2004-01-09 23:24:54 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 8 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2004-01-26 22:16:43 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 9 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2004-01-26 22:16:43 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 10 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2004-01-26 22:16:43 +00:00
dissect_t38_NULL } ,
2006-06-29 15:26:41 +00:00
{ 11 , & hf_t38_null , ASN1_NOT_EXTENSION_ROOT ,
2004-01-26 22:16:43 +00:00
dissect_t38_NULL } ,
2004-01-09 23:24:54 +00:00
{ 0 , NULL , 0 , NULL }
} ;
2003-09-11 13:36:04 +00:00
static const value_string Data_Field_field_type_vals [ ] = {
{ 0 , " hdlc-data " } ,
{ 1 , " hdlc-sig-end " } ,
{ 2 , " hdlc-fcs-OK " } ,
{ 3 , " hdlc-fcs-BAD " } ,
{ 4 , " hdlc-fcs-OK-sig-end " } ,
{ 5 , " hdlc-fcs-BAD-sig-end " } ,
{ 6 , " t4-non-ecm-data " } ,
{ 7 , " t4-non-ecm-sig-end " } ,
2004-01-26 22:16:43 +00:00
{ 8 , " cm-message " } ,
{ 9 , " jm-message " } ,
{ 10 , " ci-message " } ,
{ 11 , " v34-rate " } ,
2003-09-11 13:36:04 +00:00
{ 0 , NULL } ,
} ;
2006-01-15 15:01:14 +00:00
fragment_data *
force_reassmeble_seq ( tvbuff_t * tvb , int offset , packet_info * pinfo , guint32 id ,
GHashTable * fragment_table , guint32 frag_number )
{
fragment_key key ;
fragment_data * fd_head ;
fragment_data * fd_i ;
fragment_data * last_fd ;
guint32 dfpos , size , packet_lost , burst_lost , seq_num ;
/* create key to search hash with */
key . src = pinfo - > src ;
key . dst = pinfo - > dst ;
key . id = id ;
fd_head = g_hash_table_lookup ( fragment_table , & key ) ;
/* have we already seen this frame ?*/
if ( pinfo - > fd - > flags . visited ) {
if ( fd_head ! = NULL & & fd_head - > flags & FD_DEFRAGMENTED ) {
return fd_head ;
} else {
return NULL ;
}
}
if ( fd_head = = NULL ) {
/* we must have it to continue */
return NULL ;
}
/* check for packet lost and count the burst of packet lost */
packet_lost = 0 ;
burst_lost = 0 ;
seq_num = 0 ;
for ( fd_i = fd_head - > next ; fd_i ; fd_i = fd_i - > next ) {
if ( seq_num ! = fd_i - > offset ) {
packet_lost + = fd_i - > offset - seq_num ;
if ( ( fd_i - > offset - seq_num ) > burst_lost ) {
burst_lost = fd_i - > offset - seq_num ;
}
}
seq_num = fd_i - > offset + 1 ;
}
/* we have received an entire packet, defragment it and
* free all fragments
*/
size = 0 ;
last_fd = NULL ;
for ( fd_i = fd_head - > next ; fd_i ; fd_i = fd_i - > next ) {
if ( ! last_fd | | last_fd - > offset ! = fd_i - > offset ) {
size + = fd_i - > len ;
}
last_fd = fd_i ;
}
fd_head - > data = g_malloc ( size ) ;
fd_head - > len = size ; /* record size for caller */
/* add all data fragments */
dfpos = 0 ;
last_fd = NULL ;
for ( fd_i = fd_head - > next ; fd_i & & fd_i - > len + dfpos < = size ; fd_i = fd_i - > next ) {
if ( fd_i - > len ) {
if ( ! last_fd | | last_fd - > offset ! = fd_i - > offset ) {
memcpy ( fd_head - > data + dfpos , fd_i - > data , fd_i - > len ) ;
dfpos + = fd_i - > len ;
} else {
/* duplicate/retransmission/overlap */
fd_i - > flags | = FD_OVERLAP ;
fd_head - > flags | = FD_OVERLAP ;
if ( ( last_fd - > len ! = fd_i - > datalen )
| | memcmp ( last_fd - > data , fd_i - > data , last_fd - > len ) ) {
2006-01-17 06:39:57 +00:00
fd_i - > flags | = FD_OVERLAPCONFLICT ;
2006-01-15 15:01:14 +00:00
fd_head - > flags | = FD_OVERLAPCONFLICT ;
}
}
}
last_fd = fd_i ;
}
/* we have defragmented the pdu, now free all fragments*/
for ( fd_i = fd_head - > next ; fd_i ; fd_i = fd_i - > next ) {
if ( fd_i - > data ) {
g_free ( fd_i - > data ) ;
fd_i - > data = NULL ;
}
}
/* mark this packet as defragmented */
fd_head - > flags | = FD_DEFRAGMENTED ;
fd_head - > reassembled_in = pinfo - > fd - > num ;
if ( check_col ( pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( pinfo - > cinfo , COL_INFO , " (t4-data Reassembled: %d pack lost, %d pack burst lost) " , packet_lost , burst_lost ) ;
p_t38_packet_conv_info - > packet_lost = packet_lost ;
p_t38_packet_conv_info - > burst_lost = burst_lost ;
return fd_head ;
}
2003-09-11 13:36:04 +00:00
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Data_Field_field_type ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2004-01-09 23:24:54 +00:00
if ( use_pre_corrigendum_asn1_specification ) {
2006-05-22 11:49:50 +00:00
offset = dissect_per_choice ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_Data_Field_field_type ,
2005-09-19 07:35:05 +00:00
Data_Field_field_type_PreCorrigendum_choice , & Data_Field_field_type_value ) ;
2004-01-09 23:24:54 +00:00
}
else {
2006-05-22 11:49:50 +00:00
offset = dissect_per_choice ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_Data_Field_field_type ,
2005-09-19 07:35:05 +00:00
Data_Field_field_type_choice , & Data_Field_field_type_value ) ;
2004-01-09 23:24:54 +00:00
}
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) & & primary_part ) {
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " %s " ,
2004-01-09 23:24:54 +00:00
val_to_str ( Data_Field_field_type_value , Data_Field_field_type_vals , " <unknown> " ) ) ;
}
2006-01-15 15:01:14 +00:00
/* We only reassmeble packets in the Primary part and in the first two Items. */
/* There maybe be t38 packets with more than two Items, but reassemble those packets is not easy */
/* using the current ressaemble functions. */
/* TODO: reassemble all the Items in one frame */
if ( primary_part & & ( Data_Field_item_num < 2 ) ) {
if ( Data_Field_field_type_value = = 2 | | Data_Field_field_type_value = = 4 | | Data_Field_field_type_value = = 7 ) { /* hdlc-fcs-OK or hdlc-fcs-OK-sig-end or t4-non-ecm-sig-end*/
fragment_data * frag_msg = NULL ;
tvbuff_t * new_tvb = NULL ;
2006-05-22 11:49:50 +00:00
gboolean save_fragmented = actx - > pinfo - > fragmented ;
2006-01-15 15:01:14 +00:00
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = TRUE ;
2006-01-15 15:01:14 +00:00
2006-01-23 07:12:26 +00:00
/* if reass_start_seqnum=-1 it means we have received the end of the fragmente, without received any fragment data */
if ( p_t38_packet_conv_info - > reass_start_seqnum ! = - 1 ) {
2006-05-22 11:49:50 +00:00
frag_msg = fragment_add_seq ( tvb , offset , actx - > pinfo ,
2006-01-23 07:12:26 +00:00
p_t38_packet_conv_info - > reass_ID , /* ID for fragments belonging together */
data_fragment_table , /* list of message fragments */
seq_number + Data_Field_item_num - ( guint32 ) p_t38_packet_conv_info - > reass_start_seqnum , /* fragment sequence number */
/*0,*/
0 , /* fragment length */
FALSE ) ; /* More fragments */
if ( Data_Field_field_type_value = = 7 ) {
/* if there was packet lost or other errors during the defrag then frag_msg is NULL. This could also means
* there are out of order packets ( e . g , got the tail frame t4 - non - ecm - sig - end before the last fragment ) ,
* but we will assume there was packet lost instead , which is more usual . So , we are going to reassemble the packet
* and get some stat , like packet lost and burst number of packet lost
*/
if ( ! frag_msg ) {
2006-05-22 11:49:50 +00:00
force_reassmeble_seq ( tvb , offset , actx - > pinfo ,
2006-01-23 07:12:26 +00:00
p_t38_packet_conv_info - > reass_ID , /* ID for fragments belonging together */
data_fragment_table , /* list of message fragments */
seq_number + Data_Field_item_num - ( guint32 ) p_t38_packet_conv_info - > reass_start_seqnum ) ; /* fragment sequence number */
} else {
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) )
col_append_str ( actx - > pinfo - > cinfo , COL_INFO , " (t4-data Reassembled: No packet lost) " ) ;
2006-01-23 07:12:26 +00:00
g_snprintf ( t38_info - > desc_comment , MAX_T38_DESC , " No packet lost " ) ;
}
2006-01-15 15:01:14 +00:00
2006-01-23 07:12:26 +00:00
if ( p_t38_packet_conv_info - > packet_lost ) {
g_snprintf ( t38_info - > desc_comment , MAX_T38_DESC , " Pack lost: %d, Pack burst lost: %d " , p_t38_packet_conv_info - > packet_lost , p_t38_packet_conv_info - > burst_lost ) ;
} else {
g_snprintf ( t38_info - > desc_comment , MAX_T38_DESC , " No packet lost " ) ;
}
2005-10-03 05:55:26 +00:00
2006-05-22 11:49:50 +00:00
new_tvb = process_reassembled_data ( tvb , offset , actx - > pinfo ,
2006-01-23 07:12:26 +00:00
" Reassembled Message " , frag_msg , & data_frag_items , NULL , tree ) ;
2006-01-15 15:01:14 +00:00
2006-01-23 07:12:26 +00:00
/* Now reset fragmentation information in pinfo */
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = save_fragmented ;
2006-01-15 15:01:14 +00:00
2006-01-23 07:12:26 +00:00
t38_info - > time_first_t4_data = p_t38_packet_conv_info - > time_first_t4_data ;
t38_info - > frame_num_first_t4_data = p_t38_packet_conv_info - > reass_ID ; /* The reass_ID is the Frame number of the first t4 fragment */
2006-01-15 15:01:14 +00:00
2006-01-23 07:12:26 +00:00
} else {
2006-05-22 11:49:50 +00:00
new_tvb = process_reassembled_data ( tvb , offset , actx - > pinfo ,
2006-01-23 07:12:26 +00:00
" Reassembled Message " , frag_msg , & data_frag_items , NULL , tree ) ;
2005-10-03 05:55:26 +00:00
2006-01-23 07:12:26 +00:00
/* Now reset fragmentation information in pinfo */
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = save_fragmented ;
2006-01-15 15:01:14 +00:00
2006-05-22 11:49:50 +00:00
if ( new_tvb ) dissect_t30_hdlc ( new_tvb , 0 , actx - > pinfo , tree ) ;
2006-01-23 07:12:26 +00:00
}
} else {
if ( tree ) {
proto_tree_add_text ( tree , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) ,
" [RECEIVED END OF FRAGMENT W/OUT ANY FRAGMENT DATA] " ) ;
}
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) ) {
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " [Malformed?] " ) ;
2006-01-23 07:12:26 +00:00
}
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = save_fragmented ;
2006-01-15 15:01:14 +00:00
}
}
2005-10-03 05:55:26 +00:00
2006-01-15 15:01:14 +00:00
/* reset the reassemble ID and the start seq number if it is not HDLC data */
if ( p_t38_conv & & ( ( ( Data_Field_field_type_value > 0 ) & & ( Data_Field_field_type_value < 6 ) ) | | ( Data_Field_field_type_value = = 7 ) ) ) {
p_t38_conv_info - > reass_ID = 0 ;
p_t38_conv_info - > reass_start_seqnum = - 1 ;
}
t38_info - > Data_Field_field_type_value = Data_Field_field_type_value ;
}
2003-09-11 13:36:04 +00:00
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Data_Field_field_data ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2005-07-04 09:35:25 +00:00
tvbuff_t * value_tvb = NULL ;
guint32 value_len ;
2004-01-09 23:24:54 +00:00
2006-05-22 11:49:50 +00:00
offset = dissect_per_octet_string ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , 1 , 65535 ,
2005-07-04 09:35:25 +00:00
& value_tvb ) ;
value_len = tvb_length ( value_tvb ) ;
2004-01-09 23:24:54 +00:00
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) & & primary_part ) {
2004-01-09 23:24:54 +00:00
if ( value_len < 8 ) {
2006-05-22 11:49:50 +00:00
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " [%s] " ,
2005-07-04 09:35:25 +00:00
tvb_bytes_to_str ( value_tvb , 0 , value_len ) ) ;
2004-01-09 23:24:54 +00:00
}
else {
2006-05-22 11:49:50 +00:00
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " [%s...] " ,
2005-07-04 09:35:25 +00:00
tvb_bytes_to_str ( value_tvb , 0 , 7 ) ) ;
2004-01-09 23:24:54 +00:00
}
}
2005-10-03 05:55:26 +00:00
2006-01-15 15:01:14 +00:00
/* We only reassmeble packets in the Primary part and in the first two Items. */
/* There maybe be t38 packets with more than two Items, but reassemble those packets is not easy */
/* using the current ressaemble functions. */
/* TODO: reassemble all the Items in one frame */
if ( primary_part & & ( Data_Field_item_num < 2 ) ) {
tvbuff_t * new_tvb = NULL ;
fragment_data * frag_msg = NULL ;
/* HDLC Data or t4-non-ecm-data */
if ( Data_Field_field_type_value = = 0 | | Data_Field_field_type_value = = 6 ) { /* 0=HDLC Data or 6=t4-non-ecm-data*/
2006-05-22 11:49:50 +00:00
gboolean save_fragmented = actx - > pinfo - > fragmented ;
2006-01-15 15:01:14 +00:00
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = TRUE ;
2006-01-15 15:01:14 +00:00
/* if we have not reassembled this packet and it is the first fragment, reset the reassemble ID and the start seq number*/
if ( p_t38_packet_conv & & p_t38_conv & & ( p_t38_packet_conv_info - > reass_ID = = 0 ) ) {
/* we use the first fragment's frame_number as fragment ID because the protocol doesn't provide it */
2006-05-22 11:49:50 +00:00
p_t38_conv_info - > reass_ID = actx - > pinfo - > fd - > num ;
2006-01-15 15:01:14 +00:00
p_t38_conv_info - > reass_start_seqnum = seq_number ;
2006-05-22 11:49:50 +00:00
p_t38_conv_info - > time_first_t4_data = nstime_to_sec ( & actx - > pinfo - > fd - > rel_ts ) ;
2006-01-15 15:01:14 +00:00
p_t38_packet_conv_info - > reass_ID = p_t38_conv_info - > reass_ID ;
p_t38_packet_conv_info - > reass_start_seqnum = p_t38_conv_info - > reass_start_seqnum ;
p_t38_packet_conv_info - > time_first_t4_data = p_t38_conv_info - > time_first_t4_data ;
}
2005-10-03 05:55:26 +00:00
2006-05-22 11:49:50 +00:00
frag_msg = fragment_add_seq ( value_tvb , 0 , actx - > pinfo ,
2006-01-15 15:01:14 +00:00
p_t38_packet_conv_info - > reass_ID , /* ID for fragments belonging together */
data_fragment_table , /* list of message fragments */
seq_number - ( guint32 ) p_t38_packet_conv_info - > reass_start_seqnum , /* fragment sequence number */
value_len , /* fragment length */
TRUE ) ; /* More fragments */
2006-05-22 11:49:50 +00:00
new_tvb = process_reassembled_data ( tvb , offset , actx - > pinfo ,
2006-01-15 15:01:14 +00:00
" Reassembled Message " , frag_msg , & data_frag_items , NULL , tree ) ;
if ( ! frag_msg ) { /* Not last packet of reassembled */
if ( Data_Field_field_type_value = = 0 ) {
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " (HDLC fragment %u) " , seq_number - ( guint32 ) p_t38_packet_conv_info - > reass_start_seqnum ) ;
2006-01-15 15:01:14 +00:00
} else {
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) )
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " (t4-data fragment %u) " , seq_number - ( guint32 ) p_t38_packet_conv_info - > reass_start_seqnum ) ;
2006-01-15 15:01:14 +00:00
}
}
/* Now reset fragmentation information in pinfo */
2006-05-22 11:49:50 +00:00
actx - > pinfo - > fragmented = save_fragmented ;
2005-10-03 05:55:26 +00:00
}
}
2003-09-11 13:36:04 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_sequence_t Data_Field_item_sequence [ ] = {
2006-06-29 15:26:41 +00:00
{ " field-type " , & hf_t38_Data_Field_field_type , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
2003-09-11 13:36:04 +00:00
dissect_t38_Data_Field_field_type } ,
2006-06-29 15:26:41 +00:00
{ " field-data " , & hf_t38_Data_Field_field_data , ASN1_NO_EXTENSIONS , ASN1_OPTIONAL ,
2003-09-11 13:36:04 +00:00
dissect_t38_Data_Field_field_data } ,
2006-06-30 10:18:25 +00:00
{ NULL , NULL , 0 , 0 , NULL }
2003-09-11 13:36:04 +00:00
} ;
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Data_Field_item ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_Data_Field_item ,
2003-09-11 13:36:04 +00:00
Data_Field_item_sequence ) ;
2006-01-15 15:01:14 +00:00
if ( primary_part ) Data_Field_item_num + + ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2005-07-16 21:34:19 +00:00
static const per_sequence_t t38_Data_Field_sequence_of [ 1 ] = {
2006-06-29 15:26:41 +00:00
{ " " , & hf_t38_Data_Field_item , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL , dissect_t38_Data_Field_item } ,
2005-07-16 21:34:19 +00:00
} ;
2003-09-11 13:36:04 +00:00
static int
2006-05-22 11:49:50 +00:00
dissect_t38_Data_Field ( tvbuff_t * tvb , int offset , asn_ctx_t * actx _U_ , proto_tree * tree , int hf_index ) {
offset = dissect_per_sequence_of ( tvb , offset , actx , tree , hf_index ,
2005-07-16 21:34:19 +00:00
ett_t38_Data_Field , t38_Data_Field_sequence_of ) ;
return offset ;
}
2003-09-11 13:36:04 +00:00
2004-10-11 06:51:19 +00:00
static const per_sequence_t IFPPacket_sequence [ ] = {
2006-06-29 15:26:41 +00:00
{ " type-of-msg " , & hf_t38_Type_of_msg , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL , dissect_t38_Type_of_msg } ,
{ " data-field " , & hf_t38_Data_Field , ASN1_NO_EXTENSIONS , ASN1_OPTIONAL , dissect_t38_Data_Field } ,
2006-06-30 10:18:25 +00:00
{ NULL , NULL , 0 , 0 , NULL }
2003-09-11 13:36:04 +00:00
} ;
static int
2006-05-22 11:49:50 +00:00
dissect_t38_IFPPacket ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence ( tvb , offset , actx ,
2003-09-11 13:36:04 +00:00
tree , hf_t38_IFPPacket , ett_t38_IFPPacket ,
IFPPacket_sequence ) ;
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Seq_number ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_constrained_integer ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , 0 , 65535 ,
2006-05-22 11:49:50 +00:00
& seq_number , FALSE ) ;
2005-10-03 05:55:26 +00:00
/* info for tap */
if ( primary_part )
t38_info - > seq_num = seq_number ;
2004-01-09 23:24:54 +00:00
2006-05-22 11:49:50 +00:00
if ( check_col ( actx - > pinfo - > cinfo , COL_INFO ) ) {
col_append_fstr ( actx - > pinfo - > cinfo , COL_INFO , " Seq=%05u " , seq_number ) ;
2004-01-09 23:24:54 +00:00
}
2003-09-11 13:36:04 +00:00
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Primary_ifp_packet ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
guint32 length ;
2006-01-15 15:01:14 +00:00
2004-01-09 23:24:54 +00:00
primary_part = TRUE ;
2003-09-11 13:36:04 +00:00
2006-05-22 11:49:50 +00:00
offset = dissect_per_length_determinant ( tvb , offset , actx ,
2003-09-11 13:36:04 +00:00
tree , hf_t38_primary_ifp_packet_length , & length ) ;
2006-05-22 11:49:50 +00:00
offset = dissect_t38_IFPPacket ( tvb , offset , actx , tree ) ;
2006-01-15 15:01:14 +00:00
/* if is a valid t38 packet, add to tap */
2006-05-22 11:49:50 +00:00
if ( p_t38_packet_conv & & ( ! actx - > pinfo - > in_error_pkt ) & & ( ( gint32 ) seq_number ! = p_t38_packet_conv_info - > last_seqnum ) )
tap_queue_packet ( t38_tap , actx - > pinfo , t38_info ) ;
2006-01-15 15:01:14 +00:00
if ( p_t38_conv ) p_t38_conv_info - > last_seqnum = ( gint32 ) seq_number ;
2003-09-11 13:36:04 +00:00
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Secondary_ifp_packets_item ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
guint32 length ;
2006-05-22 11:49:50 +00:00
offset = dissect_per_length_determinant ( tvb , offset , actx ,
2003-09-11 13:36:04 +00:00
tree , hf_t38_secondary_ifp_packets_item_length , & length ) ;
2006-05-22 11:49:50 +00:00
offset = dissect_t38_IFPPacket ( tvb , offset , actx , tree ) ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2005-07-16 21:34:19 +00:00
static const per_sequence_t SEQUENCE_OF_t38_secondary_ifp_packets_sequence_of [ 1 ] = {
2006-06-29 15:26:41 +00:00
{ " " , & hf_t38_dummy , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL , dissect_t38_Secondary_ifp_packets_item } ,
2005-07-16 21:34:19 +00:00
} ;
2003-09-11 13:36:04 +00:00
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Secondary_ifp_packets ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
/* When the field-data is not present, we MUST offset 1 byte*/
if ( ( Data_Field_field_type_value ! = 0 ) & &
2005-10-03 05:55:26 +00:00
( Data_Field_field_type_value ! = 6 ) & &
( Data_Field_field_type_value ! = 7 ) )
2003-09-11 13:36:04 +00:00
{
offset = offset + 8 ;
}
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence_of ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_secondary_ifp_packets ,
2005-07-16 21:34:19 +00:00
SEQUENCE_OF_t38_secondary_ifp_packets_sequence_of ) ;
2003-09-11 13:36:04 +00:00
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Fec_npackets ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-06-29 15:26:41 +00:00
offset = dissect_per_integer ( tvb , offset , actx , tree , hf_index , NULL ) ;
2003-09-11 13:36:04 +00:00
return offset ;
}
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Fec_data_item ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_octet_string ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , NO_BOUND , NO_BOUND ,
2005-07-04 09:35:25 +00:00
NULL ) ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2005-07-16 21:34:19 +00:00
static const per_sequence_t T_t38_fec_data_sequence_of [ 1 ] = {
2006-06-29 15:26:41 +00:00
{ " " , & hf_t38_fec_data_item , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL , dissect_t38_Fec_data_item } ,
2005-07-16 21:34:19 +00:00
} ;
2003-09-11 13:36:04 +00:00
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Fec_data ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence_of ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_fec_data ,
2005-07-16 21:34:19 +00:00
T_t38_fec_data_sequence_of ) ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_sequence_t fec_info_sequence [ ] = {
2006-06-29 15:26:41 +00:00
{ " fec-npackets " , & hf_t38_fec_npackets , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
dissect_t38_Fec_npackets } ,
{ " fec-data " , & hf_t38_fec_data , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
dissect_t38_Fec_data } ,
2006-06-30 10:18:25 +00:00
{ NULL , NULL , 0 , 0 , NULL }
2003-09-11 13:36:04 +00:00
} ;
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Fec_info ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_fec_info ,
2003-09-11 13:36:04 +00:00
fec_info_sequence ) ;
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_choice_t error_recovery_choice [ ] = {
2006-06-29 15:26:41 +00:00
{ 0 , & hf_t38_secondary_ifp_packets , ASN1_NO_EXTENSIONS ,
dissect_t38_Secondary_ifp_packets } ,
{ 1 , & hf_t38_fec_info , ASN1_NO_EXTENSIONS ,
dissect_t38_Fec_info } ,
2003-09-11 13:36:04 +00:00
{ 0 , NULL , 0 , NULL }
} ;
static const value_string error_recovery_vals [ ] = {
{ 0 , " secondary-ifp-packets " } ,
{ 1 , " fec-info " } ,
{ 0 , NULL }
} ;
static int
2006-06-29 15:26:41 +00:00
dissect_t38_Error_recovery ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree , int hf_index )
2003-09-11 13:36:04 +00:00
{
2004-01-09 23:24:54 +00:00
primary_part = FALSE ;
2006-05-22 11:49:50 +00:00
offset = dissect_per_choice ( tvb , offset , actx ,
2006-06-29 15:26:41 +00:00
tree , hf_index , ett_t38_error_recovery ,
2005-09-19 07:35:05 +00:00
error_recovery_choice , NULL ) ;
2004-01-09 23:24:54 +00:00
primary_part = TRUE ;
2003-09-11 13:36:04 +00:00
return offset ;
}
2004-10-11 06:51:19 +00:00
static const per_sequence_t UDPTLPacket_sequence [ ] = {
2006-06-29 15:26:41 +00:00
{ " seq-number " , & hf_t38_seq_number , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
dissect_t38_Seq_number } ,
{ " primary-ifp-packet " , & hf_t38_dummy , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
dissect_t38_Primary_ifp_packet } ,
{ " error-recovery " , & hf_t38_error_recovery , ASN1_NO_EXTENSIONS , ASN1_NOT_OPTIONAL ,
dissect_t38_Error_recovery } ,
2006-06-30 10:18:25 +00:00
{ NULL , NULL , 0 , 0 , NULL }
2003-09-11 13:36:04 +00:00
} ;
static int
2006-05-22 11:49:50 +00:00
dissect_t38_UDPTLPacket ( tvbuff_t * tvb , int offset , asn_ctx_t * actx , proto_tree * tree )
2003-09-11 13:36:04 +00:00
{
/* Initialize to something else than data type */
Data_Field_field_type_value = 1 ;
2006-05-22 11:49:50 +00:00
offset = dissect_per_sequence ( tvb , offset , actx ,
2003-09-11 13:36:04 +00:00
tree , hf_t38_UDPTLPacket , ett_t38_UDPTLPacket ,
UDPTLPacket_sequence ) ;
return offset ;
}
2006-01-23 07:12:26 +00:00
/* initialize the tap t38_info and the conversation */
2003-09-11 13:36:04 +00:00
static void
2006-01-23 07:12:26 +00:00
init_t38_info_conv ( packet_info * pinfo )
2003-09-11 13:36:04 +00:00
{
2006-01-15 15:01:14 +00:00
/* tap info */
t38_info_current + + ;
if ( t38_info_current = = MAX_T38_MESSAGES_IN_PACKET ) {
t38_info_current = 0 ;
}
t38_info = & t38_info_arr [ t38_info_current ] ;
t38_info - > seq_num = 0 ;
t38_info - > type_msg = 0 ;
t38_info - > data_value = 0 ;
t38_info - > t30ind_value = 0 ;
t38_info - > setup_frame_number = 0 ;
t38_info - > Data_Field_field_type_value = 0 ;
t38_info - > desc [ 0 ] = ' \0 ' ;
t38_info - > desc_comment [ 0 ] = ' \0 ' ;
t38_info - > time_first_t4_data = 0 ;
t38_info - > frame_num_first_t4_data = 0 ;
2003-09-11 13:36:04 +00:00
2006-01-15 15:01:14 +00:00
/*
p_t38_packet_conv hold the conversation info in each of the packets .
p_t38_conv hold the conversation info used to reassemble the HDLC packets , and also the Setup info ( e . g SDP )
If we already have p_t38_packet_conv in the packet , it means we already reassembled the HDLC packets , so we don ' t
need to use p_t38_conv
*/
p_t38_packet_conv = NULL ;
p_t38_conv = NULL ;
/* Use existing packet info if available */
2006-01-22 23:14:58 +00:00
p_t38_packet_conv = p_get_proto_data ( pinfo - > fd , proto_t38 ) ;
2006-01-15 15:01:14 +00:00
/* find the conversation used for Reassemble and Setup Info */
p_conv = find_conversation ( pinfo - > fd - > num , & pinfo - > net_src , & pinfo - > net_dst ,
pinfo - > ptype ,
pinfo - > srcport , pinfo - > destport , NO_ADDR_B | NO_PORT_B ) ;
/* create a conv if it doen't exist */
if ( ! p_conv ) {
p_conv = conversation_new ( pinfo - > fd - > num , & pinfo - > net_src , & pinfo - > net_dst ,
pinfo - > ptype , pinfo - > srcport , pinfo - > destport , NO_ADDR_B | NO_PORT_B ) ;
/* Set dissector */
2006-01-22 23:14:58 +00:00
conversation_set_dissector ( p_conv , t38_udp_handle ) ;
2006-01-15 15:01:14 +00:00
}
if ( ! p_t38_packet_conv ) {
p_t38_conv = conversation_get_proto_data ( p_conv , proto_t38 ) ;
/* create the conversation if it doen't exist */
if ( ! p_t38_conv ) {
p_t38_conv = se_alloc ( sizeof ( t38_conv ) ) ;
2006-01-22 23:14:58 +00:00
p_t38_conv - > setup_method [ 0 ] = ' \0 ' ;
p_t38_conv - > setup_frame_number = 0 ;
2006-01-15 15:01:14 +00:00
p_t38_conv - > src_t38_info . reass_ID = 0 ;
p_t38_conv - > src_t38_info . reass_start_seqnum = - 1 ;
p_t38_conv - > src_t38_info . reass_data_type = 0 ;
p_t38_conv - > src_t38_info . last_seqnum = - 1 ;
p_t38_conv - > src_t38_info . packet_lost = 0 ;
p_t38_conv - > src_t38_info . burst_lost = 0 ;
p_t38_conv - > src_t38_info . time_first_t4_data = 0 ;
p_t38_conv - > dst_t38_info . reass_ID = 0 ;
p_t38_conv - > dst_t38_info . reass_start_seqnum = - 1 ;
p_t38_conv - > dst_t38_info . reass_data_type = 0 ;
p_t38_conv - > dst_t38_info . last_seqnum = - 1 ;
p_t38_conv - > dst_t38_info . packet_lost = 0 ;
p_t38_conv - > dst_t38_info . burst_lost = 0 ;
p_t38_conv - > dst_t38_info . time_first_t4_data = 0 ;
conversation_add_proto_data ( p_conv , proto_t38 , p_t38_conv ) ;
}
/* copy the t38 conversation info to the packet t38 conversation */
p_t38_packet_conv = se_alloc ( sizeof ( t38_conv ) ) ;
strcpy ( p_t38_packet_conv - > setup_method , p_t38_conv - > setup_method ) ;
2006-01-22 23:14:58 +00:00
p_t38_packet_conv - > setup_frame_number = p_t38_conv - > setup_frame_number ;
2006-01-15 15:01:14 +00:00
memcpy ( & ( p_t38_packet_conv - > src_t38_info ) , & ( p_t38_conv - > src_t38_info ) , sizeof ( t38_conv_info ) ) ;
memcpy ( & ( p_t38_packet_conv - > dst_t38_info ) , & ( p_t38_conv - > dst_t38_info ) , sizeof ( t38_conv_info ) ) ;
p_add_proto_data ( pinfo - > fd , proto_t38 , p_t38_packet_conv ) ;
}
if ( ADDRESSES_EQUAL ( & p_conv - > key_ptr - > addr1 , & pinfo - > net_src ) ) {
p_t38_conv_info = & ( p_t38_conv - > src_t38_info ) ;
p_t38_packet_conv_info = & ( p_t38_packet_conv - > src_t38_info ) ;
} else {
p_t38_conv_info = & ( p_t38_conv - > dst_t38_info ) ;
p_t38_packet_conv_info = & ( p_t38_packet_conv - > dst_t38_info ) ;
}
2006-01-23 07:12:26 +00:00
/* update t38_info */
t38_info - > setup_frame_number = p_t38_packet_conv - > setup_frame_number ;
}
/* Entry point for dissection */
static void
dissect_t38_udp ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree )
{
guint8 octet1 ;
proto_item * it ;
proto_tree * tr ;
guint32 offset = 0 ;
2006-05-22 11:49:50 +00:00
asn_ctx_t asn_ctx ;
2006-01-23 07:12:26 +00:00
/*
* XXX - heuristic to check for misidentified packets .
*/
if ( dissect_possible_rtpv2_packets_as_rtp ) {
octet1 = tvb_get_guint8 ( tvb , offset ) ;
if ( RTP_VERSION ( octet1 ) = = 2 ) {
call_dissector ( rtp_handle , tvb , pinfo , tree ) ;
return ;
}
}
if ( check_col ( pinfo - > cinfo , COL_PROTOCOL ) ) {
col_set_str ( pinfo - > cinfo , COL_PROTOCOL , " T.38 " ) ;
}
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
col_clear ( pinfo - > cinfo , COL_INFO ) ;
}
primary_part = TRUE ;
/* This indicate the item number in the primary part of the T38 message, it is used for the reassemble of T30 packets */
Data_Field_item_num = 0 ;
it = proto_tree_add_protocol_format ( tree , proto_t38 , tvb , 0 , - 1 , " ITU-T Recommendation T.38 " ) ;
tr = proto_item_add_subtree ( it , ett_t38 ) ;
/* init tap and conv info */
init_t38_info_conv ( pinfo ) ;
2006-01-22 23:14:58 +00:00
/* Show Conversation setup info if exists*/
if ( global_t38_show_setup_info ) {
2006-01-15 15:01:14 +00:00
show_setup_info ( tvb , pinfo , tr , p_conv , p_t38_packet_conv ) ;
}
2004-01-09 23:24:54 +00:00
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
2004-01-26 22:52:22 +00:00
col_append_fstr ( pinfo - > cinfo , COL_INFO , " UDP: UDPTLPacket " ) ;
2004-01-09 23:24:54 +00:00
}
2006-05-22 11:49:50 +00:00
asn_ctx_init ( & asn_ctx , ASN_ENC_PER , TRUE , pinfo ) ;
offset = dissect_t38_UDPTLPacket ( tvb , offset , & asn_ctx , tr ) ;
2003-09-11 13:36:04 +00:00
2006-01-23 07:12:26 +00:00
if ( offset & 0x07 ) {
2004-01-26 22:52:22 +00:00
offset = ( offset & 0xfffffff8 ) + 8 ;
2004-01-26 22:16:43 +00:00
}
2006-01-23 07:12:26 +00:00
if ( tvb_length_remaining ( tvb , offset > > 3 ) > 0 ) {
if ( tr ) {
2004-01-26 22:16:43 +00:00
proto_tree_add_text ( tr , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) ,
" [MALFORMED PACKET or wrong preference settings] " ) ;
}
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
col_append_fstr ( pinfo - > cinfo , COL_INFO , " [Malformed?] " ) ;
}
}
2003-09-11 13:36:04 +00:00
}
static void
2004-01-26 22:52:22 +00:00
dissect_t38_tcp_pdu ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree )
2003-09-11 13:36:04 +00:00
{
2004-01-26 22:52:22 +00:00
proto_item * it ;
2003-09-11 13:36:04 +00:00
proto_tree * tr ;
2004-01-26 22:52:22 +00:00
guint32 offset = 0 ;
2004-01-26 22:16:43 +00:00
guint16 ifp_packet_number = 1 ;
2006-05-22 11:49:50 +00:00
asn_ctx_t asn_ctx ;
2005-10-04 14:41:39 +00:00
2003-09-11 13:36:04 +00:00
if ( check_col ( pinfo - > cinfo , COL_PROTOCOL ) ) {
col_set_str ( pinfo - > cinfo , COL_PROTOCOL , " T.38 " ) ;
}
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
col_clear ( pinfo - > cinfo , COL_INFO ) ;
}
2006-01-23 07:12:26 +00:00
primary_part = TRUE ;
/* This indicate the item number in the primary part of the T38 message, it is used for the reassemble of T30 packets */
Data_Field_item_num = 0 ;
it = proto_tree_add_protocol_format ( tree , proto_t38 , tvb , 0 , - 1 , " ITU-T Recommendation T.38 " ) ;
2003-09-11 13:36:04 +00:00
tr = proto_item_add_subtree ( it , ett_t38 ) ;
2006-01-23 07:12:26 +00:00
/* init tap and conv info */
init_t38_info_conv ( pinfo ) ;
/* Show Conversation setup info if exists*/
if ( global_t38_show_setup_info ) {
show_setup_info ( tvb , pinfo , tr , p_conv , p_t38_packet_conv ) ;
}
2004-01-09 23:24:54 +00:00
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
2004-01-26 22:52:22 +00:00
col_append_fstr ( pinfo - > cinfo , COL_INFO , " TCP: IFPPacket " ) ;
2004-01-09 23:24:54 +00:00
}
2004-01-26 22:16:43 +00:00
while ( tvb_length_remaining ( tvb , offset > > 3 ) > 0 )
{
2006-05-22 11:49:50 +00:00
asn_ctx_init ( & asn_ctx , ASN_ENC_PER , TRUE , pinfo ) ;
offset = dissect_t38_IFPPacket ( tvb , offset , & asn_ctx , tr ) ;
2004-01-26 22:16:43 +00:00
ifp_packet_number + + ;
if ( offset & 0x07 ) {
offset = ( offset & 0xfffffff8 ) + 8 ;
}
if ( tvb_length_remaining ( tvb , offset > > 3 ) > 0 ) {
if ( t38_tpkt_usage = = T38_TPKT_ALWAYS ) {
if ( tr ) {
proto_tree_add_text ( tr , tvb , offset , tvb_reported_length_remaining ( tvb , offset ) ,
" [MALFORMED PACKET or wrong preference settings] " ) ;
}
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
col_append_fstr ( pinfo - > cinfo , COL_INFO , " [Malformed?] " ) ;
}
break ;
}
else {
if ( check_col ( pinfo - > cinfo , COL_INFO ) ) {
col_append_fstr ( pinfo - > cinfo , COL_INFO , " IFPPacket#%u " , ifp_packet_number ) ;
}
}
}
}
2003-09-11 13:36:04 +00:00
}
static void
2004-01-26 22:52:22 +00:00
dissect_t38_tcp ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree )
2003-09-11 13:36:04 +00:00
{
2004-01-09 23:24:54 +00:00
primary_part = TRUE ;
2004-01-26 22:52:22 +00:00
if ( t38_tpkt_usage = = T38_TPKT_ALWAYS ) {
dissect_tpkt_encap ( tvb , pinfo , tree , t38_tpkt_reassembly , t38_tcp_pdu_handle ) ;
}
else if ( ( t38_tpkt_usage = = T38_TPKT_NEVER ) | | ( is_tpkt ( tvb , 1 ) = = - 1 ) ) {
dissect_t38_tcp_pdu ( tvb , pinfo , tree ) ;
}
else {
dissect_tpkt_encap ( tvb , pinfo , tree , t38_tpkt_reassembly , t38_tcp_pdu_handle ) ;
}
}
static void
dissect_t38 ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree )
{
2003-09-11 13:36:04 +00:00
if ( pinfo - > ipproto = = IP_PROTO_TCP )
{
2004-01-26 22:52:22 +00:00
dissect_t38_tcp ( tvb , pinfo , tree ) ;
2003-09-11 13:36:04 +00:00
}
else if ( pinfo - > ipproto = = IP_PROTO_UDP )
2004-01-09 23:24:54 +00:00
{
2003-09-11 13:36:04 +00:00
dissect_t38_udp ( tvb , pinfo , tree ) ;
}
}
2004-10-22 06:19:00 +00:00
/* Look for conversation info and display any setup info found */
2006-01-15 15:01:14 +00:00
void show_setup_info ( tvbuff_t * tvb , packet_info * pinfo , proto_tree * tree , conversation_t * p_conv , t38_conv * p_t38_conv )
2004-10-22 06:19:00 +00:00
{
2006-01-15 15:01:14 +00:00
proto_tree * t38_setup_tree ;
proto_item * ti ;
2004-10-22 06:19:00 +00:00
2006-01-15 15:01:14 +00:00
if ( ! p_t38_conv | | p_t38_conv - > setup_frame_number = = 0 ) {
/* there is no Setup info */
return ;
}
2004-10-22 06:19:00 +00:00
2006-01-15 15:01:14 +00:00
ti = proto_tree_add_string_format ( tree , hf_t38_setup , tvb , 0 , 0 ,
" " ,
" Stream setup by %s (frame %u) " ,
p_t38_conv - > setup_method ,
p_t38_conv - > setup_frame_number ) ;
PROTO_ITEM_SET_GENERATED ( ti ) ;
t38_setup_tree = proto_item_add_subtree ( ti , ett_t38_setup ) ;
if ( t38_setup_tree )
{
/* Add details into subtree */
proto_item * item = proto_tree_add_uint ( t38_setup_tree , hf_t38_setup_frame ,
tvb , 0 , 0 , p_t38_conv - > setup_frame_number ) ;
PROTO_ITEM_SET_GENERATED ( item ) ;
item = proto_tree_add_string ( t38_setup_tree , hf_t38_setup_method ,
tvb , 0 , 0 , p_t38_conv - > setup_method ) ;
PROTO_ITEM_SET_GENERATED ( item ) ;
}
2004-10-22 06:19:00 +00:00
}
2006-05-21 04:49:01 +00:00
/* Wireshark Protocol Registration */
2003-09-11 13:36:04 +00:00
void
proto_register_t38 ( void )
{
static hf_register_info hf [ ] =
{
{ & hf_t38_IFPPacket ,
{ " IFPPacket " , " t38.IFPPacket " , FT_NONE , BASE_NONE ,
NULL , 0 , " IFPPacket sequence " , HFILL } } ,
{ & hf_t38_Type_of_msg ,
{ " Type of msg " , " t38.Type_of_msg_type " , FT_UINT32 , BASE_DEC ,
VALS ( Type_of_msg_vals ) , 0 , " Type_of_msg choice " , HFILL } } ,
{ & hf_t38_t30_indicator ,
{ " T30 indicator " , " t38.t30_indicator " , FT_UINT32 , BASE_DEC ,
VALS ( t30_indicator_vals ) , 0 , " t30_indicator " , HFILL } } ,
{ & hf_t38_data ,
{ " data " , " t38.t38_data " , FT_UINT32 , BASE_DEC ,
2006-01-15 15:01:14 +00:00
VALS ( t30_data_vals ) , 0 , " data " , HFILL } } ,
2003-09-11 13:36:04 +00:00
{ & hf_t38_Data_Field ,
{ " Data Field " , " t38.Data_Field " , FT_NONE , BASE_NONE ,
NULL , 0 , " Data_Field sequence of " , HFILL } } ,
{ & hf_t38_Data_Field_item ,
{ " Data_Field_item " , " t38.Data_Field_item " , FT_NONE , BASE_NONE ,
NULL , 0 , " Data_Field_item sequence " , HFILL } } ,
{ & hf_t38_Data_Field_field_type ,
{ " Data_Field_field_type " , " t38.Data_Field_field_type " , FT_UINT32 , BASE_DEC ,
VALS ( Data_Field_field_type_vals ) , 0 , " Data_Field_field_type choice " , HFILL } } ,
{ & hf_t38_Data_Field_field_data ,
{ " Data_Field_field_data " , " t38.Data_Field_field_data " , FT_BYTES , BASE_HEX ,
NULL , 0 , " Data_Field_field_data octet string " , HFILL } } ,
{ & hf_t38_UDPTLPacket ,
{ " UDPTLPacket " , " t38.UDPTLPacket " , FT_NONE , BASE_NONE ,
NULL , 0 , " UDPTLPacket sequence " , HFILL } } ,
{ & hf_t38_seq_number ,
{ " Sequence number " , " t38.seq_number " , FT_UINT32 , BASE_DEC ,
NULL , 0 , " seq_number " , HFILL } } ,
{ & hf_t38_primary_ifp_packet ,
{ " Primary IFPPacket " , " t38.primary_ifp_packet " , FT_BYTES , BASE_HEX ,
NULL , 0 , " primary_ifp_packet octet string " , HFILL } } ,
{ & hf_t38_primary_ifp_packet_length ,
{ " primary_ifp_packet_length " , " t38.primary_ifp_packet_length " , FT_UINT32 , BASE_DEC ,
NULL , 0 , " primary_ifp_packet_length " , HFILL } } ,
{ & hf_t38_error_recovery ,
{ " Error recovery " , " t38.error_recovery " , FT_UINT32 , BASE_DEC ,
VALS ( error_recovery_vals ) , 0 , " error_recovery choice " , HFILL } } ,
{ & hf_t38_secondary_ifp_packets ,
{ " Secondary IFPPackets " , " t38.secondary_ifp_packets " , FT_NONE , BASE_NONE ,
NULL , 0 , " secondary_ifp_packets sequence of " , HFILL } } ,
{ & hf_t38_secondary_ifp_packets_item ,
{ " Secondary IFPPackets item " , " t38.secondary_ifp_packets_item " , FT_BYTES , BASE_HEX ,
NULL , 0 , " secondary_ifp_packets_item octet string " , HFILL } } ,
{ & hf_t38_secondary_ifp_packets_item_length ,
{ " secondary_ifp_packets_item_length " , " t38.secondary_ifp_packets_item_length " , FT_UINT32 , BASE_DEC ,
NULL , 0 , " secondary_ifp_packets_item_length " , HFILL } } ,
{ & hf_t38_fec_info ,
{ " Fec info " , " t38.fec_info " , FT_NONE , BASE_NONE ,
NULL , 0 , " fec_info sequence " , HFILL } } ,
{ & hf_t38_fec_npackets ,
{ " Fec npackets " , " h245.fec_npackets " , FT_INT32 , BASE_DEC ,
NULL , 0 , " fec_npackets value " , HFILL } } ,
{ & hf_t38_fec_data ,
{ " Fec data " , " t38.fec_data " , FT_NONE , BASE_NONE ,
NULL , 0 , " fec_data sequence of " , HFILL } } ,
{ & hf_t38_fec_data_item ,
{ " t38_fec_data_item " , " t38.t38_fec_data_item " , FT_BYTES , BASE_HEX ,
NULL , 0 , " t38_fec_data_item octet string " , HFILL } } ,
2006-01-15 15:01:14 +00:00
{ & hf_t38_setup ,
{ " Stream setup " , " t38.setup " , FT_STRING , BASE_NONE ,
NULL , 0x0 , " Stream setup, method and frame number " , HFILL } } ,
{ & hf_t38_setup_frame ,
2004-10-22 06:19:00 +00:00
{ " Stream frame " , " t38.setup-frame " , FT_FRAMENUM , BASE_NONE ,
NULL , 0x0 , " Frame that set up this stream " , HFILL } } ,
{ & hf_t38_setup_method ,
{ " Stream Method " , " t38.setup-method " , FT_STRING , BASE_NONE ,
NULL , 0x0 , " Method used to set up this stream " , HFILL } } ,
2006-01-15 15:01:14 +00:00
{ & hf_data_fragments ,
{ " Message fragments " , " data.fragments " ,
FT_NONE , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment ,
{ " Message fragment " , " data.fragment " ,
FT_FRAMENUM , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment_overlap ,
{ " Message fragment overlap " , " data.fragment.overlap " ,
FT_BOOLEAN , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment_overlap_conflicts ,
{ " Message fragment overlapping with conflicting data " ,
" data.fragment.overlap.conflicts " ,
FT_BOOLEAN , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment_multiple_tails ,
{ " Message has multiple tail fragments " ,
" data.fragment.multiple_tails " ,
FT_BOOLEAN , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment_too_long_fragment ,
{ " Message fragment too long " , " data.fragment.too_long_fragment " ,
FT_BOOLEAN , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_fragment_error ,
{ " Message defragmentation error " , " data.fragment.error " ,
FT_FRAMENUM , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
{ & hf_data_reassembled_in ,
{ " Reassembled in " , " data.reassembled.in " ,
FT_FRAMENUM , BASE_NONE , NULL , 0x00 , NULL , HFILL } } ,
} ;
2003-09-11 13:36:04 +00:00
static gint * ett [ ] =
{
& ett_t38 ,
2004-01-26 22:52:22 +00:00
& ett_t38_IFPPacket ,
& ett_t38_Type_of_msg ,
& ett_t38_t30_indicator ,
& ett_t38_data ,
& ett_t38_Data_Field ,
& ett_t38_Data_Field_item ,
& ett_t38_Data_Field_field_type ,
& ett_t38_UDPTLPacket ,
& ett_t38_error_recovery ,
& ett_t38_secondary_ifp_packets ,
& ett_t38_fec_info ,
& ett_t38_fec_data ,
2004-10-22 06:19:00 +00:00
& ett_t38_setup ,
2006-01-15 15:01:14 +00:00
& ett_data_fragment ,
& ett_data_fragments
2003-09-11 13:36:04 +00:00
} ;
2006-01-15 15:01:14 +00:00
static hf_register_info hf_t30 [ ] =
{
{ & hf_t30_Address ,
{ " Address " , " t30.Address " , FT_UINT8 , BASE_HEX ,
NULL , 0 , " Address Field " , HFILL } } ,
{ & hf_t30_Control ,
{ " Control " , " t30.Control " , FT_UINT8 , BASE_HEX ,
VALS ( t30_control_vals ) , 0 , " Address Field " , HFILL } } ,
{ & hf_t30_Facsimile_Control ,
{ " Facsimile Control " , " t30.FacsimileControl " , FT_UINT8 , BASE_DEC ,
VALS ( t30_facsimile_control_field_vals ) , 0 , " Facsimile Control " , HFILL } } ,
{ & hf_t30_fif_sm ,
{ " Store and forward Internet fax- Simple mode (ITU-T T.37) " , " t30.fif.sm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_rtif ,
{ " Real-time Internet fax (ITU T T.38) " , " t30.fif.rtif " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_3gmn ,
{ " 3rd Generation Mobile Network " , " t30.fif.3gmn " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_v8c ,
{ " V.8 capabilities " , " t30.fif.v8c " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_op ,
{ " Octets preferred " , " t30.fif.op " , FT_BOOLEAN , 8 ,
TFS ( & t30_octets_preferred_value ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_rtfc ,
{ " Ready to transmit a facsimile document (polling) " , " t30.fif.rtfc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_rfo ,
{ " Receiver fax operation " , " t30.fif.rfo " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_dsr ,
{ " Data signalling rate " , " t30.fif.dsr " , FT_UINT8 , BASE_HEX ,
VALS ( t30_data_signalling_rate_vals ) , 0x3C , " " , HFILL } } ,
{ & hf_t30_fif_dsr_dcs ,
{ " Data signalling rate " , " t30.fif.dsr_dcs " , FT_UINT8 , BASE_HEX ,
VALS ( t30_data_signalling_rate_dcs_vals ) , 0x3C , " " , HFILL } } ,
{ & hf_t30_fif_res ,
{ " R8x7.7 lines/mm and/or 200x200 pels/25.4 mm " , " t30.fif.res " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_tdcc ,
{ " Two dimensional coding capability " , " t30.fif.tdcc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x01 , " " , HFILL } } ,
{ & hf_t30_fif_rwc ,
{ " Recording width capabilities " , " t30.fif.rwc " , FT_UINT8 , BASE_HEX ,
VALS ( t30_recording_width_capabilities_vals ) , 0xC0 , " " , HFILL } } ,
{ & hf_t30_fif_rw_dcs ,
{ " Recording width " , " t30.fif.rw_dcs " , FT_UINT8 , BASE_HEX ,
VALS ( t30_recording_width_dcs_vals ) , 0xC0 , " " , HFILL } } ,
{ & hf_t30_fif_rlc ,
{ " Recording length capability " , " t30.fif.rlc " , FT_UINT8 , BASE_HEX ,
VALS ( t30_recording_length_capability_vals ) , 0x30 , " " , HFILL } } ,
{ & hf_t30_fif_rl_dcs ,
{ " Recording length capability " , " t30.fif.rl_dcs " , FT_UINT8 , BASE_HEX ,
VALS ( t30_recording_length_dcs_vals ) , 0x30 , " " , HFILL } } ,
{ & hf_t30_fif_msltcr ,
{ " Minimum scan line time capability at the receiver " , " t30.fif.msltcr " , FT_UINT8 , BASE_HEX ,
VALS ( t30_minimum_scan_line_time_rec_vals ) , 0x0E , " " , HFILL } } ,
{ & hf_t30_fif_mslt_dcs ,
{ " Minimum scan line time " , " t30.fif.mslt_dcs " , FT_UINT8 , BASE_HEX ,
VALS ( t30_minimum_scan_line_time_dcs_vals ) , 0x0E , " " , HFILL } } ,
{ & hf_t30_fif_ext ,
{ " Extension indicator " , " t30.fif.ext " , FT_BOOLEAN , 8 ,
TFS ( & t30_extension_ind_value ) , 0x01 , " " , HFILL } } ,
{ & hf_t30_fif_cm ,
{ " Compress/Uncompress mode " , " t30.fif.cm " , FT_BOOLEAN , 8 ,
TFS ( & t30_compress_value ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_ecm ,
{ " Error correction mode " , " t30.fif.ecm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_fs_dcs ,
{ " Frame size " , " t30.fif.fs_dcm " , FT_BOOLEAN , 8 ,
TFS ( & t30_frame_size_dcs_value ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_t6 ,
{ " T.6 coding capability " , " t30.fif.t6 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_fvc ,
{ " Field valid capability " , " t30.fif.fvc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_mspc ,
{ " Multiple selective polling capability " , " t30.fif.mspc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_ps ,
{ " Polled Subaddress " , " t30.fif.ps " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_t43 ,
{ " T.43 coding " , " t30.fif.t43 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_pi ,
{ " Plane interleave " , " t30.fif.pi " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_vc32k ,
{ " Voice coding with 32k ADPCM (ITU T G.726) " , " t30.fif.vc32k " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_r8x15 ,
{ " R8x15.4 lines/mm " , " t30.fif.r8x15 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_300x300 ,
{ " 300x300 pels/25.4 mm " , " t30.fif.300x300 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_r16x15 ,
{ " R16x15.4 lines/mm and/or 400x400 pels/25.4 mm " , " t30.fif.r16x15 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_ibrp ,
{ " Inch based resolution preferred " , " t30.fif.ibrp " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_mbrp ,
{ " Metric based resolution preferred " , " t30.fif.mbrp " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_msltchr ,
{ " Minimum scan line time capability for higher resolutions " , " t30.fif.msltchr " , FT_BOOLEAN , 8 ,
TFS ( & t30_minimum_scan_value ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_rts ,
{ " Resolution type selection " , " t30.fif.rts " , FT_BOOLEAN , 8 ,
TFS ( & t30_res_type_sel_value ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_sp ,
{ " Selective polling " , " t30.fif.sp " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_sc ,
{ " Subaddressing capability " , " t30.fif.sc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_passw ,
{ " Password " , " t30.fif.passw " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_sit ,
{ " Sender Identification transmission " , " t30.fif.sit " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_rttd ,
{ " Ready to transmit a data file (polling) " , " t30.fif.rttd " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_bft ,
{ " Binary File Transfer (BFT) " , " t30.fif.bft " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_dtm ,
{ " Document Transfer Mode (DTM) " , " t30.fif.dtm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_edi ,
{ " Electronic Data Interchange (EDI) " , " t30.fif.edi " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_btm ,
{ " Basic Transfer Mode (BTM) " , " t30.fif.btm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_rttcmmd ,
{ " Ready to transmit a character or mixed mode document (polling) " , " t30.fif.rttcmmd " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_chrm ,
{ " Character mode " , " t30.fif.cm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_mm ,
{ " Mixed mode (Annex E/T.4) " , " t30.fif.mm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_pm26 ,
{ " Processable mode 26 (ITU T T.505) " , " t30.fif.pm26 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_dnc ,
{ " Digital network capability " , " t30.fif.dnc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_do ,
{ " Duplex operation " , " t30.fif.do " , FT_BOOLEAN , 8 ,
TFS ( & t30_duplex_operation_value ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_jpeg ,
{ " JPEG coding " , " t30.fif.jpeg " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_fcm ,
{ " Full colour mode " , " t30.fif.fcm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_pht ,
{ " Preferred Huffman tables " , " t30.fif.pht " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_12c ,
{ " 12 bits/pel component " , " t30.fif.12c " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_ns ,
{ " No subsampling (1:1:1) " , " t30.fif.ns " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_ci ,
{ " Custom illuminant " , " t30.fif.ci " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_cgr ,
{ " Custom gamut range " , " t30.fif.cgr " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_nalet ,
{ " North American Letter (215.9 x 279.4 mm) capability " , " t30.fif.nalet " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_naleg ,
{ " North American Legal (215.9 x 355.6 mm) capability " , " t30.fif.naleg " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_spscb ,
{ " Single-progression sequential coding (ITU-T T.85) basic capability " , " t30.fif.spscb " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_spsco ,
{ " Single-progression sequential coding (ITU-T T.85) optional L0 capability " , " t30.fif.spsco " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_hkm ,
{ " HKM key management capability " , " t30.fif.hkm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_rsa ,
{ " RSA key management capability " , " t30.fif.rsa " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_oc ,
{ " Override capability " , " t30.fif.oc " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_hfx40 ,
{ " HFX40 cipher capability " , " t30.fif.hfx40 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_acn2c ,
{ " Alternative cipher number 2 capability " , " t30.fif.acn2c " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_acn3c ,
{ " Alternative cipher number 3 capability " , " t30.fif.acn3c " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_hfx40i ,
{ " HFX40-I hashing capability " , " t30.fif.hfx40i " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_ahsn2 ,
{ " Alternative hashing system number 2 capability " , " t30.fif.ahsn2 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_ahsn3 ,
{ " Alternative hashing system number 3 capability " , " t30.fif.ahsn3 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_t441 ,
{ " T.44 (Mixed Raster Content) " , " t30.fif.t441 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_t442 ,
{ " T.44 (Mixed Raster Content) " , " t30.fif.t442 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_t443 ,
{ " T.44 (Mixed Raster Content) " , " t30.fif.t443 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_plmss ,
{ " Page length maximum strip size for T.44 (Mixed Raster Content) " , " t30.fif.plmss " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_cg300 ,
{ " Colour/gray-scale 300 pels/25.4 mm x 300 lines/25.4 mm or 400 pels/25.4 mm x 400 lines/25.4 mm resolution " , " t30.fif.cg300 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_100x100cg ,
{ " 100 pels/25.4 mm x 100 lines/25.4 mm for colour/gray scale " , " t30.fif.100x100cg " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_spcbft ,
{ " Simple Phase C BFT Negotiations capability " , " t30.fif.spcbft " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_ebft ,
{ " Extended BFT Negotiations capability " , " t30.fif.ebft " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_isp ,
{ " Internet Selective Polling Address (ISP) " , " t30.fif.isp " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_ira ,
{ " Internet Routing Address (IRA) " , " t30.fif.ira " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_600x600 ,
{ " 600 pels/25.4 mm x 600 lines/25.4 mm " , " t30.fif.600x600 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_1200x1200 ,
{ " 1200 pels/25.4 mm x 1200 lines/25.4 mm " , " t30.fif.1200x1200 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_300x600 ,
{ " 300 pels/25.4 mm x 600 lines/25.4 mm " , " t30.fif.300x600 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_400x800 ,
{ " 400 pels/25.4 mm x 800 lines/25.4 mm " , " t30.fif.400x800 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_600x1200 ,
{ " 600 pels/25.4 mm x 1200 lines/25.4 mm " , " t30.fif.600x1200 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x08 , " " , HFILL } } ,
{ & hf_t30_fif_cg600x600 ,
{ " Colour/gray scale 600 pels/25.4 mm x 600 lines/25.4 mm resolution " , " t30.fif.cg600x600 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x04 , " " , HFILL } } ,
{ & hf_t30_fif_cg1200x1200 ,
{ " Colour/gray scale 1200 pels/25.4 mm x 1200 lines/25.4 mm resolution " , " t30.fif.cg1200x1200 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x02 , " " , HFILL } } ,
{ & hf_t30_fif_dspcam ,
{ " Double sided printing capability (alternate mode) " , " t30.fif.dspcam " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x80 , " " , HFILL } } ,
{ & hf_t30_fif_dspccm ,
{ " Double sided printing capability (continuous mode) " , " t30.fif.dspccm " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x40 , " " , HFILL } } ,
{ & hf_t30_fif_bwmrcp ,
{ " Black and white mixed raster content profile (MRCbw) " , " t30.fif.bwmrcp " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x20 , " " , HFILL } } ,
{ & hf_t30_fif_t45 ,
{ " T.45 (run length colour encoding) " , " t30.fif.t45 " , FT_BOOLEAN , 8 ,
TFS ( & flags_set_truth ) , 0x10 , " " , HFILL } } ,
{ & hf_t30_fif_sdmc ,
{ " SharedDataMemory capacity " , " t30.fif.sdmc " , FT_UINT8 , BASE_HEX ,
VALS ( t30_SharedDataMemory_capacity_vals ) , 0x0C , " " , HFILL } } ,
{ & hf_t30_fif_number ,
{ " Number " , " t30.fif.number " , FT_STRING , BASE_NONE , NULL , 0x0 ,
" " , HFILL } } ,
{ & hf_t30_fif_country_code ,
{ " ITU-T Country code " , " t30.fif.country_code " , FT_UINT8 , BASE_DEC ,
NULL , 0 , " ITU-T Country code " , HFILL } } ,
{ & hf_t30_fif_non_stand_bytes ,
{ " Non-standard capabilities " , " t30.fif.non_standard_cap " , FT_BYTES , BASE_HEX ,
NULL , 0 , " Non-standard capabilities " , HFILL } } ,
{ & hf_t30_t4_frame_num ,
{ " T.4 Frame number " , " t30.t4.frame_num " , FT_UINT8 , BASE_DEC ,
NULL , 0 , " T.4 Frame number " , HFILL } } ,
{ & hf_t30_t4_data ,
{ " T.4 Facsimile data field " , " t30.t4.data " , FT_BYTES , BASE_HEX ,
NULL , 0 , " T.4 Facsimile data field " , HFILL } } ,
{ & hf_t30_partial_page_fcf2 ,
{ " Post-message command " , " t30.pps.fcf2 " , FT_UINT8 , BASE_DEC ,
VALS ( t30_partial_page_fcf2_vals ) , 0 , " Post-message command " , HFILL } } ,
{ & hf_t30_partial_page_i1 ,
{ " Page counter " , " t30.t4.page_count " , FT_UINT8 , BASE_DEC ,
NULL , 0 , " Page counter " , HFILL } } ,
{ & hf_t30_partial_page_i2 ,
{ " Block counter " , " t30.t4.block_count " , FT_UINT8 , BASE_DEC ,
NULL , 0 , " Block counter " , HFILL } } ,
{ & hf_t30_partial_page_i3 ,
{ " Frame counter " , " t30.t4.frame_count " , FT_UINT8 , BASE_DEC ,
NULL , 0 , " Frame counter " , HFILL } } ,
} ;
static gint * t30_ett [ ] =
{
& ett_t30 ,
& ett_t30_fif ,
} ;
2004-01-09 23:24:54 +00:00
module_t * t38_module ;
2003-09-11 13:36:04 +00:00
2004-10-22 06:19:00 +00:00
proto_t38 = proto_register_protocol ( " T.38 " , " T.38 " , " t38 " ) ;
2003-09-11 13:36:04 +00:00
proto_register_field_array ( proto_t38 , hf , array_length ( hf ) ) ;
proto_register_subtree_array ( ett , array_length ( ett ) ) ;
register_dissector ( " t38 " , dissect_t38 , proto_t38 ) ;
2004-01-09 23:24:54 +00:00
2006-01-15 15:01:14 +00:00
/* Init reassemble tables for HDLC */
register_init_routine ( t38_defragment_init ) ;
2005-10-03 05:55:26 +00:00
t38_tap = register_tap ( " t38 " ) ;
2004-01-09 23:24:54 +00:00
t38_module = prefs_register_protocol ( proto_t38 , proto_reg_handoff_t38 ) ;
prefs_register_bool_preference ( t38_module , " use_pre_corrigendum_asn1_specification " ,
" Use the Pre-Corrigendum ASN.1 specification " ,
" Whether the T.38 dissector should decode using the Pre-Corrigendum T.38 "
" ASN.1 specification (1998). " ,
& use_pre_corrigendum_asn1_specification ) ;
prefs_register_bool_preference ( t38_module , " dissect_possible_rtpv2_packets_as_rtp " ,
" Dissect possible RTP version 2 packets with RTP dissector " ,
" Whether a UDP packet that looks like RTP version 2 packet will "
" be dissected as RTP packet or T.38 packet. If enabled there is a risk that T.38 UDPTL "
" packets with sequence number higher than 32767 may be dissected as RTP. " ,
& dissect_possible_rtpv2_packets_as_rtp ) ;
prefs_register_uint_preference ( t38_module , " tcp.port " ,
" T.38 TCP Port " ,
" Set the TCP port for T.38 messages " ,
10 , & global_t38_tcp_port ) ;
prefs_register_uint_preference ( t38_module , " udp.port " ,
" T.38 UDP Port " ,
" Set the UDP port for T.38 messages " ,
10 , & global_t38_udp_port ) ;
2004-01-26 22:16:43 +00:00
prefs_register_bool_preference ( t38_module , " reassembly " ,
" Reassemble T.38 PDUs over TPKT over TCP " ,
" Whether the dissector should reassemble T.38 PDUs spanning multiple TCP segments "
2004-08-21 09:02:52 +00:00
" when TPKT is used over TCP. "
" To use this option, you must also enable \" Allow subdissectors to reassemble TCP streams \" in the TCP protocol settings. " ,
2004-01-26 22:16:43 +00:00
& t38_tpkt_reassembly ) ;
prefs_register_enum_preference ( t38_module , " tpkt_usage " ,
" TPKT used over TCP " ,
" Whether T.38 is used with TPKT for TCP " ,
( gint * ) & t38_tpkt_usage , t38_tpkt_options , FALSE ) ;
2004-10-22 06:19:00 +00:00
prefs_register_bool_preference ( t38_module , " show_setup_info " ,
" Show stream setup information " ,
" Where available, show which protocol and frame caused "
" this T.38 stream to be created " ,
& global_t38_show_setup_info ) ;
2006-01-15 15:01:14 +00:00
/* T30 */
proto_t30 = proto_register_protocol ( " T.30 " , " T.30 " , " t30 " ) ;
proto_register_field_array ( proto_t30 , hf_t30 , array_length ( hf_t30 ) ) ;
proto_register_subtree_array ( t30_ett , array_length ( t30_ett ) ) ;
2003-09-11 13:36:04 +00:00
}
void
proto_reg_handoff_t38 ( void )
{
2004-01-09 23:24:54 +00:00
static int t38_prefs_initialized = FALSE ;
if ( ! t38_prefs_initialized ) {
2004-01-26 22:52:22 +00:00
t38_udp_handle = create_dissector_handle ( dissect_t38_udp , proto_t38 ) ;
t38_tcp_handle = create_dissector_handle ( dissect_t38_tcp , proto_t38 ) ;
t38_tcp_pdu_handle = create_dissector_handle ( dissect_t38_tcp_pdu , proto_t38 ) ;
2004-01-09 23:24:54 +00:00
t38_prefs_initialized = TRUE ;
}
else {
2004-01-26 22:52:22 +00:00
dissector_delete ( " tcp.port " , tcp_port , t38_tcp_handle ) ;
dissector_delete ( " udp.port " , udp_port , t38_udp_handle ) ;
2004-01-09 23:24:54 +00:00
}
tcp_port = global_t38_tcp_port ;
udp_port = global_t38_udp_port ;
2004-01-26 22:52:22 +00:00
dissector_add ( " tcp.port " , tcp_port , t38_tcp_handle ) ;
dissector_add ( " udp.port " , udp_port , t38_udp_handle ) ;
2004-01-09 23:24:54 +00:00
rtp_handle = find_dissector ( " rtp " ) ;
2003-09-11 13:36:04 +00:00
}
2006-01-15 15:01:14 +00:00