xref: /netbsd-src/external/bsd/tcpdump/dist/print-m3ua.c (revision 2f62cc9c12bc202c40224f32c879f81443fee079)
1 /* Copyright (c) 2013, The TCPDUMP project
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are met:
6  *
7  * 1. Redistributions of source code must retain the above copyright notice, this
8  *    list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright notice,
10  *    this list of conditions and the following disclaimer in the documentation
11  *    and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
14  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
15  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
16  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
17  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
18  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
19  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
20  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
21  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
22  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23  */
24 
25 #include <sys/cdefs.h>
26 #ifndef lint
27 __RCSID("$NetBSD: print-m3ua.c,v 1.7 2023/08/17 20:19:40 christos Exp $");
28 #endif
29 
30 /* \summary: Message Transfer Part 3 (MTP3) User Adaptation Layer (M3UA) printer */
31 
32 /* RFC 4666 */
33 
34 #ifdef HAVE_CONFIG_H
35 #include <config.h>
36 #endif
37 
38 #include "netdissect-stdinc.h"
39 
40 #define ND_LONGJMP_FROM_TCHECK
41 #include "netdissect.h"
42 #include "extract.h"
43 
44 
45 #define M3UA_REL_1_0 1
46 
47 struct m3ua_common_header {
48   nd_uint8_t  v;
49   nd_uint8_t  reserved;
50   nd_uint8_t  msg_class;
51   nd_uint8_t  msg_type;
52   nd_uint32_t len;
53 };
54 
55 struct m3ua_param_header {
56   nd_uint16_t tag;
57   nd_uint16_t len;
58 };
59 
60 /* message classes */
61 #define M3UA_MSGC_MGMT 0
62 #define M3UA_MSGC_TRANSFER 1
63 #define M3UA_MSGC_SSNM 2
64 #define M3UA_MSGC_ASPSM 3
65 #define M3UA_MSGC_ASPTM 4
66 /* reserved values */
67 #define M3UA_MSGC_RKM 9
68 
69 static const struct tok MessageClasses[] = {
70 	{ M3UA_MSGC_MGMT,     "Management"            },
71 	{ M3UA_MSGC_TRANSFER, "Transfer"              },
72 	{ M3UA_MSGC_SSNM,     "SS7"                   },
73 	{ M3UA_MSGC_ASPSM,    "ASP"                   },
74 	{ M3UA_MSGC_ASPTM,    "ASP"                   },
75 	{ M3UA_MSGC_RKM,      "Routing Key Management"},
76 	{ 0, NULL }
77 };
78 
79 /* management messages */
80 #define M3UA_MGMT_ERROR 0
81 #define M3UA_MGMT_NOTIFY 1
82 
83 static const struct tok MgmtMessages[] = {
84   { M3UA_MGMT_ERROR, "Error" },
85   { M3UA_MGMT_NOTIFY, "Notify" },
86   { 0, NULL }
87 };
88 
89 /* transfer messages */
90 #define M3UA_TRANSFER_DATA 1
91 
92 static const struct tok TransferMessages[] = {
93   { M3UA_TRANSFER_DATA, "Data" },
94   { 0, NULL }
95 };
96 
97 /* SS7 Signaling Network Management messages */
98 #define M3UA_SSNM_DUNA 1
99 #define M3UA_SSNM_DAVA 2
100 #define M3UA_SSNM_DAUD 3
101 #define M3UA_SSNM_SCON 4
102 #define M3UA_SSNM_DUPU 5
103 #define M3UA_SSNM_DRST 6
104 
105 static const struct tok SS7Messages[] = {
106   { M3UA_SSNM_DUNA, "Destination Unavailable" },
107   { M3UA_SSNM_DAVA, "Destination Available" },
108   { M3UA_SSNM_DAUD, "Destination State Audit" },
109   { M3UA_SSNM_SCON, "Signalling Congestion" },
110   { M3UA_SSNM_DUPU, "Destination User Part Unavailable" },
111   { M3UA_SSNM_DRST, "Destination Restricted" },
112   { 0, NULL }
113 };
114 
115 /* ASP State Maintenance messages */
116 #define M3UA_ASP_UP 1
117 #define M3UA_ASP_DN 2
118 #define M3UA_ASP_BEAT 3
119 #define M3UA_ASP_UP_ACK 4
120 #define M3UA_ASP_DN_ACK 5
121 #define M3UA_ASP_BEAT_ACK 6
122 
123 static const struct tok ASPStateMessages[] = {
124   { M3UA_ASP_UP, "Up" },
125   { M3UA_ASP_DN, "Down" },
126   { M3UA_ASP_BEAT, "Heartbeat" },
127   { M3UA_ASP_UP_ACK, "Up Acknowledgement" },
128   { M3UA_ASP_DN_ACK, "Down Acknowledgement" },
129   { M3UA_ASP_BEAT_ACK, "Heartbeat Acknowledgement" },
130   { 0, NULL }
131 };
132 
133 /* ASP Traffic Maintenance messages */
134 #define M3UA_ASP_AC 1
135 #define M3UA_ASP_IA 2
136 #define M3UA_ASP_AC_ACK 3
137 #define M3UA_ASP_IA_ACK 4
138 
139 static const struct tok ASPTrafficMessages[] = {
140   { M3UA_ASP_AC, "Active" },
141   { M3UA_ASP_IA, "Inactive" },
142   { M3UA_ASP_AC_ACK, "Active Acknowledgement" },
143   { M3UA_ASP_IA_ACK, "Inactive Acknowledgement" },
144   { 0, NULL }
145 };
146 
147 /* Routing Key Management messages */
148 #define M3UA_RKM_REQ 1
149 #define M3UA_RKM_RSP 2
150 #define M3UA_RKM_DEREQ 3
151 #define M3UA_RKM_DERSP 4
152 
153 static const struct tok RoutingKeyMgmtMessages[] = {
154   { M3UA_RKM_REQ, "Registration Request" },
155   { M3UA_RKM_RSP, "Registration Response" },
156   { M3UA_RKM_DEREQ, "Deregistration Request" },
157   { M3UA_RKM_DERSP, "Deregistration Response" },
158   { 0, NULL }
159 };
160 
161 static const struct uint_tokary m3ua_msgc2tokary[] = {
162 	{ M3UA_MSGC_MGMT,     MgmtMessages           },
163 	{ M3UA_MSGC_TRANSFER, TransferMessages       },
164 	{ M3UA_MSGC_SSNM,     SS7Messages            },
165 	{ M3UA_MSGC_ASPSM,    ASPStateMessages       },
166 	{ M3UA_MSGC_ASPTM,    ASPTrafficMessages     },
167 	{ M3UA_MSGC_RKM,      RoutingKeyMgmtMessages },
168 	/* uint2tokary() does not use array termination. */
169 };
170 
171 /* M3UA Parameters */
172 #define M3UA_PARAM_INFO 0x0004
173 #define M3UA_PARAM_ROUTING_CTX 0x0006
174 #define M3UA_PARAM_DIAGNOSTIC 0x0007
175 #define M3UA_PARAM_HB_DATA 0x0009
176 #define M3UA_PARAM_TRAFFIC_MODE_TYPE 0x000b
177 #define M3UA_PARAM_ERROR_CODE 0x000c
178 #define M3UA_PARAM_STATUS 0x000d
179 #define M3UA_PARAM_ASP_ID 0x0011
180 #define M3UA_PARAM_AFFECTED_POINT_CODE 0x0012
181 #define M3UA_PARAM_CORR_ID 0x0013
182 
183 #define M3UA_PARAM_NETWORK_APPEARANCE 0x0200
184 #define M3UA_PARAM_USER 0x0204
185 #define M3UA_PARAM_CONGESTION_INDICATION 0x0205
186 #define M3UA_PARAM_CONCERNED_DST 0x0206
187 #define M3UA_PARAM_ROUTING_KEY 0x0207
188 #define M3UA_PARAM_REG_RESULT 0x0208
189 #define M3UA_PARAM_DEREG_RESULT 0x0209
190 #define M3UA_PARAM_LOCAL_ROUTING_KEY_ID 0x020a
191 #define M3UA_PARAM_DST_POINT_CODE 0x020b
192 #define M3UA_PARAM_SI 0x020c
193 #define M3UA_PARAM_ORIGIN_POINT_CODE_LIST 0x020e
194 #define M3UA_PARAM_PROTO_DATA 0x0210
195 #define M3UA_PARAM_REG_STATUS 0x0212
196 #define M3UA_PARAM_DEREG_STATUS 0x0213
197 
198 static const struct tok ParamName[] = {
199   { M3UA_PARAM_INFO, "INFO String" },
200   { M3UA_PARAM_ROUTING_CTX, "Routing Context" },
201   { M3UA_PARAM_DIAGNOSTIC, "Diagnostic Info" },
202   { M3UA_PARAM_HB_DATA, "Heartbeat Data" },
203   { M3UA_PARAM_TRAFFIC_MODE_TYPE, "Traffic Mode Type" },
204   { M3UA_PARAM_ERROR_CODE, "Error Code" },
205   { M3UA_PARAM_STATUS, "Status" },
206   { M3UA_PARAM_ASP_ID, "ASP Identifier" },
207   { M3UA_PARAM_AFFECTED_POINT_CODE, "Affected Point Code" },
208   { M3UA_PARAM_CORR_ID, "Correlation ID" },
209   { M3UA_PARAM_NETWORK_APPEARANCE, "Network Appearance" },
210   { M3UA_PARAM_USER, "User/Cause" },
211   { M3UA_PARAM_CONGESTION_INDICATION, "Congestion Indications" },
212   { M3UA_PARAM_CONCERNED_DST, "Concerned Destination" },
213   { M3UA_PARAM_ROUTING_KEY, "Routing Key" },
214   { M3UA_PARAM_REG_RESULT, "Registration Result" },
215   { M3UA_PARAM_DEREG_RESULT, "Deregistration Result" },
216   { M3UA_PARAM_LOCAL_ROUTING_KEY_ID, "Local Routing Key Identifier" },
217   { M3UA_PARAM_DST_POINT_CODE, "Destination Point Code" },
218   { M3UA_PARAM_SI, "Service Indicators" },
219   { M3UA_PARAM_ORIGIN_POINT_CODE_LIST, "Originating Point Code List" },
220   { M3UA_PARAM_PROTO_DATA, "Protocol Data" },
221   { M3UA_PARAM_REG_STATUS, "Registration Status" },
222   { M3UA_PARAM_DEREG_STATUS, "Deregistration Status" },
223   { 0, NULL }
224 };
225 
226 static void
227 tag_value_print(netdissect_options *ndo,
228                 const u_char *buf, const uint16_t tag, const uint16_t size)
229 {
230   switch (tag) {
231   case M3UA_PARAM_NETWORK_APPEARANCE:
232   case M3UA_PARAM_ROUTING_CTX:
233   case M3UA_PARAM_CORR_ID:
234     /* buf and size don't include the header */
235     if (size < 4)
236       goto invalid;
237     ND_PRINT("0x%08x", GET_BE_U_4(buf));
238     break;
239   /* ... */
240   default:
241     ND_PRINT("(length %zu)", size + sizeof(struct m3ua_param_header));
242   }
243   ND_TCHECK_LEN(buf, size);
244   return;
245 
246 invalid:
247   nd_print_invalid(ndo);
248   ND_TCHECK_LEN(buf, size);
249 }
250 
251 /*
252  *     0                   1                   2                   3
253  *     0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
254  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
255  *    |          Parameter Tag        |       Parameter Length        |
256  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
257  *    \                                                               \
258  *    /                       Parameter Value                         /
259  *    \                                                               \
260  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
261  */
262 static void
263 m3ua_tags_print(netdissect_options *ndo,
264                 const u_char *buf, const u_int size)
265 {
266   const u_char *p = buf;
267   int align;
268   uint16_t hdr_tag;
269   uint16_t hdr_len;
270 
271   while (p < buf + size) {
272     if (p + sizeof(struct m3ua_param_header) > buf + size)
273       goto invalid;
274     /* Parameter Tag */
275     hdr_tag = GET_BE_U_2(p);
276     ND_PRINT("\n\t\t\t%s: ", tok2str(ParamName, "Unknown Parameter (0x%04x)", hdr_tag));
277     /* Parameter Length */
278     hdr_len = GET_BE_U_2(p + 2);
279     if (hdr_len < sizeof(struct m3ua_param_header))
280       goto invalid;
281     /* Parameter Value */
282     align = (p + hdr_len - buf) % 4;
283     align = align ? 4 - align : 0;
284     ND_TCHECK_LEN(p, hdr_len + align);
285     tag_value_print(ndo, p, hdr_tag, hdr_len - sizeof(struct m3ua_param_header));
286     p += hdr_len + align;
287   }
288   return;
289 
290 invalid:
291   nd_print_invalid(ndo);
292   ND_TCHECK_LEN(buf, size);
293 }
294 
295 /*
296  *     0                   1                   2                   3
297  *     0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
298  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
299  *    |    Version    |   Reserved    | Message Class | Message Type  |
300  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
301  *    |                        Message Length                         |
302  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
303  *    \                                                               \
304  *    /                                                               /
305  */
306 UNALIGNED_OK
307 void
308 m3ua_print(netdissect_options *ndo,
309            const u_char *buf, const u_int size)
310 {
311   const struct m3ua_common_header *hdr = (const struct m3ua_common_header *) buf;
312   const struct tok *dict;
313   uint8_t msg_class;
314 
315   ndo->ndo_protocol = "m3ua";
316   /* size includes the header */
317   if (size < sizeof(struct m3ua_common_header))
318     goto invalid;
319   ND_TCHECK_SIZE(hdr);
320   if (GET_U_1(hdr->v) != M3UA_REL_1_0)
321     return;
322 
323   msg_class = GET_U_1(hdr->msg_class);
324   dict = uint2tokary(m3ua_msgc2tokary, msg_class);
325 
326   ND_PRINT("\n\t\t%s", tok2str(MessageClasses, "Unknown message class %i", msg_class));
327   if (dict != NULL)
328     ND_PRINT(" %s Message",
329              tok2str(dict, "Unknown (0x%02x)", GET_U_1(hdr->msg_type)));
330 
331   if (size != GET_BE_U_4(hdr->len))
332     ND_PRINT("\n\t\t\t@@@@@@ Corrupted length %u of message @@@@@@",
333              GET_BE_U_4(hdr->len));
334   else
335     m3ua_tags_print(ndo, buf + sizeof(struct m3ua_common_header),
336                     GET_BE_U_4(hdr->len) - sizeof(struct m3ua_common_header));
337   return;
338 
339 invalid:
340   nd_print_invalid(ndo);
341   ND_TCHECK_LEN(buf, size);
342 }
343