From Mike Morrin:
[obnox/wireshark/wip.git] / asn1 / sabp / packet-sabp-template.c
1 /* packet-sbap.c
2  * Routines for UTRAN Iu-BC Interface: Service Area Broadcast Protocol (SBAP) packet dissection
3  * Copyright 2007, Tomas Kukosa <tomas.kukosa@siemens.com>
4  *
5  * $Id$
6  *
7  * Wireshark - Network traffic analyzer
8  * By Gerald Combs <gerald@wireshark.org>
9  * Copyright 1998 Gerald Combs
10  *
11  * This program is free software; you can redistribute it and/or
12  * modify it under the terms of the GNU General Public License
13  * as published by the Free Software Foundation; either version 2
14  * of the License, or (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
19  * GNU General Public License for more details.
20  *
21  * You should have received a copy of the GNU General Public License
22  * along with this program; if not, write to the Free Software
23  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
24  *
25  * Ref: 3GPP TS 25.419 version  V9.0.0 (2009-12)
26  */
27
28 #ifdef HAVE_CONFIG_H
29 # include "config.h"
30 #endif
31
32 #include <glib.h>
33 #include <epan/packet.h>
34
35 #include <epan/asn1.h>
36
37 #include "packet-tcp.h"
38 #include "packet-per.h"
39 #include "packet-e212.h"
40 #include "packet-gsm_map.h"
41 #include "packet-gsm_sms.h"
42 #include <epan/sctpppids.h>
43 #include "packet-cell_broadcast.h"
44
45 #define PNAME  "UTRAN IuBC interface SABP signaling"
46 #define PSNAME "SABP"
47 #define PFNAME "sabp"
48
49 #include "packet-sabp-val.h"
50
51 /* Initialize the protocol and registered fields */
52 static int proto_sabp = -1;
53
54 static int hf_sabp_no_of_pages = -1;
55 #include "packet-sabp-hf.c"
56
57 /* Initialize the subtree pointers */
58 static int ett_sabp = -1;
59 static int ett_sabp_e212 = -1;
60 static int ett_sabp_cbs_data_coding = -1;
61 static int ett_sabp_bcast_msg = -1;
62 static int ett_sabp_cbs_serial_number = -1;
63 static int ett_sabp_cbs_new_serial_number = -1;
64
65 #include "packet-sabp-ett.c"
66
67 /* Global variables */
68 static guint32 ProcedureCode;
69 static guint32 ProtocolIE_ID;
70 static guint32 ProtocolExtensionID;
71 static guint8 sms_encoding;
72
73 /* desegmentation of sabp over TCP */
74 static gboolean gbl_sabp_desegment = TRUE;
75
76 /* Dissector tables */
77 static dissector_table_t sabp_ies_dissector_table;
78 static dissector_table_t sabp_extension_dissector_table;
79 static dissector_table_t sabp_proc_imsg_dissector_table;
80 static dissector_table_t sabp_proc_sout_dissector_table;
81 static dissector_table_t sabp_proc_uout_dissector_table;
82
83 static int dissect_ProtocolIEFieldValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree);
84 static int dissect_ProtocolExtensionFieldExtensionValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree);
85 static int dissect_InitiatingMessageValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree);
86 static int dissect_SuccessfulOutcomeValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree);
87 static int dissect_UnsuccessfulOutcomeValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree);
88
89 #include "packet-sabp-fn.c"
90
91 static int dissect_ProtocolIEFieldValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
92 {
93   return (dissector_try_uint(sabp_ies_dissector_table, ProtocolIE_ID, tvb, pinfo, tree)) ? tvb_length(tvb) : 0;
94 }
95
96 static int dissect_ProtocolExtensionFieldExtensionValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
97 {
98   return (dissector_try_uint(sabp_extension_dissector_table, ProtocolExtensionID, tvb, pinfo, tree)) ? tvb_length(tvb) : 0;
99 }
100
101 static int dissect_InitiatingMessageValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
102 {
103   return (dissector_try_uint(sabp_proc_imsg_dissector_table, ProcedureCode, tvb, pinfo, tree)) ? tvb_length(tvb) : 0;
104 }
105
106 static int dissect_SuccessfulOutcomeValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
107 {
108   return (dissector_try_uint(sabp_proc_sout_dissector_table, ProcedureCode, tvb, pinfo, tree)) ? tvb_length(tvb) : 0;
109 }
110
111 static int dissect_UnsuccessfulOutcomeValue(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
112 {
113   return (dissector_try_uint(sabp_proc_uout_dissector_table, ProcedureCode, tvb, pinfo, tree)) ? tvb_length(tvb) : 0;
114 }
115
116 static guint
117 get_sabp_pdu_len(packet_info *pinfo _U_, tvbuff_t *tvb, int offset)
118 {
119         guint32 type_length;
120         int bit_offset;
121         asn1_ctx_t asn1_ctx;
122         asn1_ctx_init(&asn1_ctx, ASN1_ENC_PER, TRUE, pinfo);
123
124         /* Length should be in the 3:d octet */
125         offset = offset + 3;
126
127         bit_offset = offset<<3;
128         /* Get the length of the sabp packet. offset in bits  */
129         dissect_per_length_determinant(tvb, bit_offset, &asn1_ctx, NULL, -1, &type_length);
130
131         /*
132          * Return the length of the PDU
133          * which is 3 + the length of the length, we only care about length up to 16K
134          * ("n" less than 128) a single octet containing "n" with bit 8 set to zero;
135          * ("n" less than 16K) two octets containing "n" with bit 8 of the first octet set to 1 and bit 7 set to zero;
136          */
137         if (type_length < 128)
138                 return type_length+4;
139
140         return type_length+5;
141 }
142
143
144 static void
145 dissect_sabp(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
146 {
147         proto_item      *sabp_item = NULL;
148         proto_tree      *sabp_tree = NULL;
149
150         /* make entry in the Protocol column on summary display */
151         col_set_str(pinfo->cinfo, COL_PROTOCOL, PSNAME);
152
153         /* create the sbap protocol tree */
154         sabp_item = proto_tree_add_item(tree, proto_sabp, tvb, 0, -1, ENC_NA);
155         sabp_tree = proto_item_add_subtree(sabp_item, ett_sabp);
156
157         dissect_SABP_PDU_PDU(tvb, pinfo, sabp_tree);
158 }
159
160 /* Note a little bit of a hack assumes length max takes two bytes and that the length starts at byte 4 */
161 static void
162 dissect_sabp_tcp(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree)
163 {
164         tcp_dissect_pdus(tvb, pinfo, tree, gbl_sabp_desegment, 5,
165                                          get_sabp_pdu_len, dissect_sabp);
166 }
167
168 /*--- proto_register_sbap -------------------------------------------*/
169 void proto_register_sabp(void) {
170
171   /* List of fields */
172
173   static hf_register_info hf[] = {
174     { &hf_sabp_no_of_pages,
175       { "Number-of-Pages", "sabp.no_of_pages",
176         FT_UINT8, BASE_DEC, NULL, 0,
177         NULL, HFILL }},
178
179 #include "packet-sabp-hfarr.c"
180   };
181
182   /* List of subtrees */
183   static gint *ett[] = {
184                   &ett_sabp,
185                   &ett_sabp_e212,
186                   &ett_sabp_cbs_data_coding,
187                   &ett_sabp_bcast_msg,
188           &ett_sabp_cbs_serial_number,
189           &ett_sabp_cbs_new_serial_number,
190 #include "packet-sabp-ettarr.c"
191   };
192
193
194   /* Register protocol */
195   proto_sabp = proto_register_protocol(PNAME, PSNAME, PFNAME);
196   /* Register fields and subtrees */
197   proto_register_field_array(proto_sabp, hf, array_length(hf));
198   proto_register_subtree_array(ett, array_length(ett));
199
200   /* Register dissector */
201   register_dissector("sabp", dissect_sabp, proto_sabp);
202   register_dissector("sabp.tcp", dissect_sabp_tcp, proto_sabp);
203
204   /* Register dissector tables */
205   sabp_ies_dissector_table = register_dissector_table("sabp.ies", "SABP-PROTOCOL-IES", FT_UINT32, BASE_DEC);
206   sabp_extension_dissector_table = register_dissector_table("sabp.extension", "SABP-PROTOCOL-EXTENSION", FT_UINT32, BASE_DEC);
207   sabp_proc_imsg_dissector_table = register_dissector_table("sabp.proc.imsg", "SABP-ELEMENTARY-PROCEDURE InitiatingMessage", FT_UINT32, BASE_DEC);
208   sabp_proc_sout_dissector_table = register_dissector_table("sabp.proc.sout", "SABP-ELEMENTARY-PROCEDURE SuccessfulOutcome", FT_UINT32, BASE_DEC);
209   sabp_proc_uout_dissector_table = register_dissector_table("sabp.proc.uout", "SABP-ELEMENTARY-PROCEDURE UnsuccessfulOutcome", FT_UINT32, BASE_DEC);
210
211 }
212
213
214 /*--- proto_reg_handoff_sbap ---------------------------------------*/
215 void
216 proto_reg_handoff_sabp(void)
217 {
218   dissector_handle_t sabp_handle;
219   dissector_handle_t sabp_tcp_handle;
220
221   sabp_handle = find_dissector("sabp");
222   sabp_tcp_handle = find_dissector("sabp.tcp");
223   dissector_add_uint("udp.port", 3452, sabp_handle);
224   dissector_add_uint("tcp.port", 3452, sabp_tcp_handle);
225   dissector_add_uint("sctp.ppi", SABP_PAYLOAD_PROTOCOL_ID,   sabp_handle);
226
227 #include "packet-sabp-dis-tab.c"
228
229 }
230
231