Put "WTP" into the Info column for segmented invoke/result PDUs on which
[obnox/wireshark/wip.git] / packet-iua.c
1 /* packet-iua.c
2  * Routines for ISDN Q.921-User Adaptation Layer dissection
3  *
4  * It is hopefully (needs testing) compilant to
5  *   http://www.ietf.org/rfc/rfc3057.txt
6  *   http://www.ietf.org/internet-drafts/draft-ietf-sigtran-iua-imp-guide-01.txt
7  * To do: - provide better handling of length parameters
8  *
9  * Copyright 2002, Michael Tuexen <tuexen [AT] fh-muenster.de>
10  *
11  * $Id: packet-iua.c,v 1.21 2003/04/22 13:47:37 tuexen Exp $
12  *
13  * Ethereal - Network traffic analyzer
14  * By Gerald Combs <gerald@ethereal.com>
15  * Copyright 1998 Gerald Combs
16  *
17  * Copied from README.developer
18  *
19  * This program is free software; you can redistribute it and/or
20  * modify it under the terms of the GNU General Public License
21  * as published by the Free Software Foundation; either version 2
22  * of the License, or (at your option) any later version.
23  *
24  * This program is distributed in the hope that it will be useful,
25  * but WITHOUT ANY WARRANTY; without even the implied warranty of
26  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
27  * GNU General Public License for more details.
28  *
29  * You should have received a copy of the GNU General Public License
30  * along with this program; if not, write to the Free Software
31  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
32  */
33
34 #ifdef HAVE_CONFIG_H
35 # include "config.h"
36 #endif
37
38 #include <epan/packet.h>
39 #include "prefs.h"
40 #include "sctpppids.h"
41
42 #define NETWORK_BYTE_ORDER          FALSE
43
44 static module_t *iua_module;
45
46 /* Initialize the protocol and registered fields */
47 static int proto_iua                = -1;
48 static int hf_int_interface_id      = -1;
49 static int hf_text_interface_id     = -1;
50 static int hf_info_string           = -1;
51 static int hf_dlci_zero_bit         = -1;
52 static int hf_dlci_spare_bit        = -1;
53 static int hf_dlci_sapi             = -1;
54 static int hf_dlci_one_bit          = -1;
55 static int hf_dlci_tei              = -1;
56 static int hf_dlci_spare            = -1;
57 static int hf_diag_info             = -1;
58 static int hf_interface_range_start = -1;
59 static int hf_interface_range_end   = -1;
60 static int hf_heartbeat_data        = -1;
61 static int hf_asp_reason            = -1;
62 static int hf_traffic_mode_type     = -1;
63 static int hf_error_code            = -1;
64 static int hf_error_code_ig         = -1;
65 static int hf_status_type           = -1;
66 static int hf_status_id             = -1;
67 static int hf_release_reason        = -1;
68 static int hf_tei_status            = -1;
69 static int hf_asp_id                = -1;
70 static int hf_parameter_tag         = -1;
71 static int hf_parameter_tag_ig      = -1;
72 static int hf_parameter_length      = -1;
73 static int hf_parameter_value       = -1;
74 static int hf_parameter_padding     = -1;
75 static int hf_version               = -1;
76 static int hf_reserved              = -1;
77 static int hf_message_class         = -1;
78 static int hf_message_type          = -1;
79 static int hf_message_length        = -1;
80
81 /* Initialize the subtree pointers */
82 static gint ett_iua                 = -1;
83 static gint ett_iua_parameter       = -1;
84
85 /* option setable via preferences, default is plain RFC 3057 */
86 static gboolean support_IG          = FALSE;
87
88 static dissector_handle_t q931_handle;
89
90 #define ADD_PADDING(x) ((((x) + 3) >> 2) << 2)
91
92 #define PARAMETER_TAG_LENGTH    2
93 #define PARAMETER_LENGTH_LENGTH 2
94 #define PARAMETER_HEADER_LENGTH (PARAMETER_TAG_LENGTH + PARAMETER_LENGTH_LENGTH)
95
96 #define PARAMETER_TAG_OFFSET    0
97 #define PARAMETER_LENGTH_OFFSET (PARAMETER_TAG_OFFSET + PARAMETER_TAG_LENGTH)
98 #define PARAMETER_VALUE_OFFSET  (PARAMETER_LENGTH_OFFSET + PARAMETER_LENGTH_LENGTH)
99 #define PARAMETER_HEADER_OFFSET PARAMETER_TAG_OFFSET
100
101 #define INT_INTERFACE_ID_OFFSET PARAMETER_VALUE_OFFSET
102 #define INT_INTERFACE_ID_LENGTH 4
103
104 static void
105 dissect_int_interface_identifier_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
106 {
107   proto_tree_add_item(parameter_tree, hf_int_interface_id, parameter_tvb, INT_INTERFACE_ID_OFFSET, INT_INTERFACE_ID_LENGTH, NETWORK_BYTE_ORDER);
108   proto_item_append_text(parameter_item, " (%d)", tvb_get_ntohl(parameter_tvb, INT_INTERFACE_ID_OFFSET));
109 }
110
111 #define TEXT_INTERFACE_ID_OFFSET PARAMETER_VALUE_OFFSET
112
113 static void
114 dissect_text_interface_identifier_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
115 {
116   guint16 interface_id_length;
117
118   interface_id_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
119
120   proto_tree_add_item(parameter_tree, hf_text_interface_id, parameter_tvb, TEXT_INTERFACE_ID_OFFSET, interface_id_length, NETWORK_BYTE_ORDER);
121   proto_item_append_text(parameter_item, " (%.*s)", interface_id_length,
122                          (const char *)tvb_get_ptr(parameter_tvb, TEXT_INTERFACE_ID_OFFSET, interface_id_length));
123 }
124
125 #define INFO_STRING_OFFSET PARAMETER_VALUE_OFFSET
126
127 static void
128 dissect_info_string_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
129 {
130   guint16 info_string_length;
131
132   info_string_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
133   proto_tree_add_item(parameter_tree, hf_info_string, parameter_tvb, INFO_STRING_OFFSET, info_string_length, NETWORK_BYTE_ORDER);
134   proto_item_append_text(parameter_item, " (%.*s)", info_string_length,
135                          (const char *)tvb_get_ptr(parameter_tvb, INFO_STRING_OFFSET, info_string_length));
136 }
137
138 #define DLCI_SAPI_LENGTH  1
139 #define DLCI_TEI_LENGTH   1
140 #define DLCI_SPARE_LENGTH 2
141
142 #define DLCI_SAPI_OFFSET  PARAMETER_VALUE_OFFSET
143 #define DLCI_TEI_OFFSET   (DLCI_SAPI_OFFSET + DLCI_SAPI_LENGTH)
144 #define DLCI_SPARE_OFFSET (DLCI_TEI_OFFSET + DLCI_TEI_LENGTH)
145
146 #define ZERO_BIT_MASK     0x01
147 #define SPARE_BIT_MASK    0x02
148 #define SAPI_MASK         0xfc
149 #define ONE_BIT_MASK      0x01
150 #define TEI_MASK          0xfe
151
152 static void
153 dissect_dlci_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree)
154 {
155   proto_tree_add_item(parameter_tree, hf_dlci_zero_bit,  parameter_tvb, DLCI_SAPI_OFFSET,  DLCI_SAPI_LENGTH,  NETWORK_BYTE_ORDER);
156   proto_tree_add_item(parameter_tree, hf_dlci_spare_bit, parameter_tvb, DLCI_SAPI_OFFSET,  DLCI_SAPI_LENGTH,  NETWORK_BYTE_ORDER);
157   proto_tree_add_item(parameter_tree, hf_dlci_sapi,      parameter_tvb, DLCI_SAPI_OFFSET,  DLCI_SAPI_LENGTH,  NETWORK_BYTE_ORDER);
158   proto_tree_add_item(parameter_tree, hf_dlci_one_bit,   parameter_tvb, DLCI_TEI_OFFSET,   DLCI_TEI_LENGTH,   NETWORK_BYTE_ORDER);
159   proto_tree_add_item(parameter_tree, hf_dlci_tei,       parameter_tvb, DLCI_TEI_OFFSET,   DLCI_TEI_LENGTH,   NETWORK_BYTE_ORDER);
160   proto_tree_add_item(parameter_tree, hf_dlci_spare,     parameter_tvb, DLCI_SPARE_OFFSET, DLCI_SPARE_LENGTH, NETWORK_BYTE_ORDER);
161 }
162
163 static void
164 dissect_diagnostic_information_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
165 {
166   guint16 diag_info_length;
167
168   diag_info_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
169   proto_tree_add_item(parameter_tree, hf_diag_info, parameter_tvb, PARAMETER_VALUE_OFFSET, diag_info_length, NETWORK_BYTE_ORDER);
170   proto_item_append_text(parameter_item, " (%u byte%s)", diag_info_length, plurality(diag_info_length, "", "s"));
171 }
172
173 #define START_LENGTH 4
174 #define END_LENGTH   4
175 #define INTERVAL_LENGTH (START_LENGTH + END_LENGTH)
176
177 #define START_OFFSET 0
178 #define END_OFFSET   (START_OFFSET + START_LENGTH)
179
180 static void
181 dissect_integer_range_interface_identifier_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
182 {
183   guint16 number_of_ranges, range_number;
184   gint offset;
185
186   number_of_ranges = (tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH) / INTERVAL_LENGTH;
187   offset = PARAMETER_VALUE_OFFSET;
188   for(range_number = 1; range_number <= number_of_ranges; range_number++) {
189     proto_tree_add_item(parameter_tree, hf_interface_range_start, parameter_tvb, offset + START_OFFSET, START_LENGTH, NETWORK_BYTE_ORDER);
190     proto_tree_add_item(parameter_tree, hf_interface_range_end,   parameter_tvb, offset + END_OFFSET,   END_LENGTH,   NETWORK_BYTE_ORDER);
191     offset += INTERVAL_LENGTH;
192   };
193
194   proto_item_append_text(parameter_item, " (%u range%s)", number_of_ranges, plurality(number_of_ranges, "", "s"));
195 }
196
197 #define HEARTBEAT_DATA_OFFSET PARAMETER_VALUE_OFFSET
198
199 static void
200 dissect_heartbeat_data_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
201 {
202   guint16 heartbeat_data_length;
203
204   heartbeat_data_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
205   proto_tree_add_item(parameter_tree, hf_heartbeat_data, parameter_tvb, HEARTBEAT_DATA_OFFSET, heartbeat_data_length, NETWORK_BYTE_ORDER);
206   proto_item_append_text(parameter_item, " (%u byte%s)", heartbeat_data_length, plurality(heartbeat_data_length, "", "s"));
207 }
208
209 #define ASP_MGMT_REASON   1
210
211 static const value_string asp_reason_values[] = {
212   { ASP_MGMT_REASON,      "Management inhibit" },
213   { 0,                    NULL } };
214
215 #define ASP_REASON_LENGTH 4
216 #define ASP_REASON_OFFSET PARAMETER_VALUE_OFFSET
217
218 static void
219 dissect_asp_reason_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
220 {
221   proto_tree_add_item(parameter_tree, hf_asp_reason, parameter_tvb, ASP_REASON_OFFSET, ASP_REASON_LENGTH, NETWORK_BYTE_ORDER);
222   proto_item_append_text(parameter_item, " (%s)", val_to_str(tvb_get_ntohl(parameter_tvb, ASP_REASON_OFFSET), asp_reason_values, "unknown"));
223 }
224
225 #define OVER_RIDE_TRAFFIC_MODE_TYPE  1
226 #define LOAD_SHARE_TRAFFIC_MODE_TYPE 2
227
228 static const value_string traffic_mode_type_values[] = {
229   { OVER_RIDE_TRAFFIC_MODE_TYPE,      "Over-ride" },
230   { LOAD_SHARE_TRAFFIC_MODE_TYPE,     "Load-share" },
231   { 0,                    NULL } };
232
233 #define TRAFFIC_MODE_TYPE_LENGTH 4
234 #define TRAFFIC_MODE_TYPE_OFFSET PARAMETER_VALUE_OFFSET
235
236 static void
237 dissect_traffic_mode_type_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
238 {
239   proto_tree_add_item(parameter_tree, hf_traffic_mode_type, parameter_tvb, TRAFFIC_MODE_TYPE_OFFSET, TRAFFIC_MODE_TYPE_LENGTH, NETWORK_BYTE_ORDER);
240   proto_item_append_text(parameter_item, " (%s)",
241                          val_to_str(tvb_get_ntohl(parameter_tvb, TRAFFIC_MODE_TYPE_OFFSET), traffic_mode_type_values, "unknown"));
242 }
243
244 #define INVALID_VERSION_ERROR                         0x01
245 #define INVALID_INTERFACE_IDENTIFIER_ERROR            0x02
246 #define UNSUPPORTED_MESSAGE_CLASS_ERROR               0x03
247 #define UNSUPPORTED_MESSAGE_TYPE_ERROR                0x04
248 #define UNSUPPORTED_TRAFFIC_HANDLING_MODE_ERROR       0x05
249 #define UNEXPECTED_MESSAGE_ERROR                      0x06
250 #define PROTOCOL_ERROR                                0x07
251 #define UNSUPPORTED_INTERFACE_IDENTIFIER_TYPE_ERROR   0x08
252 #define INVALID_STREAM_IDENTIFIER_ERROR               0x09
253 #define UNASSIGNED_TEI_ERROR                          0x0a
254 #define UNRECOGNIZED_SAPI_ERROR                       0x0b
255 #define INVALID_TEI_SAPI_COMBINATION_ERROR            0x0c
256 #define REFUSED_MANAGEMENT_BLOCKING_ERROR             0x0d
257 #define ASP_IDENTIFIER_REQUIRED_ERROR                 0x0e
258 #define INVALID_ASP_IDENTIFIER_ERROR                  0x0f
259
260 static const value_string error_code_values[] = {
261   { INVALID_VERSION_ERROR,                       "Invalid version" },
262   { INVALID_INTERFACE_IDENTIFIER_ERROR,          "Invalid interface identifier" },
263   { UNSUPPORTED_MESSAGE_CLASS_ERROR,             "Unsuported message class" },
264   { UNSUPPORTED_MESSAGE_TYPE_ERROR,              "Unsupported message type" },
265   { UNSUPPORTED_TRAFFIC_HANDLING_MODE_ERROR,     "Unsupported traffic handling mode" },
266   { UNEXPECTED_MESSAGE_ERROR,                    "Unexpected message" },
267   { PROTOCOL_ERROR,                              "Protocol error" },
268   { UNSUPPORTED_INTERFACE_IDENTIFIER_TYPE_ERROR, "Unsupported interface identifier type" },
269   { INVALID_STREAM_IDENTIFIER_ERROR,             "Invalid stream identifier" },
270   { UNASSIGNED_TEI_ERROR,                        "Unassigned TEI" },
271   { UNRECOGNIZED_SAPI_ERROR,                     "Unrecognized SAPI" },
272   { INVALID_TEI_SAPI_COMBINATION_ERROR,          "Invalid TEI/SAPI combination" },
273   { 0,                                           NULL } };
274
275 static const value_string error_code_ig_values[] = {
276   { INVALID_VERSION_ERROR,                       "Invalid version" },
277   { INVALID_INTERFACE_IDENTIFIER_ERROR,          "Invalid interface identifier" },
278   { UNSUPPORTED_MESSAGE_CLASS_ERROR,             "Unsuported message class" },
279   { UNSUPPORTED_MESSAGE_TYPE_ERROR,              "Unsupported message type" },
280   { UNSUPPORTED_TRAFFIC_HANDLING_MODE_ERROR,     "Unsupported traffic handling mode" },
281   { UNEXPECTED_MESSAGE_ERROR,                    "Unexpected message" },
282   { PROTOCOL_ERROR,                              "Protocol error" },
283   { UNSUPPORTED_INTERFACE_IDENTIFIER_TYPE_ERROR, "Unsupported interface identifier type" },
284   { INVALID_STREAM_IDENTIFIER_ERROR,             "Invalid stream identifier" },
285   { UNASSIGNED_TEI_ERROR,                        "Unassigned TEI" },
286   { UNRECOGNIZED_SAPI_ERROR,                     "Unrecognized SAPI" },
287   { INVALID_TEI_SAPI_COMBINATION_ERROR,          "Invalid TEI/SAPI combination" },
288   { REFUSED_MANAGEMENT_BLOCKING_ERROR,           "Refused - Management blocking" },
289   { ASP_IDENTIFIER_REQUIRED_ERROR,               "ASP identifier required" },
290   { INVALID_ASP_IDENTIFIER_ERROR,                "Invalid ASP identifier" },
291   { 0,                                           NULL } };
292
293 #define ERROR_CODE_LENGTH 4
294 #define ERROR_CODE_OFFSET PARAMETER_VALUE_OFFSET
295
296 static void
297 dissect_error_code_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
298 {
299   proto_tree_add_item(parameter_tree, support_IG?hf_error_code_ig:hf_error_code, parameter_tvb, ERROR_CODE_OFFSET, ERROR_CODE_LENGTH, NETWORK_BYTE_ORDER);
300   proto_item_append_text(parameter_item, " (%s)",
301                          val_to_str(tvb_get_ntohl(parameter_tvb, ERROR_CODE_OFFSET), support_IG?error_code_ig_values:error_code_values, "unknown"));
302 }
303
304 #define ASP_STATE_CHANGE_STATUS_TYPE  0x01
305 #define OTHER_STATUS_TYPE             0x02
306
307 static const value_string status_type_values[] = {
308   { ASP_STATE_CHANGE_STATUS_TYPE,        "Application server state change" },
309   { OTHER_STATUS_TYPE,                   "Other" },
310   { 0,                                   NULL } };
311
312 #define AS_DOWN_STATUS_IDENT          0x01
313 #define AS_INACTIVE_STATUS_IDENT      0x02
314 #define AS_ACTIVE_STATUS_IDENT        0x03
315 #define AS_PENDING_STATUS_IDENT       0x04
316
317 #define INSUFFICIENT_ASP_RESOURCES_STATUS_IDENT 0x01
318 #define ALTERNATE_ASP_ACTIVE_STATUS_IDENT       0x02
319
320 static const value_string status_type_id_values[] = {
321   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_DOWN_STATUS_IDENT,         "Application server down" },
322   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_INACTIVE_STATUS_IDENT,     "Application server inactive" },
323   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_ACTIVE_STATUS_IDENT,       "Application server active" },
324   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_PENDING_STATUS_IDENT,      "Application server pending" },
325   { OTHER_STATUS_TYPE * 256 * 256 + INSUFFICIENT_ASP_RESOURCES_STATUS_IDENT, "Insufficient ASP resources active in AS" },
326   { OTHER_STATUS_TYPE * 256 * 256 + ALTERNATE_ASP_ACTIVE_STATUS_IDENT,       "Alternate ASP active" },
327   { 0,                                           NULL } };
328
329 static const value_string status_type_id_ig_values[] = {
330   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_INACTIVE_STATUS_IDENT,     "Application server inactive" },
331   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_ACTIVE_STATUS_IDENT,       "Application server active" },
332   { ASP_STATE_CHANGE_STATUS_TYPE * 256 * 256 + AS_PENDING_STATUS_IDENT,      "Application server pending" },
333   { OTHER_STATUS_TYPE * 256 * 256 + INSUFFICIENT_ASP_RESOURCES_STATUS_IDENT, "Insufficient ASP resources active in AS" },
334   { OTHER_STATUS_TYPE * 256 * 256 + ALTERNATE_ASP_ACTIVE_STATUS_IDENT,       "Alternate ASP active" },
335   { 0,                                           NULL } };
336
337 #define STATUS_TYPE_LENGTH  2
338 #define STATUS_IDENT_LENGTH 2
339 #define STATUS_TYPE_OFFSET  PARAMETER_VALUE_OFFSET
340 #define STATUS_IDENT_OFFSET (STATUS_TYPE_OFFSET + STATUS_TYPE_LENGTH)
341
342 static void
343 dissect_status_type_identification_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
344 {
345   guint16 status_type, status_id;
346
347   status_type = tvb_get_ntohs(parameter_tvb, STATUS_TYPE_OFFSET);
348   status_id   = tvb_get_ntohs(parameter_tvb, STATUS_IDENT_OFFSET);
349
350   proto_tree_add_item(parameter_tree, hf_status_type, parameter_tvb, STATUS_TYPE_OFFSET, STATUS_TYPE_LENGTH, NETWORK_BYTE_ORDER);
351   proto_tree_add_uint_format(parameter_tree, hf_status_id,  parameter_tvb, STATUS_IDENT_OFFSET, STATUS_IDENT_LENGTH,
352                              status_id, "Status identification: %u (%s)", status_id,
353                              val_to_str(status_type * 256 * 256 + status_id, support_IG?status_type_id_ig_values:status_type_id_values, "unknown"));
354
355   proto_item_append_text(parameter_item, " (%s)",
356                          val_to_str(status_type * 256 * 256 + status_id, support_IG?status_type_id_ig_values:status_type_id_values, "unknown status information"));
357 }
358
359 #define PROTOCOL_DATA_OFFSET PARAMETER_VALUE_OFFSET
360
361 static void
362 dissect_protocol_data_parameter(tvbuff_t *parameter_tvb, proto_item *parameter_item, packet_info *pinfo, proto_tree *tree)
363 {
364   guint16 protocol_data_length;
365   tvbuff_t *protocol_data_tvb;
366
367   protocol_data_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
368   protocol_data_tvb    = tvb_new_subset(parameter_tvb, PROTOCOL_DATA_OFFSET, protocol_data_length, protocol_data_length);
369   call_dissector(q931_handle, protocol_data_tvb, pinfo, tree);
370
371   proto_item_append_text(parameter_item, " (%u byte%s)", protocol_data_length, plurality(protocol_data_length, "", "s"));
372 }
373
374 #define RELEASE_MGMT_REASON   0
375 #define RELEASE_PHYS_REASON   1
376 #define RELEASE_DM_REASON     2
377 #define RELEASE_OTHER_REASON  4
378
379 static const value_string release_reason_values[] = {
380   { RELEASE_MGMT_REASON,  "Management layer generated release" },
381   { RELEASE_PHYS_REASON,  "Physical layer alarm generated release" },
382   { RELEASE_DM_REASON,    "Layer 2 should release" },
383   { RELEASE_OTHER_REASON, "Other reason" },
384   { 0,                    NULL } };
385
386 #define RELEASE_REASON_OFFSET PARAMETER_VALUE_OFFSET
387 #define RELEASE_REASON_LENGTH 4
388
389 static void
390 dissect_release_reason_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
391 {
392   proto_tree_add_item(parameter_tree, hf_release_reason, parameter_tvb, RELEASE_REASON_OFFSET, RELEASE_REASON_LENGTH, NETWORK_BYTE_ORDER);
393   proto_item_append_text(parameter_item, " (%s)",
394                          val_to_str(tvb_get_ntohl(parameter_tvb, RELEASE_REASON_OFFSET), release_reason_values, "unknown"));
395 }
396
397 #define TEI_STATUS_ASSIGNED       0
398 #define TEI_STATUS_UNASSIGNED     1
399
400 static const value_string tei_status_values[] = {
401   { TEI_STATUS_ASSIGNED,   "TEI is considered assigned by Q.921" },
402   { TEI_STATUS_UNASSIGNED, "TEI is considered unassigned by Q.921" },
403   { 0,                    NULL } };
404
405 #define TEI_STATUS_LENGTH 4
406 #define TEI_STATUS_OFFSET PARAMETER_VALUE_OFFSET
407
408 static void
409 dissect_tei_status_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
410 {
411   proto_tree_add_item(parameter_tree, hf_tei_status, parameter_tvb, TEI_STATUS_OFFSET, TEI_STATUS_LENGTH, NETWORK_BYTE_ORDER);
412   proto_item_append_text(parameter_item, " (%s)",
413                       val_to_str(tvb_get_ntohl(parameter_tvb, TEI_STATUS_OFFSET), tei_status_values, "unknown"));
414 }
415
416 #define ASP_ID_LENGTH 4
417 #define ASP_ID_OFFSET PARAMETER_VALUE_OFFSET
418
419 static void
420 dissect_asp_identifier_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
421 {
422   proto_tree_add_item(parameter_tree, hf_asp_id, parameter_tvb, ASP_ID_OFFSET, ASP_ID_LENGTH, NETWORK_BYTE_ORDER);
423   proto_item_append_text(parameter_item, " (%u)", tvb_get_ntohl(parameter_tvb, ASP_ID_OFFSET));
424 }
425
426 static void
427 dissect_unknown_parameter(tvbuff_t *parameter_tvb, proto_tree *parameter_tree, proto_item *parameter_item)
428 {
429   guint16 parameter_value_length;
430
431   parameter_value_length = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET) - PARAMETER_HEADER_LENGTH;
432   if (parameter_value_length > 0)
433     proto_tree_add_item(parameter_tree, hf_parameter_value, parameter_tvb, PARAMETER_VALUE_OFFSET, parameter_value_length, NETWORK_BYTE_ORDER);
434   proto_item_append_text(parameter_item, " with tag %u and %u byte%s value",
435                          tvb_get_ntohs(parameter_tvb, PARAMETER_TAG_OFFSET), parameter_value_length, plurality(parameter_value_length, "", "s"));
436 }
437
438 #define INT_INTERFACE_IDENTIFIER_PARAMETER_TAG           0x01
439 #define TEXT_INTERFACE_IDENTIFIER_PARAMETER_TAG          0x03
440 #define INFO_PARAMETER_TAG                               0x04
441 #define DLCI_PARAMETER_TAG                               0x05
442 #define DIAGNOSTIC_INFORMATION_PARAMETER_TAG             0x07
443 #define INTEGER_RANGE_INTERFACE_IDENTIFIER_PARAMETER_TAG 0x08
444 #define HEARTBEAT_DATA_PARAMETER_TAG                     0x09
445 #define ASP_REASON_PARAMETER_TAG                         0x0a
446 #define TRAFFIC_MODE_TYPE_PARAMETER_TAG                  0x0b
447 #define ERROR_CODE_PARAMETER_TAG                         0x0c
448 #define STATUS_TYPE_INDENTIFICATION_PARAMETER_TAG        0x0d
449 #define PROTOCOL_DATA_PARAMETER_TAG                      0x0e
450 #define RELEASE_REASON_PARAMETER_TAG                     0x0f
451 #define TEI_STATUS_PARAMETER_TAG                         0x10
452 #define ASP_IDENTIFIER_PARAMETER_TAG                     0x11
453
454 static const value_string parameter_tag_values[] = {
455   { INT_INTERFACE_IDENTIFIER_PARAMETER_TAG,                "Integer interface identifier" },
456   { TEXT_INTERFACE_IDENTIFIER_PARAMETER_TAG,               "Text interface identifier" },
457   { INFO_PARAMETER_TAG,                                    "Info" },
458   { DLCI_PARAMETER_TAG,                                    "DLCI" },
459   { DIAGNOSTIC_INFORMATION_PARAMETER_TAG,                  "Diagnostic information" },
460   { INTEGER_RANGE_INTERFACE_IDENTIFIER_PARAMETER_TAG,      "Integer range interface identifier" },
461   { HEARTBEAT_DATA_PARAMETER_TAG,                          "Hearbeat data" },
462   { ASP_REASON_PARAMETER_TAG,                              "Reason" },
463   { TRAFFIC_MODE_TYPE_PARAMETER_TAG,                       "Traffic mode type" },
464   { ERROR_CODE_PARAMETER_TAG,                              "Error code" },
465   { STATUS_TYPE_INDENTIFICATION_PARAMETER_TAG,             "Status type/identification" },
466   { PROTOCOL_DATA_PARAMETER_TAG,                           "Protocol data" },
467   { RELEASE_REASON_PARAMETER_TAG,                          "Reason" },
468   { TEI_STATUS_PARAMETER_TAG,                              "TEI status" },
469   { 0,                           NULL } };
470
471 static const value_string parameter_tag_ig_values[] = {
472   { INT_INTERFACE_IDENTIFIER_PARAMETER_TAG,                "Integer interface identifier" },
473   { TEXT_INTERFACE_IDENTIFIER_PARAMETER_TAG,               "Text interface identifier" },
474   { INFO_PARAMETER_TAG,                                    "Info" },
475   { DLCI_PARAMETER_TAG,                                    "DLCI" },
476   { DIAGNOSTIC_INFORMATION_PARAMETER_TAG,                  "Diagnostic information" },
477   { INTEGER_RANGE_INTERFACE_IDENTIFIER_PARAMETER_TAG,      "Integer range interface identifier" },
478   { HEARTBEAT_DATA_PARAMETER_TAG,                          "Hearbeat data" },
479   { TRAFFIC_MODE_TYPE_PARAMETER_TAG,                       "Traffic mode type" },
480   { ERROR_CODE_PARAMETER_TAG,                              "Error code" },
481   { STATUS_TYPE_INDENTIFICATION_PARAMETER_TAG,             "Status type/identification" },
482   { PROTOCOL_DATA_PARAMETER_TAG,                           "Protocol data" },
483   { RELEASE_REASON_PARAMETER_TAG,                          "Reason" },
484   { TEI_STATUS_PARAMETER_TAG,                              "TEI status" },
485   { ASP_IDENTIFIER_PARAMETER_TAG,                          "ASP identifier"},
486   { 0,                           NULL } };
487
488 static void
489 dissect_parameter(tvbuff_t *parameter_tvb, packet_info *pinfo, proto_tree *tree, proto_tree *iua_tree)
490 {
491   guint16 tag, length, padding_length;
492   proto_item *parameter_item;
493   proto_tree *parameter_tree;
494
495   /* extract tag and length from the parameter */
496   tag            = tvb_get_ntohs(parameter_tvb, PARAMETER_TAG_OFFSET);
497   length         = tvb_get_ntohs(parameter_tvb, PARAMETER_LENGTH_OFFSET);
498   padding_length = tvb_length(parameter_tvb) - length;
499
500   /* create proto_tree stuff */
501   parameter_item   = proto_tree_add_text(iua_tree, parameter_tvb, PARAMETER_HEADER_OFFSET, tvb_length(parameter_tvb),
502                                          val_to_str(tag, support_IG?parameter_tag_ig_values:parameter_tag_values, "Unknown parameter"));
503   parameter_tree   = proto_item_add_subtree(parameter_item, ett_iua_parameter);
504
505   /* add tag and length to the iua tree */
506   proto_tree_add_item(parameter_tree, support_IG?hf_parameter_tag_ig:hf_parameter_tag, parameter_tvb, PARAMETER_TAG_OFFSET, PARAMETER_TAG_LENGTH, NETWORK_BYTE_ORDER);
507   proto_tree_add_item(parameter_tree, hf_parameter_length, parameter_tvb, PARAMETER_LENGTH_OFFSET, PARAMETER_LENGTH_LENGTH, NETWORK_BYTE_ORDER);
508
509   switch(tag) {
510   case INT_INTERFACE_IDENTIFIER_PARAMETER_TAG:
511     dissect_int_interface_identifier_parameter(parameter_tvb, parameter_tree, parameter_item);
512     break;
513   case TEXT_INTERFACE_IDENTIFIER_PARAMETER_TAG:
514     dissect_text_interface_identifier_parameter(parameter_tvb, parameter_tree, parameter_item);
515     break;
516   case INFO_PARAMETER_TAG:
517     dissect_info_string_parameter(parameter_tvb, parameter_tree, parameter_item);
518     break;
519   case DLCI_PARAMETER_TAG:
520     dissect_dlci_parameter(parameter_tvb, parameter_tree);
521     break;
522   case DIAGNOSTIC_INFORMATION_PARAMETER_TAG:
523     dissect_diagnostic_information_parameter(parameter_tvb, parameter_tree, parameter_item);
524     break;
525   case INTEGER_RANGE_INTERFACE_IDENTIFIER_PARAMETER_TAG:
526     dissect_integer_range_interface_identifier_parameter(parameter_tvb, parameter_tree, parameter_item);
527     break;
528   case HEARTBEAT_DATA_PARAMETER_TAG:
529     dissect_heartbeat_data_parameter(parameter_tvb, parameter_tree, parameter_item);
530     break;
531   case ASP_REASON_PARAMETER_TAG:
532     if (support_IG)
533       dissect_unknown_parameter(parameter_tvb, parameter_tree, parameter_item);
534     else
535       dissect_asp_reason_parameter(parameter_tvb, parameter_tree, parameter_item);
536     break;
537   case TRAFFIC_MODE_TYPE_PARAMETER_TAG:
538     dissect_traffic_mode_type_parameter(parameter_tvb, parameter_tree, parameter_item);
539     break;
540   case ERROR_CODE_PARAMETER_TAG:
541     dissect_error_code_parameter(parameter_tvb, parameter_tree, parameter_item);
542     break;
543   case STATUS_TYPE_INDENTIFICATION_PARAMETER_TAG:
544     dissect_status_type_identification_parameter(parameter_tvb, parameter_tree, parameter_item);
545     break;
546   case PROTOCOL_DATA_PARAMETER_TAG:
547     dissect_protocol_data_parameter(parameter_tvb, parameter_item, pinfo, tree);
548     break;
549   case RELEASE_REASON_PARAMETER_TAG:
550     dissect_release_reason_parameter(parameter_tvb, parameter_tree, parameter_item);
551     break;
552   case TEI_STATUS_PARAMETER_TAG:
553     dissect_tei_status_parameter(parameter_tvb, parameter_tree, parameter_item);
554     break;
555   case ASP_IDENTIFIER_PARAMETER_TAG:
556     if (support_IG)
557       dissect_asp_identifier_parameter(parameter_tvb, parameter_tree, parameter_item);
558     else
559       dissect_unknown_parameter(parameter_tvb, parameter_tree, parameter_item);
560     break;
561   default:
562     dissect_unknown_parameter(parameter_tvb, parameter_tree, parameter_item);
563     break;
564   };
565
566   if (padding_length > 0)
567     proto_tree_add_item(parameter_tree, hf_parameter_padding, parameter_tvb, PARAMETER_HEADER_OFFSET + length, padding_length, NETWORK_BYTE_ORDER);
568 }
569
570 static void
571 dissect_parameters(tvbuff_t *parameters_tvb, packet_info *pinfo, proto_tree *tree, proto_tree *iua_tree)
572 {
573   gint offset, length, total_length, remaining_length;
574   tvbuff_t *parameter_tvb;
575
576   offset = 0;
577   while((remaining_length = tvb_length_remaining(parameters_tvb, offset))) {
578     length       = tvb_get_ntohs(parameters_tvb, offset + PARAMETER_LENGTH_OFFSET);
579     total_length = ADD_PADDING(length);
580     if (remaining_length >= length)
581       total_length = MIN(total_length, remaining_length);
582     /* create a tvb for the parameter including the padding bytes */
583     parameter_tvb  = tvb_new_subset(parameters_tvb, offset, total_length, total_length);
584     dissect_parameter(parameter_tvb, pinfo, tree, iua_tree);
585     /* get rid of the handled parameter */
586     offset += total_length;
587   }
588 }
589
590 #define VERSION_LENGTH         1
591 #define RESERVED_LENGTH        1
592 #define MESSAGE_CLASS_LENGTH   1
593 #define MESSAGE_TYPE_LENGTH    1
594 #define MESSAGE_LENGTH_LENGTH  4
595 #define COMMON_HEADER_LENGTH   (VERSION_LENGTH + RESERVED_LENGTH + MESSAGE_CLASS_LENGTH + \
596                                 MESSAGE_TYPE_LENGTH + MESSAGE_LENGTH_LENGTH)
597
598 #define COMMON_HEADER_OFFSET   0
599 #define VERSION_OFFSET         COMMON_HEADER_OFFSET
600 #define RESERVED_OFFSET        (VERSION_OFFSET + VERSION_LENGTH)
601 #define MESSAGE_CLASS_OFFSET   (RESERVED_OFFSET + RESERVED_LENGTH)
602 #define MESSAGE_TYPE_OFFSET    (MESSAGE_CLASS_OFFSET + MESSAGE_CLASS_LENGTH)
603 #define MESSAGE_LENGTH_OFFSET  (MESSAGE_TYPE_OFFSET + MESSAGE_TYPE_LENGTH)
604 #define PARAMETERS_OFFSET      (COMMON_HEADER_OFFSET + COMMON_HEADER_LENGTH)
605
606 #define PROTOCOL_VERSION_RELEASE_1             1
607
608 static const value_string protocol_version_values[] = {
609   { PROTOCOL_VERSION_RELEASE_1,  "Release 1" },
610   { 0,                           NULL } };
611
612 #define MESSAGE_CLASS_MGMT_MESSAGE        0
613 #define MESSAGE_CLASS_TFER_MESSAGE        1
614 #define MESSAGE_CLASS_SSNM_MESSAGE        2
615 #define MESSAGE_CLASS_ASPSM_MESSAGE       3
616 #define MESSAGE_CLASS_ASPTM_MESSAGE       4
617 #define MESSAGE_CLASS_QPTM_MESSAGE        5
618 #define MESSAGE_CLASS_MAUP_MESSAGE        6
619 #define MESSAGE_CLASS_CL_SUA_MESSAGE      7
620 #define MESSAGE_CLASS_CO_SUA_MESSAGE      8
621
622 static const value_string message_class_values[] = {
623   { MESSAGE_CLASS_MGMT_MESSAGE,   "Management messages" },
624   { MESSAGE_CLASS_TFER_MESSAGE,   "Transfer messages" },
625   { MESSAGE_CLASS_SSNM_MESSAGE,   "SS7 signalling network management messages" },
626   { MESSAGE_CLASS_ASPSM_MESSAGE,  "ASP state maintenance messages" },
627   { MESSAGE_CLASS_ASPTM_MESSAGE,  "ASP traffic maintenance messages" },
628   { MESSAGE_CLASS_QPTM_MESSAGE,   "Q.921/Q.931 boundary primitive transport messages" },
629   { MESSAGE_CLASS_MAUP_MESSAGE,   "MTP2 user adaptation messages" },
630   { MESSAGE_CLASS_CL_SUA_MESSAGE, "Connectionless messages (SUA)" },
631   { MESSAGE_CLASS_CO_SUA_MESSAGE, "Connection-oriented messages (SUA)" },
632   { 0,                             NULL } };
633
634 /* message types for MGMT messages */
635 #define MESSAGE_TYPE_ERR                  0
636 #define MESSAGE_TYPE_NTFY                 1
637 #define MESSAGE_TYPE_TEI_STATUS_REQ       2
638 #define MESSAGE_TYPE_TEI_STATUS_CON       3
639 #define MESSAGE_TYPE_TEI_STATUS_IND       4
640 #define MESSAGE_TYPE_TEI_QUERY_REQ        5
641
642 /* message types for ASPSM messages */
643 #define MESSAGE_TYPE_UP                   1
644 #define MESSAGE_TYPE_DOWN                 2
645 #define MESSAGE_TYPE_BEAT                 3
646 #define MESSAGE_TYPE_UP_ACK               4
647 #define MESSAGE_TYPE_DOWN_ACK             5
648 #define MESSAGE_TYPE_BEAT_ACK             6
649
650 /* message types for ASPTM messages */
651 #define MESSAGE_TYPE_ACTIVE               1
652 #define MESSAGE_TYPE_INACTIVE             2
653 #define MESSAGE_TYPE_ACTIVE_ACK           3
654 #define MESSAGE_TYPE_INACTIVE_ACK         4
655
656 /* message types for QPTM messages */
657 #define MESSAGE_TYPE_DATA_REQUEST         1
658 #define MESSAGE_TYPE_DATA_INDICATION      2
659 #define MESSAGE_TYPE_UNIT_DATA_REQUEST    3
660 #define MESSAGE_TYPE_UNIT_DATA_INDICATION 4
661 #define MESSAGE_TYPE_ESTABLISH_REQUEST    5
662 #define MESSAGE_TYPE_ESTABLISH_CONFIRM    6
663 #define MESSAGE_TYPE_ESTABLISH_INDICATION 7
664 #define MESSAGE_TYPE_RELEASE_REQUEST      8
665 #define MESSAGE_TYPE_RELEASE_CONFIRM      9
666 #define MESSAGE_TYPE_RELEASE_INDICATION  10
667
668
669 static const value_string message_class_type_values[] = {
670   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_ERR,                  "Error" },
671   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_NTFY,                 "Notify" },
672   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_REQ,       "TEI status request" },
673   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_CON,       "TEI status confirmation" },
674   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_IND,       "TEI status indication" },
675   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP,                   "ASP up" },
676   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN,                 "ASP down" },
677   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT,                 "Heartbeat" },
678   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP_ACK,               "ASP up ack" },
679   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN_ACK,             "ASP down ack" },
680   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT_ACK,             "Heartbeat ack" },
681   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE ,              "ASP active" },
682   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE ,            "ASP inactive" },
683   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE_ACK ,          "ASP active ack" },
684   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE_ACK ,        "ASP inactive ack" },
685   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_REQUEST,         "Data request" },
686   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_INDICATION,      "Data indication" },
687   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_REQUEST,    "Unit data request" },
688   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_INDICATION, "Unit data indication" },
689   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_REQUEST,    "Establish request" },
690   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_CONFIRM,    "Establish confirmation" },
691   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_INDICATION, "Establish indication" },
692   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_REQUEST,      "Release request" },
693   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_CONFIRM,      "Release confirmation" },
694   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_INDICATION,   "Release indication" },
695   { 0,                                                                     NULL } };
696
697 static const value_string message_class_type_ig_values[] = {
698   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_ERR,                  "Error" },
699   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_NTFY,                 "Notify" },
700   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_REQ,       "TEI status request" },
701   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_CON,       "TEI status confirmation" },
702   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_IND,       "TEI status indication" },
703   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_QUERY_REQ,        "TEI query request" },
704   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP,                   "ASP up" },
705   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN,                 "ASP down" },
706   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT,                 "Heartbeat" },
707   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP_ACK,               "ASP up ack" },
708   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN_ACK,             "ASP down ack" },
709   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT_ACK,             "Heartbeat ack" },
710   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE ,              "ASP active" },
711   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE ,            "ASP inactive" },
712   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE_ACK ,          "ASP active ack" },
713   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE_ACK ,        "ASP inactive ack" },
714   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_REQUEST,         "Data request" },
715   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_INDICATION,      "Data indication" },
716   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_REQUEST,    "Unit data request" },
717   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_INDICATION, "Unit data indication" },
718   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_REQUEST,    "Establish request" },
719   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_CONFIRM,    "Establish confirmation" },
720   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_INDICATION, "Establish indication" },
721   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_REQUEST,      "Release request" },
722   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_CONFIRM,      "Release confirmation" },
723   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_INDICATION,   "Release indication" },
724   { 0,                                                                     NULL } };
725
726 static const value_string message_class_type_acro_values[] = {
727   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_ERR,                  "ERR" },
728   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_NTFY,                 "NTFY" },
729   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_REQ,       "TEI_STAT_REQ" },
730   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_CON,       "TEI_STAT_CON" },
731   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_IND,       "TEI_STAT_IND" },
732   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP,                   "ASP_UP" },
733   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN,                 "ASP_DOWN" },
734   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT,                 "BEAT" },
735   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP_ACK,               "ASP_UP_ACK" },
736   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN_ACK,             "ASP_DOWN_ACK" },
737   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT_ACK,             "BEAT_ACK" },
738   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE ,              "ASP_ACTIVE" },
739   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE ,            "ASP_INACTIVE" },
740   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE_ACK ,          "ASP_ACTIVE_ACK" },
741   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE_ACK ,        "ASP_INACTIVE_ACK" },
742   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_REQUEST,         "DATA_REQ" },
743   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_INDICATION,      "DATA_IND" },
744   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_REQUEST,    "U_DATA_REQ" },
745   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_INDICATION, "U_DATA_IND" },
746   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_REQUEST,    "EST_REQ" },
747   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_CONFIRM,    "EST_CON" },
748   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_INDICATION, "EST_IND" },
749   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_REQUEST,      "REL_REQ" },
750   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_CONFIRM,      "REL_CON" },
751   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_INDICATION,   "REL_IND" },
752   { 0,                                                                     NULL } };
753
754 static const value_string message_class_type_acro_ig_values[] = {
755   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_ERR,                  "ERR" },
756   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_NTFY,                 "NTFY" },
757   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_REQ,       "TEI_STAT_REQ" },
758   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_CON,       "TEI_STAT_CON" },
759   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_STATUS_IND,       "TEI_STAT_IND" },
760   { MESSAGE_CLASS_MGMT_MESSAGE  * 256 + MESSAGE_TYPE_TEI_QUERY_REQ,        "TEI_QUERY_REQ" },
761   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP,                   "ASP_UP" },
762   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN,                 "ASP_DOWN" },
763   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT,                 "BEAT" },
764   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_UP_ACK,               "ASP_UP_ACK" },
765   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_DOWN_ACK,             "ASP_DOWN_ACK" },
766   { MESSAGE_CLASS_ASPSM_MESSAGE * 256 + MESSAGE_TYPE_BEAT_ACK,             "BEAT_ACK" },
767   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE ,              "ASP_ACTIVE" },
768   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE ,            "ASP_INACTIVE" },
769   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_ACTIVE_ACK ,          "ASP_ACTIVE_ACK" },
770   { MESSAGE_CLASS_ASPTM_MESSAGE * 256 + MESSAGE_TYPE_INACTIVE_ACK ,        "ASP_INACTIVE_ACK" },
771   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_REQUEST,         "DATA_REQ" },
772   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_DATA_INDICATION,      "DATA_IND" },
773   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_REQUEST,    "U_DATA_REQ" },
774   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_UNIT_DATA_INDICATION, "U_DATA_IND" },
775   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_REQUEST,    "EST_REQ" },
776   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_CONFIRM,    "EST_CON" },
777   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_ESTABLISH_INDICATION, "EST_IND" },
778   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_REQUEST,      "REL_REQ" },
779   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_CONFIRM,      "REL_CON" },
780   { MESSAGE_CLASS_QPTM_MESSAGE  * 256 + MESSAGE_TYPE_RELEASE_INDICATION,   "REL_IND" },
781   { 0,                                                                     NULL } };
782
783 static void
784 dissect_common_header(tvbuff_t *common_header_tvb, packet_info *pinfo, proto_tree *iua_tree)
785 {
786   guint8 message_class, message_type;
787
788   message_class  = tvb_get_guint8(common_header_tvb, MESSAGE_CLASS_OFFSET);
789   message_type   = tvb_get_guint8(common_header_tvb, MESSAGE_TYPE_OFFSET);
790
791   if (check_col(pinfo->cinfo, COL_INFO))
792     col_add_fstr(pinfo->cinfo, COL_INFO, "%s ", val_to_str(message_class * 256 + message_type, support_IG?message_class_type_acro_ig_values:message_class_type_acro_values, "UNKNOWN"));
793
794   if (iua_tree) {
795     /* add the components of the common header to the protocol tree */
796     proto_tree_add_item(iua_tree, hf_version, common_header_tvb, VERSION_OFFSET, VERSION_LENGTH, NETWORK_BYTE_ORDER);
797     proto_tree_add_item(iua_tree, hf_reserved, common_header_tvb, RESERVED_OFFSET, RESERVED_LENGTH, NETWORK_BYTE_ORDER);
798     proto_tree_add_item(iua_tree, hf_message_class, common_header_tvb, MESSAGE_CLASS_OFFSET, MESSAGE_CLASS_LENGTH, NETWORK_BYTE_ORDER);
799     proto_tree_add_uint_format(iua_tree, hf_message_type,
800                                common_header_tvb, MESSAGE_TYPE_OFFSET, MESSAGE_TYPE_LENGTH,
801                                message_type, "Message type: %u (%s)",
802                                message_type, val_to_str(message_class * 256 + message_type, support_IG?message_class_type_ig_values:message_class_type_values, "reserved"));
803     proto_tree_add_item(iua_tree, hf_message_length, common_header_tvb, MESSAGE_LENGTH_OFFSET, MESSAGE_LENGTH_LENGTH, NETWORK_BYTE_ORDER);
804   }
805 }
806
807 static void
808 dissect_iua_message(tvbuff_t *message_tvb, packet_info *pinfo, proto_tree *tree, proto_tree *iua_tree)
809 {
810   tvbuff_t *common_header_tvb, *parameters_tvb;
811
812   common_header_tvb = tvb_new_subset(message_tvb, COMMON_HEADER_OFFSET, COMMON_HEADER_LENGTH, COMMON_HEADER_LENGTH);
813   parameters_tvb    = tvb_new_subset(message_tvb, PARAMETERS_OFFSET, -1, -1);
814   dissect_common_header(common_header_tvb, pinfo, iua_tree);
815   dissect_parameters(parameters_tvb, pinfo, tree, iua_tree);
816 }
817
818 static void
819 dissect_iua(tvbuff_t *message_tvb, packet_info *pinfo, proto_tree *tree)
820 {
821   proto_item *iua_item;
822   proto_tree *iua_tree;
823
824   /* make entry in the Protocol column on summary display */
825   if (check_col(pinfo->cinfo, COL_PROTOCOL))
826     col_set_str(pinfo->cinfo, COL_PROTOCOL, support_IG?"IUA (RFC 3057 + IG)":"IUA (RFC 3057)");
827
828   /* In the interest of speed, if "tree" is NULL, don't do any work not
829      necessary to generate protocol tree items. */
830   if (tree) {
831     /* create the m3ua protocol tree */
832     iua_item = proto_tree_add_item(tree, proto_iua, message_tvb, 0, -1, FALSE);
833     iua_tree = proto_item_add_subtree(iua_item, ett_iua);
834   } else {
835     iua_tree = NULL;
836   };
837   /* dissect the message */
838   dissect_iua_message(message_tvb, pinfo, tree, iua_tree);
839 }
840
841 /* Register the protocol with Ethereal */
842 void
843 proto_register_iua(void)
844 {
845
846   /* Setup list of header fields */
847   static hf_register_info hf[] = {
848     { &hf_int_interface_id,      { "Integer interface identifier", "iua.int_interface_identifier",  FT_INT32,   BASE_HEX,  NULL,                           0x0,            "", HFILL } },
849     { &hf_text_interface_id,     { "Text interface identifier",    "iua.text_interface_identifier", FT_STRING,  BASE_NONE, NULL,                           0x0,            "", HFILL } },
850     { &hf_info_string,           { "Info string",                  "iua.info_string",               FT_STRING,  BASE_NONE, NULL,                           0x0,            "", HFILL } },
851     { &hf_dlci_zero_bit,         { "Zero bit",                     "iua.dlci_zero_bit",             FT_BOOLEAN, 8,         NULL,                           ZERO_BIT_MASK,  "", HFILL } },
852     { &hf_dlci_spare_bit,        { "Spare bit",                    "iua.dlci_spare_bit",            FT_BOOLEAN, 8,         NULL,                           SPARE_BIT_MASK, "", HFILL } },
853     { &hf_dlci_sapi,             { "SAPI",                         "iua.dlci_sapi",                 FT_UINT8,   BASE_HEX,  NULL,                           SAPI_MASK,      "", HFILL } },
854     { &hf_dlci_one_bit,          { "One bit",                      "iua.dlci_one_bit",              FT_BOOLEAN, 8,         NULL,                           ONE_BIT_MASK,   "", HFILL } },
855     { &hf_dlci_tei,              { "TEI",                          "iua.dlci_tei",                  FT_UINT8,   BASE_HEX,  NULL,                           TEI_MASK,       "", HFILL } },
856     { &hf_dlci_spare,            { "Spare",                        "iua.dlci_spare",                FT_UINT16,  BASE_HEX,  NULL,                           0x0,            "", HFILL } },
857     { &hf_diag_info,             { "Diagnostic information",       "iua.diagnostic_information",    FT_BYTES,   BASE_NONE, NULL,                           0x0,            "", HFILL } },
858     { &hf_interface_range_start, { "Start",                        "iua.interface_range_start",     FT_UINT32,  BASE_DEC,  NULL,                           0x0,            "", HFILL } },
859     { &hf_interface_range_end,   { "End",                          "iua.interface_range_end",       FT_UINT32,  BASE_DEC,  NULL,                           0x0,            "", HFILL } },
860     { &hf_heartbeat_data,        { "Heartbeat data",               "iua.heartbeat_data",            FT_BYTES,   BASE_NONE, NULL,                           0x0,            "", HFILL } },
861     { &hf_asp_reason,            { "Reason",                       "iua.asp_reason",                FT_UINT32,  BASE_HEX,  VALS(asp_reason_values),        0x0,            "", HFILL } },
862     { &hf_traffic_mode_type,     { "Traffic mode type",            "iua.traffic_mode_type",         FT_UINT32,  BASE_HEX,  VALS(traffic_mode_type_values), 0x0,            "", HFILL } },
863     { &hf_error_code,            { "Error code",                   "iua.error_code",                FT_UINT32,  BASE_DEC,  VALS(error_code_values),        0x0,            "", HFILL } },
864     { &hf_error_code_ig,         { "Error code",                   "iua.error_code",                FT_UINT32,  BASE_DEC,  VALS(error_code_ig_values),     0x0,            "", HFILL } },
865     { &hf_status_type,           { "Status type",                  "iua.status_type",               FT_UINT16,  BASE_DEC,  VALS(status_type_values),       0x0,            "", HFILL } },
866     { &hf_status_id,             { "Status identification",        "iua.status_identification",     FT_UINT16,  BASE_DEC,  NULL,                           0x0,            "", HFILL } },
867     { &hf_release_reason,        { "Reason",                       "iua.release_reason",            FT_UINT32,  BASE_HEX,  VALS(release_reason_values),    0x0,            "", HFILL } },
868     { &hf_tei_status,            { "TEI status",                   "iua.tei_status",                FT_UINT32,  BASE_HEX,  VALS(tei_status_values),        0x0,            "", HFILL } },
869     { &hf_asp_id,                { "ASP identifier",               "iua.asp_identifier",            FT_UINT32,  BASE_HEX,  NULL,                           0x0,            "", HFILL } },
870     { &hf_parameter_tag,         { "Parameter Tag",                "iua.parameter_tag",             FT_UINT16,  BASE_DEC,  VALS(parameter_tag_values),     0x0,            "", HFILL } },
871     { &hf_parameter_tag_ig,      { "Parameter Tag",                "iua.parameter_tag",             FT_UINT16,  BASE_DEC,  VALS(parameter_tag_ig_values),  0x0,            "", HFILL } },
872     { &hf_parameter_length,      { "Parameter length",             "iua.parameter_length",          FT_UINT16,  BASE_DEC,  NULL,                           0x0,            "", HFILL } },
873     { &hf_parameter_value,       { "Parameter value",              "iua.parameter_value",           FT_BYTES,   BASE_NONE, NULL,                           0x0,            "", HFILL } },
874     { &hf_parameter_padding,     { "Parameter padding",            "iua.parameter_padding",         FT_BYTES,   BASE_NONE, NULL,                           0x0,            "", HFILL } },
875     { &hf_version,               { "Version",                      "iua.version",                   FT_UINT8,   BASE_DEC,  VALS(protocol_version_values),  0x0,            "", HFILL } },
876     { &hf_reserved,              { "Reserved",                     "iua.reserved",                  FT_UINT8,   BASE_HEX,  NULL,                           0x0,            "", HFILL } },
877     { &hf_message_class,         { "Message class",                "iua.message_class",             FT_UINT8,   BASE_DEC,  VALS(message_class_values),     0x0,            "", HFILL } },
878     { &hf_message_type,          { "Message Type",                 "iua.message_type",              FT_UINT8,   BASE_DEC,  NULL,                           0x0,            "", HFILL } },
879     { &hf_message_length,        { "Message length",               "iua.message_length",            FT_UINT32,  BASE_DEC,  NULL,                           0x0,            "", HFILL } },
880    };
881   /* Setup protocol subtree array */
882   static gint *ett[] = {
883     &ett_iua,
884     &ett_iua_parameter,
885   };
886
887   /* Register the protocol name and description */
888   proto_iua = proto_register_protocol("ISDN Q.921-User Adaptation Layer", "IUA", "iua");
889   iua_module = prefs_register_protocol(proto_iua, NULL);
890
891   /* Required function calls to register the header fields and subtrees used */
892   proto_register_field_array(proto_iua, hf, array_length(hf));
893   proto_register_subtree_array(ett, array_length(ett));
894   prefs_register_bool_preference(iua_module, "support_ig", "Support Implementers Guide", "Support Implementers Guide (version 01)", &support_IG);
895 };
896
897 #define SCTP_PORT_IUA          9900
898
899 void
900 proto_reg_handoff_iua(void)
901 {
902   dissector_handle_t iua_handle;
903
904   iua_handle  = create_dissector_handle(dissect_iua, proto_iua);
905   q931_handle = find_dissector("q931");
906
907   dissector_add("sctp.port", SCTP_PORT_IUA,           iua_handle);
908   dissector_add("sctp.ppi",  IUA_PAYLOAD_PROTOCOL_ID, iua_handle);
909 }