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