]> The Tcpdump Group git mirrors - tcpdump/blobdiff - print-vrrp.c
NDOize more small decoders
[tcpdump] / print-vrrp.c
index e087d527658c719aa5ec05ecdbc2112f8dd703a2..7920699765d1d7765c29d10a98953875551e6694 100644 (file)
  * FOR A PARTICULAR PURPOSE.
  */
 
-#ifndef lint
-static const char rcsid[] =
-    "@(#) $Header: /tcpdump/master/tcpdump/print-vrrp.c,v 1.7 2003-09-05 17:42:41 hannes Exp $";
-#endif
-
 #ifdef HAVE_CONFIG_H
 #include "config.h"
 #endif
@@ -41,8 +36,11 @@ static const char rcsid[] =
 #include "extract.h"
 #include "addrtoname.h"
 
+#include "ip.h"
+#include "ipproto.h"
 /*
- * RFC 2338:
+ * RFC 2338 (VRRP v2):
+ *
  *     0                   1                   2                   3
  *     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
  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
@@ -62,6 +60,27 @@ static const char rcsid[] =
  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  *    |                     Authentication Data (2)                   |
  *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *
+ *
+ * RFC 5798 (VRRP v3):
+ *
+ *    0                   1                   2                   3
+ *    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
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |                    IPv4 Fields or IPv6 Fields                 |
+ *   ...                                                             ...
+ *    |                                                               |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |Version| Type  | Virtual Rtr ID|   Priority    |Count IPvX Addr|
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |(rsvd) |     Max Adver Int     |          Checksum             |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
+ *    |                                                               |
+ *    +                                                               +
+ *    |                       IPvX Address(es)                        |
+ *    +                                                               +
+ *    |                                                               |
+ *    +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  */
 
 /* Type */
@@ -85,9 +104,10 @@ static const struct tok auth2str[] = {
 };
 
 void
-vrrp_print(register const u_char *bp, register u_int len, int ttl)
+vrrp_print(register const u_char *bp, register u_int len,
+    register const u_char *bp2, int ttl)
 {
-       int version, type, auth_type;
+       int version, type, auth_type = VRRP_AUTH_NONE; /* keep compiler happy */
        const char *type_s;
 
        TCHECK(bp[0]);
@@ -97,22 +117,44 @@ vrrp_print(register const u_char *bp, register u_int len, int ttl)
        printf("VRRPv%u, %s", version, type_s);
        if (ttl != 255)
                printf(", (ttl %u)", ttl);
-       if (version != 2 || type != VRRP_TYPE_ADVERTISEMENT)
+       if (version < 2 || version > 3 || type != VRRP_TYPE_ADVERTISEMENT)
                return;
        TCHECK(bp[2]);
        printf(", vrid %u, prio %u", bp[1], bp[2]);
        TCHECK(bp[5]);
-       auth_type = bp[4];
-       printf(", authtype %s", tok2str(auth2str, NULL, auth_type));
-       printf(", intvl %us, length %u", bp[5],len);
+
+       if (version == 2) {
+               auth_type = bp[4];
+               printf(", authtype %s", tok2str(auth2str, NULL, auth_type));
+               printf(", intvl %us, length %u", bp[5], len);
+       } else { /* version == 3 */
+               u_int16_t intvl = (bp[4] & 0x0f) << 8 | bp[5];
+               printf(", intvl %ucs, length %u", intvl, len);
+       }
+
        if (vflag) {
                int naddrs = bp[3];
                int i;
                char c;
 
-               if (TTEST2(bp[0], len) && in_cksum((const u_short*)bp, len, 0))
-                       printf(", (bad vrrp cksum %x)",
-                               EXTRACT_16BITS(&bp[6]));
+               if (version == 2 && TTEST2(bp[0], len)) {
+                       struct cksum_vec vec[1];
+
+                       vec[0].ptr = bp;
+                       vec[0].len = len;
+                       if (in_cksum(vec, 1))
+                               printf(", (bad vrrp cksum %x)",
+                                       EXTRACT_16BITS(&bp[6]));
+               }
+
+               if (version == 3 && TTEST2(bp[0], len)) {
+                       u_int16_t cksum = nextproto4_cksum((struct ip *)bp2, bp,
+                               len, len, IPPROTO_VRRP);
+                       if (cksum)
+                               printf(", (bad vrrp cksum %x)",
+                                       EXTRACT_16BITS(&bp[6]));
+               }
+
                printf(", addrs");
                if (naddrs > 1)
                        printf("(%d)", naddrs);
@@ -125,10 +167,13 @@ vrrp_print(register const u_char *bp, register u_int len, int ttl)
                        c = ',';
                        bp += 4;
                }
-               if (auth_type == VRRP_AUTH_SIMPLE) { /* simple text password */
+               if (version == 2 && auth_type == VRRP_AUTH_SIMPLE) { /* simple text password */
                        TCHECK(bp[7]);
                        printf(" auth \"");
-                       fn_printn(bp, 8, NULL);
+                       if (fn_printn(bp, 8, snapend)) {
+                               printf("\"");
+                               goto trunc;
+                       }
                        printf("\"");
                }
        }