]> The Tcpdump Group git mirrors - tcpdump/blobdiff - print-mpls.c
add tracefiles for infinite loop testing
[tcpdump] / print-mpls.c
index 69afa17f074728c42fcd25cc42b9f9fa4a7a361a..1335a1b24341892d6969972df01b3622ef22f897 100644 (file)
  */
 
 #ifndef lint
-static const char rcsid[] =
-    "@(#) $Header: /tcpdump/master/tcpdump/print-mpls.c,v 1.1 2001-06-21 17:56:04 itojun Exp $ (LBL)";
+static const char rcsid[] _U_ =
+    "@(#) $Header: /tcpdump/master/tcpdump/print-mpls.c,v 1.8.2.2 2003-11-16 08:51:34 guy Exp $ (LBL)";
 #endif
 
 #ifdef HAVE_CONFIG_H
 #include "config.h"
 #endif
 
-#include <sys/param.h>
-#include <sys/time.h>
-#include <sys/socket.h>
+#include <tcpdump-stdinc.h>
 
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
-#include <unistd.h>
 
 #include "addrtoname.h"
 #include "interface.h"
@@ -81,26 +78,27 @@ mpls_print(const u_char *bp, u_int length)
 
        p = bp;
        printf("MPLS");
-       TCHECK2(*p, sizeof(v));
-       memcpy(&v, p, sizeof(v));
-       v = (u_int32_t)ntohl(v);
-       printf(" (");   /*)*/
-       printf("label 0x%x", MPLS_LABEL(v));
-       if (vflag &&
-           MPLS_LABEL(v) < sizeof(mpls_labelname) / sizeof(mpls_labelname[0]))
-               printf("(%s)", mpls_labelname[MPLS_LABEL(v)]);
-       if (MPLS_EXP(v))
-               printf(" exp 0x%x", MPLS_EXP(v));
-       if (MPLS_STACK(v))
-               printf("[S]");
-       printf(" TTL %u", MPLS_TTL(v));
-       /*(*/
-       printf(")");
+       do {
+               TCHECK2(*p, sizeof(v));
+               v = EXTRACT_32BITS(p);
+               printf(" (");   /*)*/
+               printf("label %u", MPLS_LABEL(v));
+               if (vflag &&
+                   MPLS_LABEL(v) < sizeof(mpls_labelname) / sizeof(mpls_labelname[0]))
+                       printf(" (%s)", mpls_labelname[MPLS_LABEL(v)]);
+               printf(", exp %u", MPLS_EXP(v));
+               if (MPLS_STACK(v))
+                       printf(", [S]");
+               printf(", ttl %u", MPLS_TTL(v));
+               /*(*/
+               printf(")");
 
-       p += sizeof(v);
+               p += sizeof(v);
+       } while (!MPLS_STACK(v));
 
        switch (MPLS_LABEL(v)) {
        case 0: /* IPv4 explicit NULL label */
+        case 3:        /* IPv4 implicit NULL label */
                ip_print(p, length - (p - bp));
                break;
 #ifdef INET6
@@ -110,12 +108,89 @@ mpls_print(const u_char *bp, u_int length)
 #endif
        default:
                /*
-                * Since there's no indication of protocol in MPLS label
-                * encoding, we can print nothing further.
+                * Generally there's no indication of protocol in MPLS label
+                * encoding, however draft-hsmit-isis-aal5mux-00.txt describes
+                 * a technique that looks at the first payload byte if the BOS (bottom of stack)
+                 * bit is set and tries to determine the network layer protocol
+                 * 0x45-0x4f is IPv4
+                 * 0x60-0x6f is IPv6
+                 * 0x81-0x83 is OSI (CLNP,ES-IS,IS-IS)
+                 * this technique is sometimes known as NULL encapsulation
+                 * and decoding is particularly useful for control-plane traffic [BGP]
+                 * which cisco by default sends MPLS encapsulated
                 */
-               return;
+
+                if (MPLS_STACK(v)) { /* only do this if the stack bit is set */
+                    switch(*p) {
+                    case 0x45:
+                    case 0x46:
+                    case 0x47:
+                    case 0x48:
+                    case 0x49:
+                    case 0x4a:
+                    case 0x4b:
+                    case 0x4c:
+                    case 0x4d:
+                    case 0x4e:
+                    case 0x4f:
+                       if (vflag>0) {
+                            printf("\n\t");
+                            ip_print(p, length - (p - bp));
+                           }
+                        else printf(", IP, length: %u",length);
+                        break;
+#ifdef INET6
+                    case 0x60:
+                    case 0x61:
+                    case 0x62:
+                    case 0x63:
+                    case 0x64:
+                    case 0x65:
+                    case 0x66:
+                    case 0x67:
+                    case 0x68:
+                    case 0x69:
+                    case 0x6a:
+                    case 0x6b:
+                    case 0x6c:
+                    case 0x6d:
+                    case 0x6e:
+                    case 0x6f:
+                       if (vflag>0) {
+                            printf("\n\t");
+                            ip6_print(p, length - (p - bp));
+                            }
+                       else printf(", IPv6, length: %u",length);
+                        break;
+#endif
+                    case 0x81:
+                    case 0x82:
+                    case 0x83:
+                       if (vflag>0) {
+                            printf("\n\t");
+                            isoclns_print(p, length - (p - bp), length - (p - bp));
+                           }
+                       else printf(", OSI, length: %u",length);
+                        break;
+                    default:
+                        /* ok bail out - we did not figure out what it is*/
+                        break;
+                    }
+                }
+                return;
        }
 
 trunc:
        printf("[|MPLS]");
 }
+
+/*
+ * draft-ietf-mpls-lsp-ping-02.txt
+ */
+void
+mpls_lsp_ping_print(const u_char *pptr, u_int length)
+{
+    printf("UDP, LSP-PING, length: %u", length);
+    if (vflag >1)
+       print_unknown_data(pptr,"\n\t  ", length);
+}