xref: /minix3/external/bsd/tcpdump/dist/print-openflow-1.0.c (revision b636d99d91c3d54204248f643c14627405d4afd1)
1*b636d99dSDavid van Moolenbroek /*
2*b636d99dSDavid van Moolenbroek  * This module implements decoding of OpenFlow protocol version 1.0 (wire
3*b636d99dSDavid van Moolenbroek  * protocol 0x01). The decoder implements terse (default), detailed (-v) and
4*b636d99dSDavid van Moolenbroek  * full (-vv) output formats and, as much as each format implies, detects and
5*b636d99dSDavid van Moolenbroek  * tries to work around sizing anomalies inside the messages. The decoder marks
6*b636d99dSDavid van Moolenbroek  * up bogus values of selected message fields and decodes partially captured
7*b636d99dSDavid van Moolenbroek  * messages up to the snapshot end. It is based on the specification below:
8*b636d99dSDavid van Moolenbroek  *
9*b636d99dSDavid van Moolenbroek  * [OF10] http://www.openflow.org/documents/openflow-spec-v1.0.0.pdf
10*b636d99dSDavid van Moolenbroek  *
11*b636d99dSDavid van Moolenbroek  * Most functions in this file take 3 arguments into account:
12*b636d99dSDavid van Moolenbroek  * * cp -- the pointer to the first octet to decode
13*b636d99dSDavid van Moolenbroek  * * len -- the length of the current structure as declared on the wire
14*b636d99dSDavid van Moolenbroek  * * ep -- the pointer to the end of the captured frame
15*b636d99dSDavid van Moolenbroek  * They return either the pointer to the next not-yet-decoded part of the frame
16*b636d99dSDavid van Moolenbroek  * or the value of ep, which means the current frame processing is over as it
17*b636d99dSDavid van Moolenbroek  * has been fully decoded or is malformed or truncated. This way it is possible
18*b636d99dSDavid van Moolenbroek  * to chain and nest such functions uniformly to decode an OF1.0 message, which
19*b636d99dSDavid van Moolenbroek  * consists of several layers of nested structures.
20*b636d99dSDavid van Moolenbroek  *
21*b636d99dSDavid van Moolenbroek  * Decoding of Ethernet frames nested in OFPT_PACKET_IN and OFPT_PACKET_OUT
22*b636d99dSDavid van Moolenbroek  * messages is done only when the verbosity level set by command-line argument
23*b636d99dSDavid van Moolenbroek  * is "-vvv" or higher. In that case the verbosity level is temporarily
24*b636d99dSDavid van Moolenbroek  * decremented by 3 during the nested frame decoding. For example, running
25*b636d99dSDavid van Moolenbroek  * tcpdump with "-vvvv" will do full decoding of OpenFlow and "-v" decoding of
26*b636d99dSDavid van Moolenbroek  * the nested frames.
27*b636d99dSDavid van Moolenbroek  *
28*b636d99dSDavid van Moolenbroek  * Partial decoding of Big Switch Networks vendor extensions is done after the
29*b636d99dSDavid van Moolenbroek  * oftest (OpenFlow Testing Framework) and Loxigen (library generator) source
30*b636d99dSDavid van Moolenbroek  * code.
31*b636d99dSDavid van Moolenbroek  *
32*b636d99dSDavid van Moolenbroek  *
33*b636d99dSDavid van Moolenbroek  * Copyright (c) 2013 The TCPDUMP project
34*b636d99dSDavid van Moolenbroek  * All rights reserved.
35*b636d99dSDavid van Moolenbroek  *
36*b636d99dSDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
37*b636d99dSDavid van Moolenbroek  * modification, are permitted provided that the following conditions
38*b636d99dSDavid van Moolenbroek  * are met:
39*b636d99dSDavid van Moolenbroek  * 1. Redistributions of source code must retain the above copyright
40*b636d99dSDavid van Moolenbroek  *    notice, this list of conditions and the following disclaimer.
41*b636d99dSDavid van Moolenbroek  * 2. Redistributions in binary form must reproduce the above copyright
42*b636d99dSDavid van Moolenbroek  *    notice, this list of conditions and the following disclaimer in the
43*b636d99dSDavid van Moolenbroek  *    documentation and/or other materials provided with the distribution.
44*b636d99dSDavid van Moolenbroek  *
45*b636d99dSDavid van Moolenbroek  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
46*b636d99dSDavid van Moolenbroek  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
47*b636d99dSDavid van Moolenbroek  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
48*b636d99dSDavid van Moolenbroek  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
49*b636d99dSDavid van Moolenbroek  * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
50*b636d99dSDavid van Moolenbroek  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
51*b636d99dSDavid van Moolenbroek  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
52*b636d99dSDavid van Moolenbroek  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
53*b636d99dSDavid van Moolenbroek  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
54*b636d99dSDavid van Moolenbroek  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
55*b636d99dSDavid van Moolenbroek  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
56*b636d99dSDavid van Moolenbroek  * POSSIBILITY OF SUCH DAMAGE.
57*b636d99dSDavid van Moolenbroek  */
58*b636d99dSDavid van Moolenbroek 
59*b636d99dSDavid van Moolenbroek #define NETDISSECT_REWORKED
60*b636d99dSDavid van Moolenbroek #ifdef HAVE_CONFIG_H
61*b636d99dSDavid van Moolenbroek #include "config.h"
62*b636d99dSDavid van Moolenbroek #endif
63*b636d99dSDavid van Moolenbroek 
64*b636d99dSDavid van Moolenbroek #include <tcpdump-stdinc.h>
65*b636d99dSDavid van Moolenbroek 
66*b636d99dSDavid van Moolenbroek #include "interface.h"
67*b636d99dSDavid van Moolenbroek #include "extract.h"
68*b636d99dSDavid van Moolenbroek #include "addrtoname.h"
69*b636d99dSDavid van Moolenbroek #include "ether.h"
70*b636d99dSDavid van Moolenbroek #include "ethertype.h"
71*b636d99dSDavid van Moolenbroek #include "ipproto.h"
72*b636d99dSDavid van Moolenbroek #include "oui.h"
73*b636d99dSDavid van Moolenbroek #include "openflow.h"
74*b636d99dSDavid van Moolenbroek 
75*b636d99dSDavid van Moolenbroek static const char tstr[] = " [|openflow]";
76*b636d99dSDavid van Moolenbroek static const char cstr[] = " (corrupt)";
77*b636d99dSDavid van Moolenbroek 
78*b636d99dSDavid van Moolenbroek #define OFPT_HELLO                    0x00
79*b636d99dSDavid van Moolenbroek #define OFPT_ERROR                    0x01
80*b636d99dSDavid van Moolenbroek #define OFPT_ECHO_REQUEST             0x02
81*b636d99dSDavid van Moolenbroek #define OFPT_ECHO_REPLY               0x03
82*b636d99dSDavid van Moolenbroek #define OFPT_VENDOR                   0x04
83*b636d99dSDavid van Moolenbroek #define OFPT_FEATURES_REQUEST         0x05
84*b636d99dSDavid van Moolenbroek #define OFPT_FEATURES_REPLY           0x06
85*b636d99dSDavid van Moolenbroek #define OFPT_GET_CONFIG_REQUEST       0x07
86*b636d99dSDavid van Moolenbroek #define OFPT_GET_CONFIG_REPLY         0x08
87*b636d99dSDavid van Moolenbroek #define OFPT_SET_CONFIG               0x09
88*b636d99dSDavid van Moolenbroek #define OFPT_PACKET_IN                0x0a
89*b636d99dSDavid van Moolenbroek #define OFPT_FLOW_REMOVED             0x0b
90*b636d99dSDavid van Moolenbroek #define OFPT_PORT_STATUS              0x0c
91*b636d99dSDavid van Moolenbroek #define OFPT_PACKET_OUT               0x0d
92*b636d99dSDavid van Moolenbroek #define OFPT_FLOW_MOD                 0x0e
93*b636d99dSDavid van Moolenbroek #define OFPT_PORT_MOD                 0x0f
94*b636d99dSDavid van Moolenbroek #define OFPT_STATS_REQUEST            0x10
95*b636d99dSDavid van Moolenbroek #define OFPT_STATS_REPLY              0x11
96*b636d99dSDavid van Moolenbroek #define OFPT_BARRIER_REQUEST          0x12
97*b636d99dSDavid van Moolenbroek #define OFPT_BARRIER_REPLY            0x13
98*b636d99dSDavid van Moolenbroek #define OFPT_QUEUE_GET_CONFIG_REQUEST 0x14
99*b636d99dSDavid van Moolenbroek #define OFPT_QUEUE_GET_CONFIG_REPLY   0x15
100*b636d99dSDavid van Moolenbroek static const struct tok ofpt_str[] = {
101*b636d99dSDavid van Moolenbroek 	{ OFPT_HELLO,                    "HELLO"                    },
102*b636d99dSDavid van Moolenbroek 	{ OFPT_ERROR,                    "ERROR"                    },
103*b636d99dSDavid van Moolenbroek 	{ OFPT_ECHO_REQUEST,             "ECHO_REQUEST"             },
104*b636d99dSDavid van Moolenbroek 	{ OFPT_ECHO_REPLY,               "ECHO_REPLY"               },
105*b636d99dSDavid van Moolenbroek 	{ OFPT_VENDOR,                   "VENDOR"                   },
106*b636d99dSDavid van Moolenbroek 	{ OFPT_FEATURES_REQUEST,         "FEATURES_REQUEST"         },
107*b636d99dSDavid van Moolenbroek 	{ OFPT_FEATURES_REPLY,           "FEATURES_REPLY"           },
108*b636d99dSDavid van Moolenbroek 	{ OFPT_GET_CONFIG_REQUEST,       "GET_CONFIG_REQUEST"       },
109*b636d99dSDavid van Moolenbroek 	{ OFPT_GET_CONFIG_REPLY,         "GET_CONFIG_REPLY"         },
110*b636d99dSDavid van Moolenbroek 	{ OFPT_SET_CONFIG,               "SET_CONFIG"               },
111*b636d99dSDavid van Moolenbroek 	{ OFPT_PACKET_IN,                "PACKET_IN"                },
112*b636d99dSDavid van Moolenbroek 	{ OFPT_FLOW_REMOVED,             "FLOW_REMOVED"             },
113*b636d99dSDavid van Moolenbroek 	{ OFPT_PORT_STATUS,              "PORT_STATUS"              },
114*b636d99dSDavid van Moolenbroek 	{ OFPT_PACKET_OUT,               "PACKET_OUT"               },
115*b636d99dSDavid van Moolenbroek 	{ OFPT_FLOW_MOD,                 "FLOW_MOD"                 },
116*b636d99dSDavid van Moolenbroek 	{ OFPT_PORT_MOD,                 "PORT_MOD"                 },
117*b636d99dSDavid van Moolenbroek 	{ OFPT_STATS_REQUEST,            "STATS_REQUEST"            },
118*b636d99dSDavid van Moolenbroek 	{ OFPT_STATS_REPLY,              "STATS_REPLY"              },
119*b636d99dSDavid van Moolenbroek 	{ OFPT_BARRIER_REQUEST,          "BARRIER_REQUEST"          },
120*b636d99dSDavid van Moolenbroek 	{ OFPT_BARRIER_REPLY,            "BARRIER_REPLY"            },
121*b636d99dSDavid van Moolenbroek 	{ OFPT_QUEUE_GET_CONFIG_REQUEST, "QUEUE_GET_CONFIG_REQUEST" },
122*b636d99dSDavid van Moolenbroek 	{ OFPT_QUEUE_GET_CONFIG_REPLY,   "QUEUE_GET_CONFIG_REPLY"   },
123*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
124*b636d99dSDavid van Moolenbroek };
125*b636d99dSDavid van Moolenbroek 
126*b636d99dSDavid van Moolenbroek #define OFPPC_PORT_DOWN    (1 << 0)
127*b636d99dSDavid van Moolenbroek #define OFPPC_NO_STP       (1 << 1)
128*b636d99dSDavid van Moolenbroek #define OFPPC_NO_RECV      (1 << 2)
129*b636d99dSDavid van Moolenbroek #define OFPPC_NO_RECV_STP  (1 << 3)
130*b636d99dSDavid van Moolenbroek #define OFPPC_NO_FLOOD     (1 << 4)
131*b636d99dSDavid van Moolenbroek #define OFPPC_NO_FWD       (1 << 5)
132*b636d99dSDavid van Moolenbroek #define OFPPC_NO_PACKET_IN (1 << 6)
133*b636d99dSDavid van Moolenbroek static const struct tok ofppc_bm[] = {
134*b636d99dSDavid van Moolenbroek 	{ OFPPC_PORT_DOWN,    "PORT_DOWN"    },
135*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_STP,       "NO_STP"       },
136*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_RECV,      "NO_RECV"      },
137*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_RECV_STP,  "NO_RECV_STP"  },
138*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_FLOOD,     "NO_FLOOD"     },
139*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_FWD,       "NO_FWD"       },
140*b636d99dSDavid van Moolenbroek 	{ OFPPC_NO_PACKET_IN, "NO_PACKET_IN" },
141*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
142*b636d99dSDavid van Moolenbroek };
143*b636d99dSDavid van Moolenbroek #define OFPPC_U (~(OFPPC_PORT_DOWN | OFPPC_NO_STP | OFPPC_NO_RECV | \
144*b636d99dSDavid van Moolenbroek                    OFPPC_NO_RECV_STP | OFPPC_NO_FLOOD | OFPPC_NO_FWD | \
145*b636d99dSDavid van Moolenbroek                    OFPPC_NO_PACKET_IN))
146*b636d99dSDavid van Moolenbroek 
147*b636d99dSDavid van Moolenbroek #define OFPPS_LINK_DOWN   (1 << 0)
148*b636d99dSDavid van Moolenbroek #define OFPPS_STP_LISTEN  (0 << 8)
149*b636d99dSDavid van Moolenbroek #define OFPPS_STP_LEARN   (1 << 8)
150*b636d99dSDavid van Moolenbroek #define OFPPS_STP_FORWARD (2 << 8)
151*b636d99dSDavid van Moolenbroek #define OFPPS_STP_BLOCK   (3 << 8)
152*b636d99dSDavid van Moolenbroek #define OFPPS_STP_MASK    (3 << 8)
153*b636d99dSDavid van Moolenbroek static const struct tok ofpps_bm[] = {
154*b636d99dSDavid van Moolenbroek 	{ OFPPS_LINK_DOWN,   "LINK_DOWN"   },
155*b636d99dSDavid van Moolenbroek 	{ OFPPS_STP_LISTEN,  "STP_LISTEN"  },
156*b636d99dSDavid van Moolenbroek 	{ OFPPS_STP_LEARN,   "STP_LEARN"   },
157*b636d99dSDavid van Moolenbroek 	{ OFPPS_STP_FORWARD, "STP_FORWARD" },
158*b636d99dSDavid van Moolenbroek 	{ OFPPS_STP_BLOCK,   "STP_BLOCK"   },
159*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
160*b636d99dSDavid van Moolenbroek };
161*b636d99dSDavid van Moolenbroek #define OFPPS_U (~(OFPPS_LINK_DOWN | OFPPS_STP_LISTEN | OFPPS_STP_LEARN | \
162*b636d99dSDavid van Moolenbroek                    OFPPS_STP_FORWARD | OFPPS_STP_BLOCK))
163*b636d99dSDavid van Moolenbroek 
164*b636d99dSDavid van Moolenbroek #define OFPP_MAX        0xff00
165*b636d99dSDavid van Moolenbroek #define OFPP_IN_PORT    0xfff8
166*b636d99dSDavid van Moolenbroek #define OFPP_TABLE      0xfff9
167*b636d99dSDavid van Moolenbroek #define OFPP_NORMAL     0xfffa
168*b636d99dSDavid van Moolenbroek #define OFPP_FLOOD      0xfffb
169*b636d99dSDavid van Moolenbroek #define OFPP_ALL        0xfffc
170*b636d99dSDavid van Moolenbroek #define OFPP_CONTROLLER 0xfffd
171*b636d99dSDavid van Moolenbroek #define OFPP_LOCAL      0xfffe
172*b636d99dSDavid van Moolenbroek #define OFPP_NONE       0xffff
173*b636d99dSDavid van Moolenbroek static const struct tok ofpp_str[] = {
174*b636d99dSDavid van Moolenbroek 	{ OFPP_MAX,        "MAX"        },
175*b636d99dSDavid van Moolenbroek 	{ OFPP_IN_PORT,    "IN_PORT"    },
176*b636d99dSDavid van Moolenbroek 	{ OFPP_TABLE,      "TABLE"      },
177*b636d99dSDavid van Moolenbroek 	{ OFPP_NORMAL,     "NORMAL"     },
178*b636d99dSDavid van Moolenbroek 	{ OFPP_FLOOD,      "FLOOD"      },
179*b636d99dSDavid van Moolenbroek 	{ OFPP_ALL,        "ALL"        },
180*b636d99dSDavid van Moolenbroek 	{ OFPP_CONTROLLER, "CONTROLLER" },
181*b636d99dSDavid van Moolenbroek 	{ OFPP_LOCAL,      "LOCAL"      },
182*b636d99dSDavid van Moolenbroek 	{ OFPP_NONE,       "NONE"       },
183*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
184*b636d99dSDavid van Moolenbroek };
185*b636d99dSDavid van Moolenbroek 
186*b636d99dSDavid van Moolenbroek #define OFPPF_10MB_HD    (1 <<  0)
187*b636d99dSDavid van Moolenbroek #define OFPPF_10MB_FD    (1 <<  1)
188*b636d99dSDavid van Moolenbroek #define OFPPF_100MB_HD   (1 <<  2)
189*b636d99dSDavid van Moolenbroek #define OFPPF_100MB_FD   (1 <<  3)
190*b636d99dSDavid van Moolenbroek #define OFPPF_1GB_HD     (1 <<  4)
191*b636d99dSDavid van Moolenbroek #define OFPPF_1GB_FD     (1 <<  5)
192*b636d99dSDavid van Moolenbroek #define OFPPF_10GB_FD    (1 <<  6)
193*b636d99dSDavid van Moolenbroek #define OFPPF_COPPER     (1 <<  7)
194*b636d99dSDavid van Moolenbroek #define OFPPF_FIBER      (1 <<  8)
195*b636d99dSDavid van Moolenbroek #define OFPPF_AUTONEG    (1 <<  9)
196*b636d99dSDavid van Moolenbroek #define OFPPF_PAUSE      (1 << 10)
197*b636d99dSDavid van Moolenbroek #define OFPPF_PAUSE_ASYM (1 << 11)
198*b636d99dSDavid van Moolenbroek static const struct tok ofppf_bm[] = {
199*b636d99dSDavid van Moolenbroek 	{ OFPPF_10MB_HD,    "10MB_HD"    },
200*b636d99dSDavid van Moolenbroek 	{ OFPPF_10MB_FD,    "10MB_FD"    },
201*b636d99dSDavid van Moolenbroek 	{ OFPPF_100MB_HD,   "100MB_HD"   },
202*b636d99dSDavid van Moolenbroek 	{ OFPPF_100MB_FD,   "100MB_FD"   },
203*b636d99dSDavid van Moolenbroek 	{ OFPPF_1GB_HD,     "1GB_HD"     },
204*b636d99dSDavid van Moolenbroek 	{ OFPPF_1GB_FD,     "1GB_FD"     },
205*b636d99dSDavid van Moolenbroek 	{ OFPPF_10GB_FD,    "10GB_FD"    },
206*b636d99dSDavid van Moolenbroek 	{ OFPPF_COPPER,     "COPPER"     },
207*b636d99dSDavid van Moolenbroek 	{ OFPPF_FIBER,      "FIBER"      },
208*b636d99dSDavid van Moolenbroek 	{ OFPPF_AUTONEG,    "AUTONEG"    },
209*b636d99dSDavid van Moolenbroek 	{ OFPPF_PAUSE,      "PAUSE"      },
210*b636d99dSDavid van Moolenbroek 	{ OFPPF_PAUSE_ASYM, "PAUSE_ASYM" },
211*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
212*b636d99dSDavid van Moolenbroek };
213*b636d99dSDavid van Moolenbroek #define OFPPF_U (~(OFPPF_10MB_HD | OFPPF_10MB_FD | OFPPF_100MB_HD | \
214*b636d99dSDavid van Moolenbroek                    OFPPF_100MB_FD | OFPPF_1GB_HD | OFPPF_1GB_FD | \
215*b636d99dSDavid van Moolenbroek                    OFPPF_10GB_FD | OFPPF_COPPER | OFPPF_FIBER | \
216*b636d99dSDavid van Moolenbroek                    OFPPF_AUTONEG | OFPPF_PAUSE | OFPPF_PAUSE_ASYM))
217*b636d99dSDavid van Moolenbroek 
218*b636d99dSDavid van Moolenbroek #define OFPQT_NONE     0x0000
219*b636d99dSDavid van Moolenbroek #define OFPQT_MIN_RATE 0x0001
220*b636d99dSDavid van Moolenbroek static const struct tok ofpqt_str[] = {
221*b636d99dSDavid van Moolenbroek 	{ OFPQT_NONE,     "NONE"     },
222*b636d99dSDavid van Moolenbroek 	{ OFPQT_MIN_RATE, "MIN_RATE" },
223*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
224*b636d99dSDavid van Moolenbroek };
225*b636d99dSDavid van Moolenbroek 
226*b636d99dSDavid van Moolenbroek #define OFPFW_IN_PORT      (1 << 0)
227*b636d99dSDavid van Moolenbroek #define OFPFW_DL_VLAN      (1 << 1)
228*b636d99dSDavid van Moolenbroek #define OFPFW_DL_SRC       (1 << 2)
229*b636d99dSDavid van Moolenbroek #define OFPFW_DL_DST       (1 << 3)
230*b636d99dSDavid van Moolenbroek #define OFPFW_DL_TYPE      (1 << 4)
231*b636d99dSDavid van Moolenbroek #define OFPFW_NW_PROTO     (1 << 5)
232*b636d99dSDavid van Moolenbroek #define OFPFW_TP_SRC       (1 << 6)
233*b636d99dSDavid van Moolenbroek #define OFPFW_TP_DST       (1 << 7)
234*b636d99dSDavid van Moolenbroek #define OFPFW_NW_SRC_SHIFT 8
235*b636d99dSDavid van Moolenbroek #define OFPFW_NW_SRC_BITS  6
236*b636d99dSDavid van Moolenbroek #define OFPFW_NW_SRC_MASK  (((1 << OFPFW_NW_SRC_BITS) - 1) << OFPFW_NW_SRC_SHIFT)
237*b636d99dSDavid van Moolenbroek #define OFPFW_NW_DST_SHIFT 14
238*b636d99dSDavid van Moolenbroek #define OFPFW_NW_DST_BITS  6
239*b636d99dSDavid van Moolenbroek #define OFPFW_NW_DST_MASK  (((1 << OFPFW_NW_DST_BITS) - 1) << OFPFW_NW_DST_SHIFT)
240*b636d99dSDavid van Moolenbroek #define OFPFW_DL_VLAN_PCP  (1 << 20)
241*b636d99dSDavid van Moolenbroek #define OFPFW_NW_TOS       (1 << 21)
242*b636d99dSDavid van Moolenbroek #define OFPFW_ALL          ((1 << 22) - 1)
243*b636d99dSDavid van Moolenbroek static const struct tok ofpfw_bm[] = {
244*b636d99dSDavid van Moolenbroek 	{ OFPFW_IN_PORT,     "IN_PORT"     },
245*b636d99dSDavid van Moolenbroek 	{ OFPFW_DL_VLAN,     "DL_VLAN"     },
246*b636d99dSDavid van Moolenbroek 	{ OFPFW_DL_SRC,      "DL_SRC"      },
247*b636d99dSDavid van Moolenbroek 	{ OFPFW_DL_DST,      "DL_DST"      },
248*b636d99dSDavid van Moolenbroek 	{ OFPFW_DL_TYPE,     "DL_TYPE"     },
249*b636d99dSDavid van Moolenbroek 	{ OFPFW_NW_PROTO,    "NW_PROTO"    },
250*b636d99dSDavid van Moolenbroek 	{ OFPFW_TP_SRC,      "TP_SRC"      },
251*b636d99dSDavid van Moolenbroek 	{ OFPFW_TP_DST,      "TP_DST"      },
252*b636d99dSDavid van Moolenbroek 	{ OFPFW_DL_VLAN_PCP, "DL_VLAN_PCP" },
253*b636d99dSDavid van Moolenbroek 	{ OFPFW_NW_TOS,      "NW_TOS"      },
254*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
255*b636d99dSDavid van Moolenbroek };
256*b636d99dSDavid van Moolenbroek /* The above array does not include bits 8~13 (OFPFW_NW_SRC_*) and 14~19
257*b636d99dSDavid van Moolenbroek  * (OFPFW_NW_DST_*), which are not a part of the bitmap and require decoding
258*b636d99dSDavid van Moolenbroek  * other than that of tok2str(). The macro below includes these bits such that
259*b636d99dSDavid van Moolenbroek  * they are not reported as bogus in the decoding. */
260*b636d99dSDavid van Moolenbroek #define OFPFW_U (~(OFPFW_ALL))
261*b636d99dSDavid van Moolenbroek 
262*b636d99dSDavid van Moolenbroek #define OFPAT_OUTPUT       0x0000
263*b636d99dSDavid van Moolenbroek #define OFPAT_SET_VLAN_VID 0x0001
264*b636d99dSDavid van Moolenbroek #define OFPAT_SET_VLAN_PCP 0x0002
265*b636d99dSDavid van Moolenbroek #define OFPAT_STRIP_VLAN   0x0003
266*b636d99dSDavid van Moolenbroek #define OFPAT_SET_DL_SRC   0x0004
267*b636d99dSDavid van Moolenbroek #define OFPAT_SET_DL_DST   0x0005
268*b636d99dSDavid van Moolenbroek #define OFPAT_SET_NW_SRC   0x0006
269*b636d99dSDavid van Moolenbroek #define OFPAT_SET_NW_DST   0x0007
270*b636d99dSDavid van Moolenbroek #define OFPAT_SET_NW_TOS   0x0008
271*b636d99dSDavid van Moolenbroek #define OFPAT_SET_TP_SRC   0x0009
272*b636d99dSDavid van Moolenbroek #define OFPAT_SET_TP_DST   0x000a
273*b636d99dSDavid van Moolenbroek #define OFPAT_ENQUEUE      0x000b
274*b636d99dSDavid van Moolenbroek #define OFPAT_VENDOR       0xffff
275*b636d99dSDavid van Moolenbroek static const struct tok ofpat_str[] = {
276*b636d99dSDavid van Moolenbroek 	{ OFPAT_OUTPUT,       "OUTPUT"       },
277*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_VLAN_VID, "SET_VLAN_VID" },
278*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_VLAN_PCP, "SET_VLAN_PCP" },
279*b636d99dSDavid van Moolenbroek 	{ OFPAT_STRIP_VLAN,   "STRIP_VLAN"   },
280*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_DL_SRC,   "SET_DL_SRC"   },
281*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_DL_DST,   "SET_DL_DST"   },
282*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_NW_SRC,   "SET_NW_SRC"   },
283*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_NW_DST,   "SET_NW_DST"   },
284*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_NW_TOS,   "SET_NW_TOS"   },
285*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_TP_SRC,   "SET_TP_SRC"   },
286*b636d99dSDavid van Moolenbroek 	{ OFPAT_SET_TP_DST,   "SET_TP_DST"   },
287*b636d99dSDavid van Moolenbroek 	{ OFPAT_ENQUEUE,      "ENQUEUE"      },
288*b636d99dSDavid van Moolenbroek 	{ OFPAT_VENDOR,       "VENDOR"       },
289*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
290*b636d99dSDavid van Moolenbroek };
291*b636d99dSDavid van Moolenbroek 
292*b636d99dSDavid van Moolenbroek /* bit-shifted, w/o vendor action */
293*b636d99dSDavid van Moolenbroek static const struct tok ofpat_bm[] = {
294*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_OUTPUT,       "OUTPUT"       },
295*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_VLAN_VID, "SET_VLAN_VID" },
296*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_VLAN_PCP, "SET_VLAN_PCP" },
297*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_STRIP_VLAN,   "STRIP_VLAN"   },
298*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_DL_SRC,   "SET_DL_SRC"   },
299*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_DL_DST,   "SET_DL_DST"   },
300*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_NW_SRC,   "SET_NW_SRC"   },
301*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_NW_DST,   "SET_NW_DST"   },
302*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_NW_TOS,   "SET_NW_TOS"   },
303*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_TP_SRC,   "SET_TP_SRC"   },
304*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_SET_TP_DST,   "SET_TP_DST"   },
305*b636d99dSDavid van Moolenbroek 	{ 1 << OFPAT_ENQUEUE,      "ENQUEUE"      },
306*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
307*b636d99dSDavid van Moolenbroek };
308*b636d99dSDavid van Moolenbroek #define OFPAT_U (~(1 << OFPAT_OUTPUT | 1 << OFPAT_SET_VLAN_VID | \
309*b636d99dSDavid van Moolenbroek                    1 << OFPAT_SET_VLAN_PCP | 1 << OFPAT_STRIP_VLAN | \
310*b636d99dSDavid van Moolenbroek                    1 << OFPAT_SET_DL_SRC | 1 << OFPAT_SET_DL_DST | \
311*b636d99dSDavid van Moolenbroek                    1 << OFPAT_SET_NW_SRC | 1 << OFPAT_SET_NW_DST | \
312*b636d99dSDavid van Moolenbroek                    1 << OFPAT_SET_NW_TOS | 1 << OFPAT_SET_TP_SRC | \
313*b636d99dSDavid van Moolenbroek                    1 << OFPAT_SET_TP_DST | 1 << OFPAT_ENQUEUE))
314*b636d99dSDavid van Moolenbroek 
315*b636d99dSDavid van Moolenbroek #define OFPC_FLOW_STATS   (1 << 0)
316*b636d99dSDavid van Moolenbroek #define OFPC_TABLE_STATS  (1 << 1)
317*b636d99dSDavid van Moolenbroek #define OFPC_PORT_STATS   (1 << 2)
318*b636d99dSDavid van Moolenbroek #define OFPC_STP          (1 << 3)
319*b636d99dSDavid van Moolenbroek #define OFPC_RESERVED     (1 << 4)
320*b636d99dSDavid van Moolenbroek #define OFPC_IP_REASM     (1 << 5)
321*b636d99dSDavid van Moolenbroek #define OFPC_QUEUE_STATS  (1 << 6)
322*b636d99dSDavid van Moolenbroek #define OFPC_ARP_MATCH_IP (1 << 7)
323*b636d99dSDavid van Moolenbroek static const struct tok ofp_capabilities_bm[] = {
324*b636d99dSDavid van Moolenbroek 	{ OFPC_FLOW_STATS,   "FLOW_STATS"   },
325*b636d99dSDavid van Moolenbroek 	{ OFPC_TABLE_STATS,  "TABLE_STATS"  },
326*b636d99dSDavid van Moolenbroek 	{ OFPC_PORT_STATS,   "PORT_STATS"   },
327*b636d99dSDavid van Moolenbroek 	{ OFPC_STP,          "STP"          },
328*b636d99dSDavid van Moolenbroek 	{ OFPC_RESERVED,     "RESERVED"     }, /* not in the mask below */
329*b636d99dSDavid van Moolenbroek 	{ OFPC_IP_REASM,     "IP_REASM"     },
330*b636d99dSDavid van Moolenbroek 	{ OFPC_QUEUE_STATS,  "QUEUE_STATS"  },
331*b636d99dSDavid van Moolenbroek 	{ OFPC_ARP_MATCH_IP, "ARP_MATCH_IP" },
332*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
333*b636d99dSDavid van Moolenbroek };
334*b636d99dSDavid van Moolenbroek #define OFPCAP_U (~(OFPC_FLOW_STATS | OFPC_TABLE_STATS | OFPC_PORT_STATS | \
335*b636d99dSDavid van Moolenbroek                     OFPC_STP | OFPC_IP_REASM | OFPC_QUEUE_STATS | \
336*b636d99dSDavid van Moolenbroek                     OFPC_ARP_MATCH_IP))
337*b636d99dSDavid van Moolenbroek 
338*b636d99dSDavid van Moolenbroek #define OFPC_FRAG_NORMAL 0x0000
339*b636d99dSDavid van Moolenbroek #define OFPC_FRAG_DROP   0x0001
340*b636d99dSDavid van Moolenbroek #define OFPC_FRAG_REASM  0x0002
341*b636d99dSDavid van Moolenbroek #define OFPC_FRAG_MASK   0x0003
342*b636d99dSDavid van Moolenbroek static const struct tok ofp_config_str[] = {
343*b636d99dSDavid van Moolenbroek 	{ OFPC_FRAG_NORMAL, "FRAG_NORMAL" },
344*b636d99dSDavid van Moolenbroek 	{ OFPC_FRAG_DROP,   "FRAG_DROP"   },
345*b636d99dSDavid van Moolenbroek 	{ OFPC_FRAG_REASM,  "FRAG_REASM"  },
346*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
347*b636d99dSDavid van Moolenbroek };
348*b636d99dSDavid van Moolenbroek 
349*b636d99dSDavid van Moolenbroek #define OFPFC_ADD           0x0000
350*b636d99dSDavid van Moolenbroek #define OFPFC_MODIFY        0x0001
351*b636d99dSDavid van Moolenbroek #define OFPFC_MODIFY_STRICT 0x0002
352*b636d99dSDavid van Moolenbroek #define OFPFC_DELETE        0x0003
353*b636d99dSDavid van Moolenbroek #define OFPFC_DELETE_STRICT 0x0004
354*b636d99dSDavid van Moolenbroek static const struct tok ofpfc_str[] = {
355*b636d99dSDavid van Moolenbroek 	{ OFPFC_ADD,           "ADD"           },
356*b636d99dSDavid van Moolenbroek 	{ OFPFC_MODIFY,        "MODIFY"        },
357*b636d99dSDavid van Moolenbroek 	{ OFPFC_MODIFY_STRICT, "MODIFY_STRICT" },
358*b636d99dSDavid van Moolenbroek 	{ OFPFC_DELETE,        "DELETE"        },
359*b636d99dSDavid van Moolenbroek 	{ OFPFC_DELETE_STRICT, "DELETE_STRICT" },
360*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
361*b636d99dSDavid van Moolenbroek };
362*b636d99dSDavid van Moolenbroek 
363*b636d99dSDavid van Moolenbroek static const struct tok bufferid_str[] = {
364*b636d99dSDavid van Moolenbroek 	{ 0xffffffff, "NONE" },
365*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
366*b636d99dSDavid van Moolenbroek };
367*b636d99dSDavid van Moolenbroek 
368*b636d99dSDavid van Moolenbroek #define OFPFF_SEND_FLOW_REM (1 << 0)
369*b636d99dSDavid van Moolenbroek #define OFPFF_CHECK_OVERLAP (1 << 1)
370*b636d99dSDavid van Moolenbroek #define OFPFF_EMERG         (1 << 2)
371*b636d99dSDavid van Moolenbroek static const struct tok ofpff_bm[] = {
372*b636d99dSDavid van Moolenbroek 	{ OFPFF_SEND_FLOW_REM, "SEND_FLOW_REM" },
373*b636d99dSDavid van Moolenbroek 	{ OFPFF_CHECK_OVERLAP, "CHECK_OVERLAP" },
374*b636d99dSDavid van Moolenbroek 	{ OFPFF_EMERG,         "EMERG"         },
375*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
376*b636d99dSDavid van Moolenbroek };
377*b636d99dSDavid van Moolenbroek #define OFPFF_U (~(OFPFF_SEND_FLOW_REM | OFPFF_CHECK_OVERLAP | OFPFF_EMERG))
378*b636d99dSDavid van Moolenbroek 
379*b636d99dSDavid van Moolenbroek #define OFPST_DESC      0x0000
380*b636d99dSDavid van Moolenbroek #define OFPST_FLOW      0x0001
381*b636d99dSDavid van Moolenbroek #define OFPST_AGGREGATE 0x0002
382*b636d99dSDavid van Moolenbroek #define OFPST_TABLE     0x0003
383*b636d99dSDavid van Moolenbroek #define OFPST_PORT      0x0004
384*b636d99dSDavid van Moolenbroek #define OFPST_QUEUE     0x0005
385*b636d99dSDavid van Moolenbroek #define OFPST_VENDOR    0xffff
386*b636d99dSDavid van Moolenbroek static const struct tok ofpst_str[] = {
387*b636d99dSDavid van Moolenbroek 	{ OFPST_DESC,      "DESC"      },
388*b636d99dSDavid van Moolenbroek 	{ OFPST_FLOW,      "FLOW"      },
389*b636d99dSDavid van Moolenbroek 	{ OFPST_AGGREGATE, "AGGREGATE" },
390*b636d99dSDavid van Moolenbroek 	{ OFPST_TABLE,     "TABLE"     },
391*b636d99dSDavid van Moolenbroek 	{ OFPST_PORT,      "PORT"      },
392*b636d99dSDavid van Moolenbroek 	{ OFPST_QUEUE,     "QUEUE"     },
393*b636d99dSDavid van Moolenbroek 	{ OFPST_VENDOR,    "VENDOR"    },
394*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
395*b636d99dSDavid van Moolenbroek };
396*b636d99dSDavid van Moolenbroek 
397*b636d99dSDavid van Moolenbroek static const struct tok tableid_str[] = {
398*b636d99dSDavid van Moolenbroek 	{ 0xfe, "EMERG" },
399*b636d99dSDavid van Moolenbroek 	{ 0xff, "ALL"   },
400*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
401*b636d99dSDavid van Moolenbroek };
402*b636d99dSDavid van Moolenbroek 
403*b636d99dSDavid van Moolenbroek #define OFPQ_ALL      0xffffffff
404*b636d99dSDavid van Moolenbroek static const struct tok ofpq_str[] = {
405*b636d99dSDavid van Moolenbroek 	{ OFPQ_ALL, "ALL" },
406*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
407*b636d99dSDavid van Moolenbroek };
408*b636d99dSDavid van Moolenbroek 
409*b636d99dSDavid van Moolenbroek #define OFPSF_REPLY_MORE 0x0001
410*b636d99dSDavid van Moolenbroek static const struct tok ofpsf_reply_bm[] = {
411*b636d99dSDavid van Moolenbroek 	{ OFPSF_REPLY_MORE, "MORE" },
412*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
413*b636d99dSDavid van Moolenbroek };
414*b636d99dSDavid van Moolenbroek #define OFPSF_REPLY_U (~(OFPSF_REPLY_MORE))
415*b636d99dSDavid van Moolenbroek 
416*b636d99dSDavid van Moolenbroek #define OFPR_NO_MATCH 0x00
417*b636d99dSDavid van Moolenbroek #define OFPR_ACTION   0x01
418*b636d99dSDavid van Moolenbroek static const struct tok ofpr_str[] = {
419*b636d99dSDavid van Moolenbroek 	{ OFPR_NO_MATCH, "NO_MATCH" },
420*b636d99dSDavid van Moolenbroek 	{ OFPR_ACTION,   "ACTION"   },
421*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
422*b636d99dSDavid van Moolenbroek };
423*b636d99dSDavid van Moolenbroek 
424*b636d99dSDavid van Moolenbroek #define OFPRR_IDLE_TIMEOUT 0x00
425*b636d99dSDavid van Moolenbroek #define OFPRR_HARD_TIMEOUT 0x01
426*b636d99dSDavid van Moolenbroek #define OFPRR_DELETE       0x02
427*b636d99dSDavid van Moolenbroek static const struct tok ofprr_str[] = {
428*b636d99dSDavid van Moolenbroek 	{ OFPRR_IDLE_TIMEOUT, "IDLE_TIMEOUT" },
429*b636d99dSDavid van Moolenbroek 	{ OFPRR_HARD_TIMEOUT, "HARD_TIMEOUT" },
430*b636d99dSDavid van Moolenbroek 	{ OFPRR_DELETE,       "DELETE"       },
431*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
432*b636d99dSDavid van Moolenbroek };
433*b636d99dSDavid van Moolenbroek 
434*b636d99dSDavid van Moolenbroek #define OFPPR_ADD    0x00
435*b636d99dSDavid van Moolenbroek #define OFPPR_DELETE 0x01
436*b636d99dSDavid van Moolenbroek #define OFPPR_MODIFY 0x02
437*b636d99dSDavid van Moolenbroek static const struct tok ofppr_str[] = {
438*b636d99dSDavid van Moolenbroek 	{ OFPPR_ADD,    "ADD"    },
439*b636d99dSDavid van Moolenbroek 	{ OFPPR_DELETE, "DELETE" },
440*b636d99dSDavid van Moolenbroek 	{ OFPPR_MODIFY, "MODIFY" },
441*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
442*b636d99dSDavid van Moolenbroek };
443*b636d99dSDavid van Moolenbroek 
444*b636d99dSDavid van Moolenbroek #define OFPET_HELLO_FAILED    0x0000
445*b636d99dSDavid van Moolenbroek #define OFPET_BAD_REQUEST     0x0001
446*b636d99dSDavid van Moolenbroek #define OFPET_BAD_ACTION      0x0002
447*b636d99dSDavid van Moolenbroek #define OFPET_FLOW_MOD_FAILED 0x0003
448*b636d99dSDavid van Moolenbroek #define OFPET_PORT_MOD_FAILED 0x0004
449*b636d99dSDavid van Moolenbroek #define OFPET_QUEUE_OP_FAILED 0x0005
450*b636d99dSDavid van Moolenbroek static const struct tok ofpet_str[] = {
451*b636d99dSDavid van Moolenbroek 	{ OFPET_HELLO_FAILED,    "HELLO_FAILED"    },
452*b636d99dSDavid van Moolenbroek 	{ OFPET_BAD_REQUEST,     "BAD_REQUEST"     },
453*b636d99dSDavid van Moolenbroek 	{ OFPET_BAD_ACTION,      "BAD_ACTION"      },
454*b636d99dSDavid van Moolenbroek 	{ OFPET_FLOW_MOD_FAILED, "FLOW_MOD_FAILED" },
455*b636d99dSDavid van Moolenbroek 	{ OFPET_PORT_MOD_FAILED, "PORT_MOD_FAILED" },
456*b636d99dSDavid van Moolenbroek 	{ OFPET_QUEUE_OP_FAILED, "QUEUE_OP_FAILED" },
457*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
458*b636d99dSDavid van Moolenbroek };
459*b636d99dSDavid van Moolenbroek 
460*b636d99dSDavid van Moolenbroek #define OFPHFC_INCOMPATIBLE 0x0000
461*b636d99dSDavid van Moolenbroek #define OFPHFC_EPERM        0x0001
462*b636d99dSDavid van Moolenbroek static const struct tok ofphfc_str[] = {
463*b636d99dSDavid van Moolenbroek 	{ OFPHFC_INCOMPATIBLE, "INCOMPATIBLE" },
464*b636d99dSDavid van Moolenbroek 	{ OFPHFC_EPERM,        "EPERM"        },
465*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
466*b636d99dSDavid van Moolenbroek };
467*b636d99dSDavid van Moolenbroek 
468*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_VERSION    0x0000
469*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_TYPE       0x0001
470*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_STAT       0x0002
471*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_VENDOR     0x0003
472*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_SUBTYPE    0x0004
473*b636d99dSDavid van Moolenbroek #define OFPBRC_EPERM          0x0005
474*b636d99dSDavid van Moolenbroek #define OFPBRC_BAD_LEN        0x0006
475*b636d99dSDavid van Moolenbroek #define OFPBRC_BUFFER_EMPTY   0x0007
476*b636d99dSDavid van Moolenbroek #define OFPBRC_BUFFER_UNKNOWN 0x0008
477*b636d99dSDavid van Moolenbroek static const struct tok ofpbrc_str[] = {
478*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_VERSION,    "BAD_VERSION"    },
479*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_TYPE,       "BAD_TYPE"       },
480*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_STAT,       "BAD_STAT"       },
481*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_VENDOR,     "BAD_VENDOR"     },
482*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_SUBTYPE,    "BAD_SUBTYPE"    },
483*b636d99dSDavid van Moolenbroek 	{ OFPBRC_EPERM,          "EPERM"          },
484*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BAD_LEN,        "BAD_LEN"        },
485*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BUFFER_EMPTY,   "BUFFER_EMPTY"   },
486*b636d99dSDavid van Moolenbroek 	{ OFPBRC_BUFFER_UNKNOWN, "BUFFER_UNKNOWN" },
487*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
488*b636d99dSDavid van Moolenbroek };
489*b636d99dSDavid van Moolenbroek 
490*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_TYPE        0x0000
491*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_LEN         0x0001
492*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_VENDOR      0x0002
493*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_VENDOR_TYPE 0x0003
494*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_OUT_PORT    0x0004
495*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_ARGUMENT    0x0005
496*b636d99dSDavid van Moolenbroek #define OFPBAC_EPERM           0x0006
497*b636d99dSDavid van Moolenbroek #define OFPBAC_TOO_MANY        0x0007
498*b636d99dSDavid van Moolenbroek #define OFPBAC_BAD_QUEUE       0x0008
499*b636d99dSDavid van Moolenbroek static const struct tok ofpbac_str[] = {
500*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_TYPE,        "BAD_TYPE"        },
501*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_LEN,         "BAD_LEN"         },
502*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_VENDOR,      "BAD_VENDOR"      },
503*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_VENDOR_TYPE, "BAD_VENDOR_TYPE" },
504*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_OUT_PORT,    "BAD_OUT_PORT"    },
505*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_ARGUMENT,    "BAD_ARGUMENT"    },
506*b636d99dSDavid van Moolenbroek 	{ OFPBAC_EPERM,           "EPERM"           },
507*b636d99dSDavid van Moolenbroek 	{ OFPBAC_TOO_MANY,        "TOO_MANY"        },
508*b636d99dSDavid van Moolenbroek 	{ OFPBAC_BAD_QUEUE,       "BAD_QUEUE"       },
509*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
510*b636d99dSDavid van Moolenbroek };
511*b636d99dSDavid van Moolenbroek 
512*b636d99dSDavid van Moolenbroek #define OFPFMFC_ALL_TABLES_FULL   0x0000
513*b636d99dSDavid van Moolenbroek #define OFPFMFC_OVERLAP           0x0001
514*b636d99dSDavid van Moolenbroek #define OFPFMFC_EPERM             0x0002
515*b636d99dSDavid van Moolenbroek #define OFPFMFC_BAD_EMERG_TIMEOUT 0x0003
516*b636d99dSDavid van Moolenbroek #define OFPFMFC_BAD_COMMAND       0x0004
517*b636d99dSDavid van Moolenbroek #define OFPFMFC_UNSUPPORTED       0x0005
518*b636d99dSDavid van Moolenbroek static const struct tok ofpfmfc_str[] = {
519*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_ALL_TABLES_FULL,   "ALL_TABLES_FULL"   },
520*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_OVERLAP,           "OVERLAP"           },
521*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_EPERM,             "EPERM"             },
522*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_BAD_EMERG_TIMEOUT, "BAD_EMERG_TIMEOUT" },
523*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_BAD_COMMAND,       "BAD_COMMAND"       },
524*b636d99dSDavid van Moolenbroek 	{ OFPFMFC_UNSUPPORTED,       "UNSUPPORTED"       },
525*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
526*b636d99dSDavid van Moolenbroek };
527*b636d99dSDavid van Moolenbroek 
528*b636d99dSDavid van Moolenbroek #define OFPPMFC_BAD_PORT    0x0000
529*b636d99dSDavid van Moolenbroek #define OFPPMFC_BAD_HW_ADDR 0x0001
530*b636d99dSDavid van Moolenbroek static const struct tok ofppmfc_str[] = {
531*b636d99dSDavid van Moolenbroek 	{ OFPPMFC_BAD_PORT,    "BAD_PORT"    },
532*b636d99dSDavid van Moolenbroek 	{ OFPPMFC_BAD_HW_ADDR, "BAD_HW_ADDR" },
533*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
534*b636d99dSDavid van Moolenbroek };
535*b636d99dSDavid van Moolenbroek 
536*b636d99dSDavid van Moolenbroek #define OFPQOFC_BAD_PORT  0x0000
537*b636d99dSDavid van Moolenbroek #define OFPQOFC_BAD_QUEUE 0x0001
538*b636d99dSDavid van Moolenbroek #define OFPQOFC_EPERM     0x0002
539*b636d99dSDavid van Moolenbroek static const struct tok ofpqofc_str[] = {
540*b636d99dSDavid van Moolenbroek 	{ OFPQOFC_BAD_PORT,  "BAD_PORT"  },
541*b636d99dSDavid van Moolenbroek 	{ OFPQOFC_BAD_QUEUE, "BAD_QUEUE" },
542*b636d99dSDavid van Moolenbroek 	{ OFPQOFC_EPERM,     "EPERM"     },
543*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
544*b636d99dSDavid van Moolenbroek };
545*b636d99dSDavid van Moolenbroek 
546*b636d99dSDavid van Moolenbroek static const struct tok empty_str[] = {
547*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
548*b636d99dSDavid van Moolenbroek };
549*b636d99dSDavid van Moolenbroek 
550*b636d99dSDavid van Moolenbroek /* lengths (fixed or minimal) of particular protocol structures */
551*b636d99dSDavid van Moolenbroek #define OF_SWITCH_CONFIG_LEN              12
552*b636d99dSDavid van Moolenbroek #define OF_PHY_PORT_LEN                   48
553*b636d99dSDavid van Moolenbroek #define OF_SWITCH_FEATURES_LEN            32
554*b636d99dSDavid van Moolenbroek #define OF_PORT_STATUS_LEN                64
555*b636d99dSDavid van Moolenbroek #define OF_PORT_MOD_LEN                   32
556*b636d99dSDavid van Moolenbroek #define OF_PACKET_IN_LEN                  20
557*b636d99dSDavid van Moolenbroek #define OF_ACTION_OUTPUT_LEN               8
558*b636d99dSDavid van Moolenbroek #define OF_ACTION_VLAN_VID_LEN             8
559*b636d99dSDavid van Moolenbroek #define OF_ACTION_VLAN_PCP_LEN             8
560*b636d99dSDavid van Moolenbroek #define OF_ACTION_DL_ADDR_LEN             16
561*b636d99dSDavid van Moolenbroek #define OF_ACTION_NW_ADDR_LEN              8
562*b636d99dSDavid van Moolenbroek #define OF_ACTION_TP_PORT_LEN              8
563*b636d99dSDavid van Moolenbroek #define OF_ACTION_NW_TOS_LEN               8
564*b636d99dSDavid van Moolenbroek #define OF_ACTION_VENDOR_HEADER_LEN        8
565*b636d99dSDavid van Moolenbroek #define OF_ACTION_HEADER_LEN               8
566*b636d99dSDavid van Moolenbroek #define OF_PACKET_OUT_LEN                 16
567*b636d99dSDavid van Moolenbroek #define OF_MATCH_LEN                      40
568*b636d99dSDavid van Moolenbroek #define OF_FLOW_MOD_LEN                   72
569*b636d99dSDavid van Moolenbroek #define OF_FLOW_REMOVED_LEN               88
570*b636d99dSDavid van Moolenbroek #define OF_ERROR_MSG_LEN                  12
571*b636d99dSDavid van Moolenbroek #define OF_STATS_REQUEST_LEN              12
572*b636d99dSDavid van Moolenbroek #define OF_STATS_REPLY_LEN                12
573*b636d99dSDavid van Moolenbroek #define OF_DESC_STATS_LEN               1056
574*b636d99dSDavid van Moolenbroek #define OF_FLOW_STATS_REQUEST_LEN         44
575*b636d99dSDavid van Moolenbroek #define OF_FLOW_STATS_LEN                 88
576*b636d99dSDavid van Moolenbroek #define OF_AGGREGATE_STATS_REQUEST_LEN    44
577*b636d99dSDavid van Moolenbroek #define OF_AGGREGATE_STATS_REPLY_LEN      24
578*b636d99dSDavid van Moolenbroek #define OF_TABLE_STATS_LEN                64
579*b636d99dSDavid van Moolenbroek #define OF_PORT_STATS_REQUEST_LEN          8
580*b636d99dSDavid van Moolenbroek #define OF_PORT_STATS_LEN                104
581*b636d99dSDavid van Moolenbroek #define OF_VENDOR_HEADER_LEN              12
582*b636d99dSDavid van Moolenbroek #define OF_QUEUE_PROP_HEADER_LEN           8
583*b636d99dSDavid van Moolenbroek #define OF_QUEUE_PROP_MIN_RATE_LEN        16
584*b636d99dSDavid van Moolenbroek #define OF_PACKET_QUEUE_LEN                8
585*b636d99dSDavid van Moolenbroek #define OF_QUEUE_GET_CONFIG_REQUEST_LEN   12
586*b636d99dSDavid van Moolenbroek #define OF_QUEUE_GET_CONFIG_REPLY_LEN     16
587*b636d99dSDavid van Moolenbroek #define OF_ACTION_ENQUEUE_LEN             16
588*b636d99dSDavid van Moolenbroek #define OF_QUEUE_STATS_REQUEST_LEN         8
589*b636d99dSDavid van Moolenbroek #define OF_QUEUE_STATS_LEN                32
590*b636d99dSDavid van Moolenbroek 
591*b636d99dSDavid van Moolenbroek /* miscellaneous constants from [OF10] */
592*b636d99dSDavid van Moolenbroek #define OFP_MAX_TABLE_NAME_LEN     32
593*b636d99dSDavid van Moolenbroek #define OFP_MAX_PORT_NAME_LEN      16
594*b636d99dSDavid van Moolenbroek #define DESC_STR_LEN              256
595*b636d99dSDavid van Moolenbroek #define SERIAL_NUM_LEN             32
596*b636d99dSDavid van Moolenbroek #define OFP_VLAN_NONE          0xffff
597*b636d99dSDavid van Moolenbroek 
598*b636d99dSDavid van Moolenbroek /* vendor extensions */
599*b636d99dSDavid van Moolenbroek #define BSN_SET_IP_MASK                    0
600*b636d99dSDavid van Moolenbroek #define BSN_GET_IP_MASK_REQUEST            1
601*b636d99dSDavid van Moolenbroek #define BSN_GET_IP_MASK_REPLY              2
602*b636d99dSDavid van Moolenbroek #define BSN_SET_MIRRORING                  3
603*b636d99dSDavid van Moolenbroek #define BSN_GET_MIRRORING_REQUEST          4
604*b636d99dSDavid van Moolenbroek #define BSN_GET_MIRRORING_REPLY            5
605*b636d99dSDavid van Moolenbroek #define BSN_SHELL_COMMAND                  6
606*b636d99dSDavid van Moolenbroek #define BSN_SHELL_OUTPUT                   7
607*b636d99dSDavid van Moolenbroek #define BSN_SHELL_STATUS                   8
608*b636d99dSDavid van Moolenbroek #define BSN_GET_INTERFACES_REQUEST         9
609*b636d99dSDavid van Moolenbroek #define BSN_GET_INTERFACES_REPLY          10
610*b636d99dSDavid van Moolenbroek #define BSN_SET_PKTIN_SUPPRESSION_REQUEST 11
611*b636d99dSDavid van Moolenbroek #define BSN_SET_L2_TABLE_REQUEST          12
612*b636d99dSDavid van Moolenbroek #define BSN_GET_L2_TABLE_REQUEST          13
613*b636d99dSDavid van Moolenbroek #define BSN_GET_L2_TABLE_REPLY            14
614*b636d99dSDavid van Moolenbroek #define BSN_VIRTUAL_PORT_CREATE_REQUEST   15
615*b636d99dSDavid van Moolenbroek #define BSN_VIRTUAL_PORT_CREATE_REPLY     16
616*b636d99dSDavid van Moolenbroek #define BSN_VIRTUAL_PORT_REMOVE_REQUEST   17
617*b636d99dSDavid van Moolenbroek #define BSN_BW_ENABLE_SET_REQUEST         18
618*b636d99dSDavid van Moolenbroek #define BSN_BW_ENABLE_GET_REQUEST         19
619*b636d99dSDavid van Moolenbroek #define BSN_BW_ENABLE_GET_REPLY           20
620*b636d99dSDavid van Moolenbroek #define BSN_BW_CLEAR_DATA_REQUEST         21
621*b636d99dSDavid van Moolenbroek #define BSN_BW_CLEAR_DATA_REPLY           22
622*b636d99dSDavid van Moolenbroek #define BSN_BW_ENABLE_SET_REPLY           23
623*b636d99dSDavid van Moolenbroek #define BSN_SET_L2_TABLE_REPLY            24
624*b636d99dSDavid van Moolenbroek #define BSN_SET_PKTIN_SUPPRESSION_REPLY   25
625*b636d99dSDavid van Moolenbroek #define BSN_VIRTUAL_PORT_REMOVE_REPLY     26
626*b636d99dSDavid van Moolenbroek #define BSN_HYBRID_GET_REQUEST            27
627*b636d99dSDavid van Moolenbroek #define BSN_HYBRID_GET_REPLY              28
628*b636d99dSDavid van Moolenbroek                                        /* 29 */
629*b636d99dSDavid van Moolenbroek                                        /* 30 */
630*b636d99dSDavid van Moolenbroek #define BSN_PDU_TX_REQUEST                31
631*b636d99dSDavid van Moolenbroek #define BSN_PDU_TX_REPLY                  32
632*b636d99dSDavid van Moolenbroek #define BSN_PDU_RX_REQUEST                33
633*b636d99dSDavid van Moolenbroek #define BSN_PDU_RX_REPLY                  34
634*b636d99dSDavid van Moolenbroek #define BSN_PDU_RX_TIMEOUT                35
635*b636d99dSDavid van Moolenbroek 
636*b636d99dSDavid van Moolenbroek static const struct tok bsn_subtype_str[] = {
637*b636d99dSDavid van Moolenbroek 	{ BSN_SET_IP_MASK,                   "SET_IP_MASK"                   },
638*b636d99dSDavid van Moolenbroek 	{ BSN_GET_IP_MASK_REQUEST,           "GET_IP_MASK_REQUEST"           },
639*b636d99dSDavid van Moolenbroek 	{ BSN_GET_IP_MASK_REPLY,             "GET_IP_MASK_REPLY"             },
640*b636d99dSDavid van Moolenbroek 	{ BSN_SET_MIRRORING,                 "SET_MIRRORING"                 },
641*b636d99dSDavid van Moolenbroek 	{ BSN_GET_MIRRORING_REQUEST,         "GET_MIRRORING_REQUEST"         },
642*b636d99dSDavid van Moolenbroek 	{ BSN_GET_MIRRORING_REPLY,           "GET_MIRRORING_REPLY"           },
643*b636d99dSDavid van Moolenbroek 	{ BSN_SHELL_COMMAND,                 "SHELL_COMMAND"                 },
644*b636d99dSDavid van Moolenbroek 	{ BSN_SHELL_OUTPUT,                  "SHELL_OUTPUT"                  },
645*b636d99dSDavid van Moolenbroek 	{ BSN_SHELL_STATUS,                  "SHELL_STATUS"                  },
646*b636d99dSDavid van Moolenbroek 	{ BSN_GET_INTERFACES_REQUEST,        "GET_INTERFACES_REQUEST"        },
647*b636d99dSDavid van Moolenbroek 	{ BSN_GET_INTERFACES_REPLY,          "GET_INTERFACES_REPLY"          },
648*b636d99dSDavid van Moolenbroek 	{ BSN_SET_PKTIN_SUPPRESSION_REQUEST, "SET_PKTIN_SUPPRESSION_REQUEST" },
649*b636d99dSDavid van Moolenbroek 	{ BSN_SET_L2_TABLE_REQUEST,          "SET_L2_TABLE_REQUEST"          },
650*b636d99dSDavid van Moolenbroek 	{ BSN_GET_L2_TABLE_REQUEST,          "GET_L2_TABLE_REQUEST"          },
651*b636d99dSDavid van Moolenbroek 	{ BSN_GET_L2_TABLE_REPLY,            "GET_L2_TABLE_REPLY"            },
652*b636d99dSDavid van Moolenbroek 	{ BSN_VIRTUAL_PORT_CREATE_REQUEST,   "VIRTUAL_PORT_CREATE_REQUEST"   },
653*b636d99dSDavid van Moolenbroek 	{ BSN_VIRTUAL_PORT_CREATE_REPLY,     "VIRTUAL_PORT_CREATE_REPLY"     },
654*b636d99dSDavid van Moolenbroek 	{ BSN_VIRTUAL_PORT_REMOVE_REQUEST,   "VIRTUAL_PORT_REMOVE_REQUEST"   },
655*b636d99dSDavid van Moolenbroek 	{ BSN_BW_ENABLE_SET_REQUEST,         "BW_ENABLE_SET_REQUEST"         },
656*b636d99dSDavid van Moolenbroek 	{ BSN_BW_ENABLE_GET_REQUEST,         "BW_ENABLE_GET_REQUEST"         },
657*b636d99dSDavid van Moolenbroek 	{ BSN_BW_ENABLE_GET_REPLY,           "BW_ENABLE_GET_REPLY"           },
658*b636d99dSDavid van Moolenbroek 	{ BSN_BW_CLEAR_DATA_REQUEST,         "BW_CLEAR_DATA_REQUEST"         },
659*b636d99dSDavid van Moolenbroek 	{ BSN_BW_CLEAR_DATA_REPLY,           "BW_CLEAR_DATA_REPLY"           },
660*b636d99dSDavid van Moolenbroek 	{ BSN_BW_ENABLE_SET_REPLY,           "BW_ENABLE_SET_REPLY"           },
661*b636d99dSDavid van Moolenbroek 	{ BSN_SET_L2_TABLE_REPLY,            "SET_L2_TABLE_REPLY"            },
662*b636d99dSDavid van Moolenbroek 	{ BSN_SET_PKTIN_SUPPRESSION_REPLY,   "SET_PKTIN_SUPPRESSION_REPLY"   },
663*b636d99dSDavid van Moolenbroek 	{ BSN_VIRTUAL_PORT_REMOVE_REPLY,     "VIRTUAL_PORT_REMOVE_REPLY"     },
664*b636d99dSDavid van Moolenbroek 	{ BSN_HYBRID_GET_REQUEST,            "HYBRID_GET_REQUEST"            },
665*b636d99dSDavid van Moolenbroek 	{ BSN_HYBRID_GET_REPLY,              "HYBRID_GET_REPLY"              },
666*b636d99dSDavid van Moolenbroek 	{ BSN_PDU_TX_REQUEST,                "PDU_TX_REQUEST"                },
667*b636d99dSDavid van Moolenbroek 	{ BSN_PDU_TX_REPLY,                  "PDU_TX_REPLY"                  },
668*b636d99dSDavid van Moolenbroek 	{ BSN_PDU_RX_REQUEST,                "PDU_RX_REQUEST"                },
669*b636d99dSDavid van Moolenbroek 	{ BSN_PDU_RX_REPLY,                  "PDU_RX_REPLY"                  },
670*b636d99dSDavid van Moolenbroek 	{ BSN_PDU_RX_TIMEOUT,                "PDU_RX_TIMEOUT"                },
671*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
672*b636d99dSDavid van Moolenbroek };
673*b636d99dSDavid van Moolenbroek 
674*b636d99dSDavid van Moolenbroek #define BSN_ACTION_MIRROR                  1
675*b636d99dSDavid van Moolenbroek #define BSN_ACTION_SET_TUNNEL_DST          2
676*b636d99dSDavid van Moolenbroek                                         /* 3 */
677*b636d99dSDavid van Moolenbroek #define BSN_ACTION_CHECKSUM                4
678*b636d99dSDavid van Moolenbroek 
679*b636d99dSDavid van Moolenbroek static const struct tok bsn_action_subtype_str[] = {
680*b636d99dSDavid van Moolenbroek 	{ BSN_ACTION_MIRROR,                 "MIRROR"                        },
681*b636d99dSDavid van Moolenbroek 	{ BSN_ACTION_SET_TUNNEL_DST,         "SET_TUNNEL_DST"                },
682*b636d99dSDavid van Moolenbroek 	{ BSN_ACTION_CHECKSUM,               "CHECKSUM"                      },
683*b636d99dSDavid van Moolenbroek 	{ 0, NULL }
684*b636d99dSDavid van Moolenbroek };
685*b636d99dSDavid van Moolenbroek 
686*b636d99dSDavid van Moolenbroek static const struct tok bsn_mirror_copy_stage_str[] = {
687*b636d99dSDavid van Moolenbroek 	{ 0, "INGRESS" },
688*b636d99dSDavid van Moolenbroek 	{ 1, "EGRESS"  },
689*b636d99dSDavid van Moolenbroek 	{ 0, NULL },
690*b636d99dSDavid van Moolenbroek };
691*b636d99dSDavid van Moolenbroek 
692*b636d99dSDavid van Moolenbroek static const struct tok bsn_onoff_str[] = {
693*b636d99dSDavid van Moolenbroek 	{ 0, "OFF" },
694*b636d99dSDavid van Moolenbroek 	{ 1, "ON"  },
695*b636d99dSDavid van Moolenbroek 	{ 0, NULL },
696*b636d99dSDavid van Moolenbroek };
697*b636d99dSDavid van Moolenbroek 
698*b636d99dSDavid van Moolenbroek static const char *
vlan_str(const uint16_t vid)699*b636d99dSDavid van Moolenbroek vlan_str(const uint16_t vid)
700*b636d99dSDavid van Moolenbroek {
701*b636d99dSDavid van Moolenbroek 	static char buf[sizeof("65535 (bogus)")];
702*b636d99dSDavid van Moolenbroek 	const char *fmt;
703*b636d99dSDavid van Moolenbroek 
704*b636d99dSDavid van Moolenbroek 	if (vid == OFP_VLAN_NONE)
705*b636d99dSDavid van Moolenbroek 		return "NONE";
706*b636d99dSDavid van Moolenbroek 	fmt = (vid > 0 && vid < 0x0fff) ? "%u" : "%u (bogus)";
707*b636d99dSDavid van Moolenbroek 	snprintf(buf, sizeof(buf), fmt, vid);
708*b636d99dSDavid van Moolenbroek 	return buf;
709*b636d99dSDavid van Moolenbroek }
710*b636d99dSDavid van Moolenbroek 
711*b636d99dSDavid van Moolenbroek static const char *
pcp_str(const uint8_t pcp)712*b636d99dSDavid van Moolenbroek pcp_str(const uint8_t pcp)
713*b636d99dSDavid van Moolenbroek {
714*b636d99dSDavid van Moolenbroek 	static char buf[sizeof("255 (bogus)")];
715*b636d99dSDavid van Moolenbroek 	snprintf(buf, sizeof(buf), pcp <= 7 ? "%u" : "%u (bogus)", pcp);
716*b636d99dSDavid van Moolenbroek 	return buf;
717*b636d99dSDavid van Moolenbroek }
718*b636d99dSDavid van Moolenbroek 
719*b636d99dSDavid van Moolenbroek static void
of10_bitmap_print(netdissect_options * ndo,const struct tok * t,const uint32_t v,const uint32_t u)720*b636d99dSDavid van Moolenbroek of10_bitmap_print(netdissect_options *ndo,
721*b636d99dSDavid van Moolenbroek                   const struct tok *t, const uint32_t v, const uint32_t u)
722*b636d99dSDavid van Moolenbroek {
723*b636d99dSDavid van Moolenbroek 	const char *sep = " (";
724*b636d99dSDavid van Moolenbroek 
725*b636d99dSDavid van Moolenbroek 	if (v == 0)
726*b636d99dSDavid van Moolenbroek 		return;
727*b636d99dSDavid van Moolenbroek 	/* assigned bits */
728*b636d99dSDavid van Moolenbroek 	for (; t->s != NULL; t++)
729*b636d99dSDavid van Moolenbroek 		if (v & t->v) {
730*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, "%s%s", sep, t->s));
731*b636d99dSDavid van Moolenbroek 			sep = ", ";
732*b636d99dSDavid van Moolenbroek 		}
733*b636d99dSDavid van Moolenbroek 	/* unassigned bits? */
734*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, v & u ? ") (bogus)" : ")"));
735*b636d99dSDavid van Moolenbroek }
736*b636d99dSDavid van Moolenbroek 
737*b636d99dSDavid van Moolenbroek static const u_char *
of10_data_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)738*b636d99dSDavid van Moolenbroek of10_data_print(netdissect_options *ndo,
739*b636d99dSDavid van Moolenbroek                 const u_char *cp, const u_char *ep, const u_int len)
740*b636d99dSDavid van Moolenbroek {
741*b636d99dSDavid van Moolenbroek 	if (len == 0)
742*b636d99dSDavid van Moolenbroek 		return cp;
743*b636d99dSDavid van Moolenbroek 	/* data */
744*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t data (%u octets)", len));
745*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
746*b636d99dSDavid van Moolenbroek 	if (ndo->ndo_vflag >= 2)
747*b636d99dSDavid van Moolenbroek 		hex_and_ascii_print(ndo, "\n\t  ", cp, len);
748*b636d99dSDavid van Moolenbroek 	return cp + len;
749*b636d99dSDavid van Moolenbroek 
750*b636d99dSDavid van Moolenbroek trunc:
751*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
752*b636d99dSDavid van Moolenbroek 	return ep;
753*b636d99dSDavid van Moolenbroek }
754*b636d99dSDavid van Moolenbroek 
755*b636d99dSDavid van Moolenbroek static const u_char *
of10_bsn_message_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)756*b636d99dSDavid van Moolenbroek of10_bsn_message_print(netdissect_options *ndo,
757*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const u_int len)
758*b636d99dSDavid van Moolenbroek {
759*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
760*b636d99dSDavid van Moolenbroek 	uint32_t subtype;
761*b636d99dSDavid van Moolenbroek 
762*b636d99dSDavid van Moolenbroek 	if (len < 4)
763*b636d99dSDavid van Moolenbroek 		goto corrupt;
764*b636d99dSDavid van Moolenbroek 	/* subtype */
765*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
766*b636d99dSDavid van Moolenbroek 	subtype = EXTRACT_32BITS(cp);
767*b636d99dSDavid van Moolenbroek 	cp += 4;
768*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t subtype %s", tok2str(bsn_subtype_str, "unknown (0x%08x)", subtype)));
769*b636d99dSDavid van Moolenbroek 	switch (subtype) {
770*b636d99dSDavid van Moolenbroek 	case BSN_GET_IP_MASK_REQUEST:
771*b636d99dSDavid van Moolenbroek 		/*
772*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
773*b636d99dSDavid van Moolenbroek 		 *  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
774*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
775*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
776*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
777*b636d99dSDavid van Moolenbroek 		 * |     index     |                      pad                      |
778*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
779*b636d99dSDavid van Moolenbroek 		 * |                              pad                              |
780*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
781*b636d99dSDavid van Moolenbroek 		 *
782*b636d99dSDavid van Moolenbroek 		 */
783*b636d99dSDavid van Moolenbroek 		if (len != 12)
784*b636d99dSDavid van Moolenbroek 			goto corrupt;
785*b636d99dSDavid van Moolenbroek 		/* index */
786*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
787*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", index %u", *cp));
788*b636d99dSDavid van Moolenbroek 		cp += 1;
789*b636d99dSDavid van Moolenbroek 		/* pad */
790*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 7);
791*b636d99dSDavid van Moolenbroek 		cp += 7;
792*b636d99dSDavid van Moolenbroek 		break;
793*b636d99dSDavid van Moolenbroek 	case BSN_SET_IP_MASK:
794*b636d99dSDavid van Moolenbroek 	case BSN_GET_IP_MASK_REPLY:
795*b636d99dSDavid van Moolenbroek 		/*
796*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
797*b636d99dSDavid van Moolenbroek 		 *  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
798*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
799*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
800*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
801*b636d99dSDavid van Moolenbroek 		 * |     index     |                      pad                      |
802*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
803*b636d99dSDavid van Moolenbroek 		 * |                              mask                             |
804*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
805*b636d99dSDavid van Moolenbroek 		 *
806*b636d99dSDavid van Moolenbroek 		 */
807*b636d99dSDavid van Moolenbroek 		if (len != 12)
808*b636d99dSDavid van Moolenbroek 			goto corrupt;
809*b636d99dSDavid van Moolenbroek 		/* index */
810*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
811*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", index %u", *cp));
812*b636d99dSDavid van Moolenbroek 		cp += 1;
813*b636d99dSDavid van Moolenbroek 		/* pad */
814*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 3);
815*b636d99dSDavid van Moolenbroek 		cp += 3;
816*b636d99dSDavid van Moolenbroek 		/* mask */
817*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
818*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", mask %s", ipaddr_string(ndo, cp)));
819*b636d99dSDavid van Moolenbroek 		cp += 4;
820*b636d99dSDavid van Moolenbroek 		break;
821*b636d99dSDavid van Moolenbroek 	case BSN_SET_MIRRORING:
822*b636d99dSDavid van Moolenbroek 	case BSN_GET_MIRRORING_REQUEST:
823*b636d99dSDavid van Moolenbroek 	case BSN_GET_MIRRORING_REPLY:
824*b636d99dSDavid van Moolenbroek 		/*
825*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
826*b636d99dSDavid van Moolenbroek 		 *  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
827*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
828*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
829*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
830*b636d99dSDavid van Moolenbroek 		 * | report m. p.  |                      pad                      |
831*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
832*b636d99dSDavid van Moolenbroek 		 *
833*b636d99dSDavid van Moolenbroek 		 */
834*b636d99dSDavid van Moolenbroek 		if (len != 8)
835*b636d99dSDavid van Moolenbroek 			goto corrupt;
836*b636d99dSDavid van Moolenbroek 		/* report_mirror_ports */
837*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
838*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", report_mirror_ports %s", tok2str(bsn_onoff_str, "bogus (%u)", *cp)));
839*b636d99dSDavid van Moolenbroek 		cp += 1;
840*b636d99dSDavid van Moolenbroek 		/* pad */
841*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 3);
842*b636d99dSDavid van Moolenbroek 		cp += 3;
843*b636d99dSDavid van Moolenbroek 		break;
844*b636d99dSDavid van Moolenbroek 	case BSN_GET_INTERFACES_REQUEST:
845*b636d99dSDavid van Moolenbroek 	case BSN_GET_L2_TABLE_REQUEST:
846*b636d99dSDavid van Moolenbroek 	case BSN_BW_ENABLE_GET_REQUEST:
847*b636d99dSDavid van Moolenbroek 	case BSN_BW_CLEAR_DATA_REQUEST:
848*b636d99dSDavid van Moolenbroek 	case BSN_HYBRID_GET_REQUEST:
849*b636d99dSDavid van Moolenbroek 		/*
850*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
851*b636d99dSDavid van Moolenbroek 		 *  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
852*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
853*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
854*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
855*b636d99dSDavid van Moolenbroek 		 *
856*b636d99dSDavid van Moolenbroek 		 */
857*b636d99dSDavid van Moolenbroek 		if (len != 4)
858*b636d99dSDavid van Moolenbroek 			goto corrupt;
859*b636d99dSDavid van Moolenbroek 		break;
860*b636d99dSDavid van Moolenbroek 	case BSN_VIRTUAL_PORT_REMOVE_REQUEST:
861*b636d99dSDavid van Moolenbroek 		/*
862*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
863*b636d99dSDavid van Moolenbroek 		 *  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
864*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
865*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
866*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
867*b636d99dSDavid van Moolenbroek 		 * |                           vport_no                            |
868*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
869*b636d99dSDavid van Moolenbroek 		 *
870*b636d99dSDavid van Moolenbroek 		 */
871*b636d99dSDavid van Moolenbroek 		if (len != 8)
872*b636d99dSDavid van Moolenbroek 			goto corrupt;
873*b636d99dSDavid van Moolenbroek 		/* vport_no */
874*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
875*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", vport_no %u", EXTRACT_32BITS(cp)));
876*b636d99dSDavid van Moolenbroek 		cp += 4;
877*b636d99dSDavid van Moolenbroek 		break;
878*b636d99dSDavid van Moolenbroek 	case BSN_SHELL_COMMAND:
879*b636d99dSDavid van Moolenbroek 		/*
880*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
881*b636d99dSDavid van Moolenbroek 		 *  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
882*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
883*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
884*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
885*b636d99dSDavid van Moolenbroek 		 * |                            service                            |
886*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
887*b636d99dSDavid van Moolenbroek 		 * |                             data ...
888*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+--------
889*b636d99dSDavid van Moolenbroek 		 *
890*b636d99dSDavid van Moolenbroek 		 */
891*b636d99dSDavid van Moolenbroek 		if (len < 8)
892*b636d99dSDavid van Moolenbroek 			goto corrupt;
893*b636d99dSDavid van Moolenbroek 		/* service */
894*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
895*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", service %u", EXTRACT_32BITS(cp)));
896*b636d99dSDavid van Moolenbroek 		cp += 4;
897*b636d99dSDavid van Moolenbroek 		/* data */
898*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", data '"));
899*b636d99dSDavid van Moolenbroek 		if (fn_printn(ndo, cp, len - 8, ep)) {
900*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, "'"));
901*b636d99dSDavid van Moolenbroek 			goto trunc;
902*b636d99dSDavid van Moolenbroek 		}
903*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "'"));
904*b636d99dSDavid van Moolenbroek 		cp += len - 8;
905*b636d99dSDavid van Moolenbroek 		break;
906*b636d99dSDavid van Moolenbroek 	case BSN_SHELL_OUTPUT:
907*b636d99dSDavid van Moolenbroek 		/*
908*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
909*b636d99dSDavid van Moolenbroek 		 *  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
910*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
911*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
912*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
913*b636d99dSDavid van Moolenbroek 		 * |                             data ...
914*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+--------
915*b636d99dSDavid van Moolenbroek 		 *
916*b636d99dSDavid van Moolenbroek 		 */
917*b636d99dSDavid van Moolenbroek 		/* already checked that len >= 4 */
918*b636d99dSDavid van Moolenbroek 		/* data */
919*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", data '"));
920*b636d99dSDavid van Moolenbroek 		if (fn_printn(ndo, cp, len - 4, ep)) {
921*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, "'"));
922*b636d99dSDavid van Moolenbroek 			goto trunc;
923*b636d99dSDavid van Moolenbroek 		}
924*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "'"));
925*b636d99dSDavid van Moolenbroek 		cp += len - 4;
926*b636d99dSDavid van Moolenbroek 		break;
927*b636d99dSDavid van Moolenbroek 	case BSN_SHELL_STATUS:
928*b636d99dSDavid van Moolenbroek 		/*
929*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
930*b636d99dSDavid van Moolenbroek 		 *  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
931*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
932*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
933*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
934*b636d99dSDavid van Moolenbroek 		 * |                            status                             |
935*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
936*b636d99dSDavid van Moolenbroek 		 *
937*b636d99dSDavid van Moolenbroek 		 */
938*b636d99dSDavid van Moolenbroek 		if (len != 8)
939*b636d99dSDavid van Moolenbroek 			goto corrupt;
940*b636d99dSDavid van Moolenbroek 		/* status */
941*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
942*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", status 0x%08x", EXTRACT_32BITS(cp)));
943*b636d99dSDavid van Moolenbroek 		cp += 4;
944*b636d99dSDavid van Moolenbroek 		break;
945*b636d99dSDavid van Moolenbroek 	default:
946*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, len - 4);
947*b636d99dSDavid van Moolenbroek 		cp += len - 4;
948*b636d99dSDavid van Moolenbroek 	}
949*b636d99dSDavid van Moolenbroek 	return cp;
950*b636d99dSDavid van Moolenbroek 
951*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized data */
952*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
953*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len);
954*b636d99dSDavid van Moolenbroek 	return cp0 + len;
955*b636d99dSDavid van Moolenbroek trunc:
956*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
957*b636d99dSDavid van Moolenbroek 	return ep;
958*b636d99dSDavid van Moolenbroek }
959*b636d99dSDavid van Moolenbroek 
960*b636d99dSDavid van Moolenbroek static const u_char *
of10_bsn_actions_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)961*b636d99dSDavid van Moolenbroek of10_bsn_actions_print(netdissect_options *ndo,
962*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const u_int len)
963*b636d99dSDavid van Moolenbroek {
964*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
965*b636d99dSDavid van Moolenbroek 	uint32_t subtype, vlan_tag;
966*b636d99dSDavid van Moolenbroek 
967*b636d99dSDavid van Moolenbroek 	if (len < 4)
968*b636d99dSDavid van Moolenbroek 		goto corrupt;
969*b636d99dSDavid van Moolenbroek 	/* subtype */
970*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
971*b636d99dSDavid van Moolenbroek 	subtype = EXTRACT_32BITS(cp);
972*b636d99dSDavid van Moolenbroek 	cp += 4;
973*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  subtype %s", tok2str(bsn_action_subtype_str, "unknown (0x%08x)", subtype)));
974*b636d99dSDavid van Moolenbroek 	switch (subtype) {
975*b636d99dSDavid van Moolenbroek 	case BSN_ACTION_MIRROR:
976*b636d99dSDavid van Moolenbroek 		/*
977*b636d99dSDavid van Moolenbroek 		 *  0                   1                   2                   3
978*b636d99dSDavid van Moolenbroek 		 *  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
979*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
980*b636d99dSDavid van Moolenbroek 		 * |                            subtype                            |
981*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
982*b636d99dSDavid van Moolenbroek 		 * |                           dest_port                           |
983*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
984*b636d99dSDavid van Moolenbroek 		 * |                           vlan_tag                            |
985*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
986*b636d99dSDavid van Moolenbroek 		 * |  copy_stage   |                      pad                      |
987*b636d99dSDavid van Moolenbroek 		 * +---------------+---------------+---------------+---------------+
988*b636d99dSDavid van Moolenbroek 		 *
989*b636d99dSDavid van Moolenbroek 		 */
990*b636d99dSDavid van Moolenbroek 		if (len != 16)
991*b636d99dSDavid van Moolenbroek 			goto corrupt;
992*b636d99dSDavid van Moolenbroek 		/* dest_port */
993*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
994*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", dest_port %u", EXTRACT_32BITS(cp)));
995*b636d99dSDavid van Moolenbroek 		cp += 4;
996*b636d99dSDavid van Moolenbroek 		/* vlan_tag */
997*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
998*b636d99dSDavid van Moolenbroek 		vlan_tag = EXTRACT_32BITS(cp);
999*b636d99dSDavid van Moolenbroek 		cp += 4;
1000*b636d99dSDavid van Moolenbroek 		switch (vlan_tag >> 16) {
1001*b636d99dSDavid van Moolenbroek 		case 0:
1002*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", vlan_tag none"));
1003*b636d99dSDavid van Moolenbroek 			break;
1004*b636d99dSDavid van Moolenbroek 		case ETHERTYPE_8021Q:
1005*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", vlan_tag 802.1Q (%s)", ieee8021q_tci_string(vlan_tag & 0xffff)));
1006*b636d99dSDavid van Moolenbroek 			break;
1007*b636d99dSDavid van Moolenbroek 		default:
1008*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", vlan_tag unknown (0x%04x)", vlan_tag >> 16));
1009*b636d99dSDavid van Moolenbroek 		}
1010*b636d99dSDavid van Moolenbroek 		/* copy_stage */
1011*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
1012*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", copy_stage %s", tok2str(bsn_mirror_copy_stage_str, "unknown (%u)", *cp)));
1013*b636d99dSDavid van Moolenbroek 		cp += 1;
1014*b636d99dSDavid van Moolenbroek 		/* pad */
1015*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 3);
1016*b636d99dSDavid van Moolenbroek 		cp += 3;
1017*b636d99dSDavid van Moolenbroek 		break;
1018*b636d99dSDavid van Moolenbroek 	default:
1019*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, len - 4);
1020*b636d99dSDavid van Moolenbroek 		cp += len - 4;
1021*b636d99dSDavid van Moolenbroek 	}
1022*b636d99dSDavid van Moolenbroek 
1023*b636d99dSDavid van Moolenbroek 	return cp;
1024*b636d99dSDavid van Moolenbroek 
1025*b636d99dSDavid van Moolenbroek corrupt:
1026*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1027*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len);
1028*b636d99dSDavid van Moolenbroek 	return cp0 + len;
1029*b636d99dSDavid van Moolenbroek trunc:
1030*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1031*b636d99dSDavid van Moolenbroek 	return ep;
1032*b636d99dSDavid van Moolenbroek }
1033*b636d99dSDavid van Moolenbroek 
1034*b636d99dSDavid van Moolenbroek static const u_char *
of10_vendor_action_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1035*b636d99dSDavid van Moolenbroek of10_vendor_action_print(netdissect_options *ndo,
1036*b636d99dSDavid van Moolenbroek                          const u_char *cp, const u_char *ep, const u_int len)
1037*b636d99dSDavid van Moolenbroek {
1038*b636d99dSDavid van Moolenbroek 	uint32_t vendor;
1039*b636d99dSDavid van Moolenbroek 	const u_char *(*decoder)(netdissect_options *, const u_char *, const u_char *, const u_int);
1040*b636d99dSDavid van Moolenbroek 
1041*b636d99dSDavid van Moolenbroek 	if (len < 4)
1042*b636d99dSDavid van Moolenbroek 		goto corrupt;
1043*b636d99dSDavid van Moolenbroek 	/* vendor */
1044*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1045*b636d99dSDavid van Moolenbroek 	vendor = EXTRACT_32BITS(cp);
1046*b636d99dSDavid van Moolenbroek 	cp += 4;
1047*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", vendor 0x%08x (%s)", vendor, of_vendor_name(vendor)));
1048*b636d99dSDavid van Moolenbroek 	/* data */
1049*b636d99dSDavid van Moolenbroek 	decoder =
1050*b636d99dSDavid van Moolenbroek 		vendor == OUI_BSN         ? of10_bsn_actions_print         :
1051*b636d99dSDavid van Moolenbroek 		of10_data_print;
1052*b636d99dSDavid van Moolenbroek 	return decoder(ndo, cp, ep, len - 4);
1053*b636d99dSDavid van Moolenbroek 
1054*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized data */
1055*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1056*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1057*b636d99dSDavid van Moolenbroek 	return cp + len;
1058*b636d99dSDavid van Moolenbroek trunc:
1059*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1060*b636d99dSDavid van Moolenbroek 	return ep;
1061*b636d99dSDavid van Moolenbroek }
1062*b636d99dSDavid van Moolenbroek 
1063*b636d99dSDavid van Moolenbroek static const u_char *
of10_vendor_message_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1064*b636d99dSDavid van Moolenbroek of10_vendor_message_print(netdissect_options *ndo,
1065*b636d99dSDavid van Moolenbroek                           const u_char *cp, const u_char *ep, const u_int len)
1066*b636d99dSDavid van Moolenbroek {
1067*b636d99dSDavid van Moolenbroek 	uint32_t vendor;
1068*b636d99dSDavid van Moolenbroek 	const u_char *(*decoder)(netdissect_options *, const u_char *, const u_char *, u_int);
1069*b636d99dSDavid van Moolenbroek 
1070*b636d99dSDavid van Moolenbroek 	if (len < 4)
1071*b636d99dSDavid van Moolenbroek 		goto corrupt;
1072*b636d99dSDavid van Moolenbroek 	/* vendor */
1073*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1074*b636d99dSDavid van Moolenbroek 	vendor = EXTRACT_32BITS(cp);
1075*b636d99dSDavid van Moolenbroek 	cp += 4;
1076*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", vendor 0x%08x (%s)", vendor, of_vendor_name(vendor)));
1077*b636d99dSDavid van Moolenbroek 	/* data */
1078*b636d99dSDavid van Moolenbroek 	decoder =
1079*b636d99dSDavid van Moolenbroek 		vendor == OUI_BSN         ? of10_bsn_message_print         :
1080*b636d99dSDavid van Moolenbroek 		of10_data_print;
1081*b636d99dSDavid van Moolenbroek 	return decoder(ndo, cp, ep, len - 4);
1082*b636d99dSDavid van Moolenbroek 
1083*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized data */
1084*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1085*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1086*b636d99dSDavid van Moolenbroek 	return cp + len;
1087*b636d99dSDavid van Moolenbroek trunc:
1088*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1089*b636d99dSDavid van Moolenbroek 	return ep;
1090*b636d99dSDavid van Moolenbroek }
1091*b636d99dSDavid van Moolenbroek 
1092*b636d99dSDavid van Moolenbroek /* Vendor ID is mandatory, data is optional. */
1093*b636d99dSDavid van Moolenbroek static const u_char *
of10_vendor_data_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1094*b636d99dSDavid van Moolenbroek of10_vendor_data_print(netdissect_options *ndo,
1095*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const u_int len)
1096*b636d99dSDavid van Moolenbroek {
1097*b636d99dSDavid van Moolenbroek 	uint32_t vendor;
1098*b636d99dSDavid van Moolenbroek 
1099*b636d99dSDavid van Moolenbroek 	if (len < 4)
1100*b636d99dSDavid van Moolenbroek 		goto corrupt;
1101*b636d99dSDavid van Moolenbroek 	/* vendor */
1102*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1103*b636d99dSDavid van Moolenbroek 	vendor = EXTRACT_32BITS(cp);
1104*b636d99dSDavid van Moolenbroek 	cp += 4;
1105*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", vendor 0x%08x (%s)", vendor, of_vendor_name(vendor)));
1106*b636d99dSDavid van Moolenbroek 	/* data */
1107*b636d99dSDavid van Moolenbroek 	return of10_data_print(ndo, cp, ep, len - 4);
1108*b636d99dSDavid van Moolenbroek 
1109*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized data */
1110*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1111*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1112*b636d99dSDavid van Moolenbroek 	return cp + len;
1113*b636d99dSDavid van Moolenbroek trunc:
1114*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1115*b636d99dSDavid van Moolenbroek 	return ep;
1116*b636d99dSDavid van Moolenbroek }
1117*b636d99dSDavid van Moolenbroek 
1118*b636d99dSDavid van Moolenbroek static const u_char *
of10_packet_data_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1119*b636d99dSDavid van Moolenbroek of10_packet_data_print(netdissect_options *ndo,
1120*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const u_int len)
1121*b636d99dSDavid van Moolenbroek {
1122*b636d99dSDavid van Moolenbroek 	if (len == 0)
1123*b636d99dSDavid van Moolenbroek 		return cp;
1124*b636d99dSDavid van Moolenbroek 	/* data */
1125*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t data (%u octets)", len));
1126*b636d99dSDavid van Moolenbroek 	if (ndo->ndo_vflag < 3)
1127*b636d99dSDavid van Moolenbroek 		return cp + len;
1128*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1129*b636d99dSDavid van Moolenbroek 	ndo->ndo_vflag -= 3;
1130*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", frame decoding below\n"));
1131*b636d99dSDavid van Moolenbroek 	ether_print(ndo, cp, len, ndo->ndo_snapend - cp, NULL, NULL);
1132*b636d99dSDavid van Moolenbroek 	ndo->ndo_vflag += 3;
1133*b636d99dSDavid van Moolenbroek 	return cp + len;
1134*b636d99dSDavid van Moolenbroek 
1135*b636d99dSDavid van Moolenbroek trunc:
1136*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1137*b636d99dSDavid van Moolenbroek 	return ep;
1138*b636d99dSDavid van Moolenbroek }
1139*b636d99dSDavid van Moolenbroek 
1140*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.2.1 */
1141*b636d99dSDavid van Moolenbroek static const u_char *
of10_phy_ports_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1142*b636d99dSDavid van Moolenbroek of10_phy_ports_print(netdissect_options *ndo,
1143*b636d99dSDavid van Moolenbroek                      const u_char *cp, const u_char *ep, u_int len)
1144*b636d99dSDavid van Moolenbroek {
1145*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1146*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1147*b636d99dSDavid van Moolenbroek 
1148*b636d99dSDavid van Moolenbroek 	while (len) {
1149*b636d99dSDavid van Moolenbroek 		if (len < OF_PHY_PORT_LEN)
1150*b636d99dSDavid van Moolenbroek 			goto corrupt;
1151*b636d99dSDavid van Moolenbroek 		/* port_no */
1152*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1153*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1154*b636d99dSDavid van Moolenbroek 		cp += 2;
1155*b636d99dSDavid van Moolenbroek 		/* hw_addr */
1156*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, ETHER_ADDR_LEN);
1157*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", hw_addr %s", etheraddr_string(ndo, cp)));
1158*b636d99dSDavid van Moolenbroek 		cp += ETHER_ADDR_LEN;
1159*b636d99dSDavid van Moolenbroek 		/* name */
1160*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, OFP_MAX_PORT_NAME_LEN);
1161*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", name '"));
1162*b636d99dSDavid van Moolenbroek 		fn_print(ndo, cp, cp + OFP_MAX_PORT_NAME_LEN);
1163*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "'"));
1164*b636d99dSDavid van Moolenbroek 		cp += OFP_MAX_PORT_NAME_LEN;
1165*b636d99dSDavid van Moolenbroek 
1166*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 2) {
1167*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 24);
1168*b636d99dSDavid van Moolenbroek 			cp += 24;
1169*b636d99dSDavid van Moolenbroek 			goto next_port;
1170*b636d99dSDavid van Moolenbroek 		}
1171*b636d99dSDavid van Moolenbroek 		/* config */
1172*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1173*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   config 0x%08x", EXTRACT_32BITS(cp)));
1174*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofppc_bm, EXTRACT_32BITS(cp), OFPPC_U);
1175*b636d99dSDavid van Moolenbroek 		cp += 4;
1176*b636d99dSDavid van Moolenbroek 		/* state */
1177*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1178*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   state 0x%08x", EXTRACT_32BITS(cp)));
1179*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofpps_bm, EXTRACT_32BITS(cp), OFPPS_U);
1180*b636d99dSDavid van Moolenbroek 		cp += 4;
1181*b636d99dSDavid van Moolenbroek 		/* curr */
1182*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1183*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   curr 0x%08x", EXTRACT_32BITS(cp)));
1184*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofppf_bm, EXTRACT_32BITS(cp), OFPPF_U);
1185*b636d99dSDavid van Moolenbroek 		cp += 4;
1186*b636d99dSDavid van Moolenbroek 		/* advertised */
1187*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1188*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   advertised 0x%08x", EXTRACT_32BITS(cp)));
1189*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofppf_bm, EXTRACT_32BITS(cp), OFPPF_U);
1190*b636d99dSDavid van Moolenbroek 		cp += 4;
1191*b636d99dSDavid van Moolenbroek 		/* supported */
1192*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1193*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   supported 0x%08x", EXTRACT_32BITS(cp)));
1194*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofppf_bm, EXTRACT_32BITS(cp), OFPPF_U);
1195*b636d99dSDavid van Moolenbroek 		cp += 4;
1196*b636d99dSDavid van Moolenbroek 		/* peer */
1197*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1198*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   peer 0x%08x", EXTRACT_32BITS(cp)));
1199*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofppf_bm, EXTRACT_32BITS(cp), OFPPF_U);
1200*b636d99dSDavid van Moolenbroek 		cp += 4;
1201*b636d99dSDavid van Moolenbroek next_port:
1202*b636d99dSDavid van Moolenbroek 		len -= OF_PHY_PORT_LEN;
1203*b636d99dSDavid van Moolenbroek 	} /* while */
1204*b636d99dSDavid van Moolenbroek 	return cp;
1205*b636d99dSDavid van Moolenbroek 
1206*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized trailing data */
1207*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1208*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1209*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1210*b636d99dSDavid van Moolenbroek trunc:
1211*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1212*b636d99dSDavid van Moolenbroek 	return ep;
1213*b636d99dSDavid van Moolenbroek }
1214*b636d99dSDavid van Moolenbroek 
1215*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.2.2 */
1216*b636d99dSDavid van Moolenbroek static const u_char *
of10_queue_props_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1217*b636d99dSDavid van Moolenbroek of10_queue_props_print(netdissect_options *ndo,
1218*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, u_int len)
1219*b636d99dSDavid van Moolenbroek {
1220*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1221*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1222*b636d99dSDavid van Moolenbroek 	uint16_t property, plen, rate;
1223*b636d99dSDavid van Moolenbroek 
1224*b636d99dSDavid van Moolenbroek 	while (len) {
1225*b636d99dSDavid van Moolenbroek 		u_char plen_bogus = 0, skip = 0;
1226*b636d99dSDavid van Moolenbroek 
1227*b636d99dSDavid van Moolenbroek 		if (len < OF_QUEUE_PROP_HEADER_LEN)
1228*b636d99dSDavid van Moolenbroek 			goto corrupt;
1229*b636d99dSDavid van Moolenbroek 		/* property */
1230*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1231*b636d99dSDavid van Moolenbroek 		property = EXTRACT_16BITS(cp);
1232*b636d99dSDavid van Moolenbroek 		cp += 2;
1233*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t   property %s", tok2str(ofpqt_str, "invalid (0x%04x)", property)));
1234*b636d99dSDavid van Moolenbroek 		/* len */
1235*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1236*b636d99dSDavid van Moolenbroek 		plen = EXTRACT_16BITS(cp);
1237*b636d99dSDavid van Moolenbroek 		cp += 2;
1238*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", len %u", plen));
1239*b636d99dSDavid van Moolenbroek 		if (plen < OF_QUEUE_PROP_HEADER_LEN || plen > len)
1240*b636d99dSDavid van Moolenbroek 			goto corrupt;
1241*b636d99dSDavid van Moolenbroek 		/* pad */
1242*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1243*b636d99dSDavid van Moolenbroek 		cp += 4;
1244*b636d99dSDavid van Moolenbroek 		/* property-specific constraints and decoding */
1245*b636d99dSDavid van Moolenbroek 		switch (property) {
1246*b636d99dSDavid van Moolenbroek 		case OFPQT_NONE:
1247*b636d99dSDavid van Moolenbroek 			plen_bogus = plen != OF_QUEUE_PROP_HEADER_LEN;
1248*b636d99dSDavid van Moolenbroek 			break;
1249*b636d99dSDavid van Moolenbroek 		case OFPQT_MIN_RATE:
1250*b636d99dSDavid van Moolenbroek 			plen_bogus = plen != OF_QUEUE_PROP_MIN_RATE_LEN;
1251*b636d99dSDavid van Moolenbroek 			break;
1252*b636d99dSDavid van Moolenbroek 		default:
1253*b636d99dSDavid van Moolenbroek 			skip = 1;
1254*b636d99dSDavid van Moolenbroek 		}
1255*b636d99dSDavid van Moolenbroek 		if (plen_bogus) {
1256*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, " (bogus)"));
1257*b636d99dSDavid van Moolenbroek 			skip = 1;
1258*b636d99dSDavid van Moolenbroek 		}
1259*b636d99dSDavid van Moolenbroek 		if (skip) {
1260*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, plen - 4);
1261*b636d99dSDavid van Moolenbroek 			cp += plen - 4;
1262*b636d99dSDavid van Moolenbroek 			goto next_property;
1263*b636d99dSDavid van Moolenbroek 		}
1264*b636d99dSDavid van Moolenbroek 		if (property == OFPQT_MIN_RATE) { /* the only case of property decoding */
1265*b636d99dSDavid van Moolenbroek 			/* rate */
1266*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1267*b636d99dSDavid van Moolenbroek 			rate = EXTRACT_16BITS(cp);
1268*b636d99dSDavid van Moolenbroek 			cp += 2;
1269*b636d99dSDavid van Moolenbroek 			if (rate > 1000)
1270*b636d99dSDavid van Moolenbroek 				ND_PRINT((ndo, ", rate disabled"));
1271*b636d99dSDavid van Moolenbroek 			else
1272*b636d99dSDavid van Moolenbroek 				ND_PRINT((ndo, ", rate %u.%u%%", rate / 10, rate % 10));
1273*b636d99dSDavid van Moolenbroek 			/* pad */
1274*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 6);
1275*b636d99dSDavid van Moolenbroek 			cp += 6;
1276*b636d99dSDavid van Moolenbroek 		}
1277*b636d99dSDavid van Moolenbroek next_property:
1278*b636d99dSDavid van Moolenbroek 		len -= plen;
1279*b636d99dSDavid van Moolenbroek 	} /* while */
1280*b636d99dSDavid van Moolenbroek 	return cp;
1281*b636d99dSDavid van Moolenbroek 
1282*b636d99dSDavid van Moolenbroek corrupt: /* skip the rest of queue properties */
1283*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1284*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1285*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1286*b636d99dSDavid van Moolenbroek trunc:
1287*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1288*b636d99dSDavid van Moolenbroek 	return ep;
1289*b636d99dSDavid van Moolenbroek }
1290*b636d99dSDavid van Moolenbroek 
1291*b636d99dSDavid van Moolenbroek /* ibid */
1292*b636d99dSDavid van Moolenbroek static const u_char *
of10_queues_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1293*b636d99dSDavid van Moolenbroek of10_queues_print(netdissect_options *ndo,
1294*b636d99dSDavid van Moolenbroek                   const u_char *cp, const u_char *ep, u_int len)
1295*b636d99dSDavid van Moolenbroek {
1296*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1297*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1298*b636d99dSDavid van Moolenbroek 	uint16_t desclen;
1299*b636d99dSDavid van Moolenbroek 
1300*b636d99dSDavid van Moolenbroek 	while (len) {
1301*b636d99dSDavid van Moolenbroek 		if (len < OF_PACKET_QUEUE_LEN)
1302*b636d99dSDavid van Moolenbroek 			goto corrupt;
1303*b636d99dSDavid van Moolenbroek 		/* queue_id */
1304*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1305*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  queue_id %u", EXTRACT_32BITS(cp)));
1306*b636d99dSDavid van Moolenbroek 		cp += 4;
1307*b636d99dSDavid van Moolenbroek 		/* len */
1308*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1309*b636d99dSDavid van Moolenbroek 		desclen = EXTRACT_16BITS(cp);
1310*b636d99dSDavid van Moolenbroek 		cp += 2;
1311*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", len %u", desclen));
1312*b636d99dSDavid van Moolenbroek 		if (desclen < OF_PACKET_QUEUE_LEN || desclen > len)
1313*b636d99dSDavid van Moolenbroek 			goto corrupt;
1314*b636d99dSDavid van Moolenbroek 		/* pad */
1315*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1316*b636d99dSDavid van Moolenbroek 		cp += 2;
1317*b636d99dSDavid van Moolenbroek 		/* properties */
1318*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 2) {
1319*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, desclen - OF_PACKET_QUEUE_LEN);
1320*b636d99dSDavid van Moolenbroek 			cp += desclen - OF_PACKET_QUEUE_LEN;
1321*b636d99dSDavid van Moolenbroek 			goto next_queue;
1322*b636d99dSDavid van Moolenbroek 		}
1323*b636d99dSDavid van Moolenbroek 		if (ep == (cp = of10_queue_props_print(ndo, cp, ep, desclen - OF_PACKET_QUEUE_LEN)))
1324*b636d99dSDavid van Moolenbroek 			return ep; /* end of snapshot */
1325*b636d99dSDavid van Moolenbroek next_queue:
1326*b636d99dSDavid van Moolenbroek 		len -= desclen;
1327*b636d99dSDavid van Moolenbroek 	} /* while */
1328*b636d99dSDavid van Moolenbroek 	return cp;
1329*b636d99dSDavid van Moolenbroek 
1330*b636d99dSDavid van Moolenbroek corrupt: /* skip the rest of queues */
1331*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1332*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1333*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1334*b636d99dSDavid van Moolenbroek trunc:
1335*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1336*b636d99dSDavid van Moolenbroek 	return ep;
1337*b636d99dSDavid van Moolenbroek }
1338*b636d99dSDavid van Moolenbroek 
1339*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.2.3 */
1340*b636d99dSDavid van Moolenbroek static const u_char *
of10_match_print(netdissect_options * ndo,const char * pfx,const u_char * cp,const u_char * ep)1341*b636d99dSDavid van Moolenbroek of10_match_print(netdissect_options *ndo,
1342*b636d99dSDavid van Moolenbroek                  const char *pfx, const u_char *cp, const u_char *ep)
1343*b636d99dSDavid van Moolenbroek {
1344*b636d99dSDavid van Moolenbroek 	uint32_t wildcards;
1345*b636d99dSDavid van Moolenbroek 	uint16_t dl_type;
1346*b636d99dSDavid van Moolenbroek 	uint8_t nw_proto;
1347*b636d99dSDavid van Moolenbroek 	u_char nw_bits;
1348*b636d99dSDavid van Moolenbroek 	const char *field_name;
1349*b636d99dSDavid van Moolenbroek 
1350*b636d99dSDavid van Moolenbroek 	/* wildcards */
1351*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1352*b636d99dSDavid van Moolenbroek 	wildcards = EXTRACT_32BITS(cp);
1353*b636d99dSDavid van Moolenbroek 	if (wildcards & OFPFW_U)
1354*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%swildcards 0x%08x (bogus)", pfx, wildcards));
1355*b636d99dSDavid van Moolenbroek 	cp += 4;
1356*b636d99dSDavid van Moolenbroek 	/* in_port */
1357*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1358*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_IN_PORT))
1359*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch in_port %s", pfx, tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1360*b636d99dSDavid van Moolenbroek 	cp += 2;
1361*b636d99dSDavid van Moolenbroek 	/* dl_src */
1362*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, ETHER_ADDR_LEN);
1363*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_DL_SRC))
1364*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch dl_src %s", pfx, etheraddr_string(ndo, cp)));
1365*b636d99dSDavid van Moolenbroek 	cp += ETHER_ADDR_LEN;
1366*b636d99dSDavid van Moolenbroek 	/* dl_dst */
1367*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, ETHER_ADDR_LEN);
1368*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_DL_DST))
1369*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch dl_dst %s", pfx, etheraddr_string(ndo, cp)));
1370*b636d99dSDavid van Moolenbroek 	cp += ETHER_ADDR_LEN;
1371*b636d99dSDavid van Moolenbroek 	/* dl_vlan */
1372*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1373*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_DL_VLAN))
1374*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch dl_vlan %s", pfx, vlan_str(EXTRACT_16BITS(cp))));
1375*b636d99dSDavid van Moolenbroek 	cp += 2;
1376*b636d99dSDavid van Moolenbroek 	/* dl_vlan_pcp */
1377*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
1378*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_DL_VLAN_PCP))
1379*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch dl_vlan_pcp %s", pfx, pcp_str(*cp)));
1380*b636d99dSDavid van Moolenbroek 	cp += 1;
1381*b636d99dSDavid van Moolenbroek 	/* pad1 */
1382*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
1383*b636d99dSDavid van Moolenbroek 	cp += 1;
1384*b636d99dSDavid van Moolenbroek 	/* dl_type */
1385*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1386*b636d99dSDavid van Moolenbroek 	dl_type = EXTRACT_16BITS(cp);
1387*b636d99dSDavid van Moolenbroek 	cp += 2;
1388*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_DL_TYPE))
1389*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch dl_type 0x%04x", pfx, dl_type));
1390*b636d99dSDavid van Moolenbroek 	/* nw_tos */
1391*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
1392*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_NW_TOS))
1393*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch nw_tos 0x%02x", pfx, *cp));
1394*b636d99dSDavid van Moolenbroek 	cp += 1;
1395*b636d99dSDavid van Moolenbroek 	/* nw_proto */
1396*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
1397*b636d99dSDavid van Moolenbroek 	nw_proto = *cp;
1398*b636d99dSDavid van Moolenbroek 	cp += 1;
1399*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_NW_PROTO)) {
1400*b636d99dSDavid van Moolenbroek 		field_name = ! (wildcards & OFPFW_DL_TYPE) && dl_type == ETHERTYPE_ARP
1401*b636d99dSDavid van Moolenbroek 		  ? "arp_opcode" : "nw_proto";
1402*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch %s %u", pfx, field_name, nw_proto));
1403*b636d99dSDavid van Moolenbroek 	}
1404*b636d99dSDavid van Moolenbroek 	/* pad2 */
1405*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1406*b636d99dSDavid van Moolenbroek 	cp += 2;
1407*b636d99dSDavid van Moolenbroek 	/* nw_src */
1408*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1409*b636d99dSDavid van Moolenbroek 	nw_bits = (wildcards & OFPFW_NW_SRC_MASK) >> OFPFW_NW_SRC_SHIFT;
1410*b636d99dSDavid van Moolenbroek 	if (nw_bits < 32)
1411*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch nw_src %s/%u", pfx, ipaddr_string(ndo, cp), 32 - nw_bits));
1412*b636d99dSDavid van Moolenbroek 	cp += 4;
1413*b636d99dSDavid van Moolenbroek 	/* nw_dst */
1414*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1415*b636d99dSDavid van Moolenbroek 	nw_bits = (wildcards & OFPFW_NW_DST_MASK) >> OFPFW_NW_DST_SHIFT;
1416*b636d99dSDavid van Moolenbroek 	if (nw_bits < 32)
1417*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch nw_dst %s/%u", pfx, ipaddr_string(ndo, cp), 32 - nw_bits));
1418*b636d99dSDavid van Moolenbroek 	cp += 4;
1419*b636d99dSDavid van Moolenbroek 	/* tp_src */
1420*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1421*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_TP_SRC)) {
1422*b636d99dSDavid van Moolenbroek 		field_name = ! (wildcards & OFPFW_DL_TYPE) && dl_type == ETHERTYPE_IP
1423*b636d99dSDavid van Moolenbroek 		  && ! (wildcards & OFPFW_NW_PROTO) && nw_proto == IPPROTO_ICMP
1424*b636d99dSDavid van Moolenbroek 		  ? "icmp_type" : "tp_src";
1425*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch %s %u", pfx, field_name, EXTRACT_16BITS(cp)));
1426*b636d99dSDavid van Moolenbroek 	}
1427*b636d99dSDavid van Moolenbroek 	cp += 2;
1428*b636d99dSDavid van Moolenbroek 	/* tp_dst */
1429*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1430*b636d99dSDavid van Moolenbroek 	if (! (wildcards & OFPFW_TP_DST)) {
1431*b636d99dSDavid van Moolenbroek 		field_name = ! (wildcards & OFPFW_DL_TYPE) && dl_type == ETHERTYPE_IP
1432*b636d99dSDavid van Moolenbroek 		  && ! (wildcards & OFPFW_NW_PROTO) && nw_proto == IPPROTO_ICMP
1433*b636d99dSDavid van Moolenbroek 		  ? "icmp_code" : "tp_dst";
1434*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%smatch %s %u", pfx, field_name, EXTRACT_16BITS(cp)));
1435*b636d99dSDavid van Moolenbroek 	}
1436*b636d99dSDavid van Moolenbroek 	return cp + 2;
1437*b636d99dSDavid van Moolenbroek 
1438*b636d99dSDavid van Moolenbroek trunc:
1439*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1440*b636d99dSDavid van Moolenbroek 	return ep;
1441*b636d99dSDavid van Moolenbroek }
1442*b636d99dSDavid van Moolenbroek 
1443*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.2.4 */
1444*b636d99dSDavid van Moolenbroek static const u_char *
of10_actions_print(netdissect_options * ndo,const char * pfx,const u_char * cp,const u_char * ep,u_int len)1445*b636d99dSDavid van Moolenbroek of10_actions_print(netdissect_options *ndo,
1446*b636d99dSDavid van Moolenbroek                    const char *pfx, const u_char *cp, const u_char *ep,
1447*b636d99dSDavid van Moolenbroek                    u_int len)
1448*b636d99dSDavid van Moolenbroek {
1449*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1450*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1451*b636d99dSDavid van Moolenbroek 	uint16_t type, alen, output_port;
1452*b636d99dSDavid van Moolenbroek 
1453*b636d99dSDavid van Moolenbroek 	while (len) {
1454*b636d99dSDavid van Moolenbroek 		u_char alen_bogus = 0, skip = 0;
1455*b636d99dSDavid van Moolenbroek 
1456*b636d99dSDavid van Moolenbroek 		if (len < OF_ACTION_HEADER_LEN)
1457*b636d99dSDavid van Moolenbroek 			goto corrupt;
1458*b636d99dSDavid van Moolenbroek 		/* type */
1459*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1460*b636d99dSDavid van Moolenbroek 		type = EXTRACT_16BITS(cp);
1461*b636d99dSDavid van Moolenbroek 		cp += 2;
1462*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "%saction type %s", pfx, tok2str(ofpat_str, "invalid (0x%04x)", type)));
1463*b636d99dSDavid van Moolenbroek 		/* length */
1464*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1465*b636d99dSDavid van Moolenbroek 		alen = EXTRACT_16BITS(cp);
1466*b636d99dSDavid van Moolenbroek 		cp += 2;
1467*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", len %u", alen));
1468*b636d99dSDavid van Moolenbroek 		/* On action size underrun/overrun skip the rest of the action list. */
1469*b636d99dSDavid van Moolenbroek 		if (alen < OF_ACTION_HEADER_LEN || alen > len)
1470*b636d99dSDavid van Moolenbroek 			goto corrupt;
1471*b636d99dSDavid van Moolenbroek 		/* On action size inappropriate for the given type or invalid type just skip
1472*b636d99dSDavid van Moolenbroek 		 * the current action, as the basic length constraint has been met. */
1473*b636d99dSDavid van Moolenbroek 		switch (type) {
1474*b636d99dSDavid van Moolenbroek 		case OFPAT_OUTPUT:
1475*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_VLAN_VID:
1476*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_VLAN_PCP:
1477*b636d99dSDavid van Moolenbroek 		case OFPAT_STRIP_VLAN:
1478*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_SRC:
1479*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_DST:
1480*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_TOS:
1481*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_TP_SRC:
1482*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_TP_DST:
1483*b636d99dSDavid van Moolenbroek 			alen_bogus = alen != 8;
1484*b636d99dSDavid van Moolenbroek 			break;
1485*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_DL_SRC:
1486*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_DL_DST:
1487*b636d99dSDavid van Moolenbroek 		case OFPAT_ENQUEUE:
1488*b636d99dSDavid van Moolenbroek 			alen_bogus = alen != 16;
1489*b636d99dSDavid van Moolenbroek 			break;
1490*b636d99dSDavid van Moolenbroek 		case OFPAT_VENDOR:
1491*b636d99dSDavid van Moolenbroek 			alen_bogus = alen % 8 != 0; /* already >= 8 so far */
1492*b636d99dSDavid van Moolenbroek 			break;
1493*b636d99dSDavid van Moolenbroek 		default:
1494*b636d99dSDavid van Moolenbroek 			skip = 1;
1495*b636d99dSDavid van Moolenbroek 		}
1496*b636d99dSDavid van Moolenbroek 		if (alen_bogus) {
1497*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, " (bogus)"));
1498*b636d99dSDavid van Moolenbroek 			skip = 1;
1499*b636d99dSDavid van Moolenbroek 		}
1500*b636d99dSDavid van Moolenbroek 		if (skip) {
1501*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, alen - 4);
1502*b636d99dSDavid van Moolenbroek 			cp += alen - 4;
1503*b636d99dSDavid van Moolenbroek 			goto next_action;
1504*b636d99dSDavid van Moolenbroek 		}
1505*b636d99dSDavid van Moolenbroek 		/* OK to decode the rest of the action structure */
1506*b636d99dSDavid van Moolenbroek 		switch (type) {
1507*b636d99dSDavid van Moolenbroek 		case OFPAT_OUTPUT:
1508*b636d99dSDavid van Moolenbroek 			/* port */
1509*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1510*b636d99dSDavid van Moolenbroek 			output_port = EXTRACT_16BITS(cp);
1511*b636d99dSDavid van Moolenbroek 			cp += 2;
1512*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", port %s", tok2str(ofpp_str, "%u", output_port)));
1513*b636d99dSDavid van Moolenbroek 			/* max_len */
1514*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1515*b636d99dSDavid van Moolenbroek 			if (output_port == OFPP_CONTROLLER)
1516*b636d99dSDavid van Moolenbroek 				ND_PRINT((ndo, ", max_len %u", EXTRACT_16BITS(cp)));
1517*b636d99dSDavid van Moolenbroek 			cp += 2;
1518*b636d99dSDavid van Moolenbroek 			break;
1519*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_VLAN_VID:
1520*b636d99dSDavid van Moolenbroek 			/* vlan_vid */
1521*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1522*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", vlan_vid %s", vlan_str(EXTRACT_16BITS(cp))));
1523*b636d99dSDavid van Moolenbroek 			cp += 2;
1524*b636d99dSDavid van Moolenbroek 			/* pad */
1525*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1526*b636d99dSDavid van Moolenbroek 			cp += 2;
1527*b636d99dSDavid van Moolenbroek 			break;
1528*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_VLAN_PCP:
1529*b636d99dSDavid van Moolenbroek 			/* vlan_pcp */
1530*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 1);
1531*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", vlan_pcp %s", pcp_str(*cp)));
1532*b636d99dSDavid van Moolenbroek 			cp += 1;
1533*b636d99dSDavid van Moolenbroek 			/* pad */
1534*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 3);
1535*b636d99dSDavid van Moolenbroek 			cp += 3;
1536*b636d99dSDavid van Moolenbroek 			break;
1537*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_DL_SRC:
1538*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_DL_DST:
1539*b636d99dSDavid van Moolenbroek 			/* dl_addr */
1540*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, ETHER_ADDR_LEN);
1541*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", dl_addr %s", etheraddr_string(ndo, cp)));
1542*b636d99dSDavid van Moolenbroek 			cp += ETHER_ADDR_LEN;
1543*b636d99dSDavid van Moolenbroek 			/* pad */
1544*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 6);
1545*b636d99dSDavid van Moolenbroek 			cp += 6;
1546*b636d99dSDavid van Moolenbroek 			break;
1547*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_SRC:
1548*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_DST:
1549*b636d99dSDavid van Moolenbroek 			/* nw_addr */
1550*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 4);
1551*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", nw_addr %s", ipaddr_string(ndo, cp)));
1552*b636d99dSDavid van Moolenbroek 			cp += 4;
1553*b636d99dSDavid van Moolenbroek 			break;
1554*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_NW_TOS:
1555*b636d99dSDavid van Moolenbroek 			/* nw_tos */
1556*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 1);
1557*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", nw_tos 0x%02x", *cp));
1558*b636d99dSDavid van Moolenbroek 			cp += 1;
1559*b636d99dSDavid van Moolenbroek 			/* pad */
1560*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 3);
1561*b636d99dSDavid van Moolenbroek 			cp += 3;
1562*b636d99dSDavid van Moolenbroek 			break;
1563*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_TP_SRC:
1564*b636d99dSDavid van Moolenbroek 		case OFPAT_SET_TP_DST:
1565*b636d99dSDavid van Moolenbroek 			/* nw_tos */
1566*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1567*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", tp_port %u", EXTRACT_16BITS(cp)));
1568*b636d99dSDavid van Moolenbroek 			cp += 2;
1569*b636d99dSDavid van Moolenbroek 			/* pad */
1570*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1571*b636d99dSDavid van Moolenbroek 			cp += 2;
1572*b636d99dSDavid van Moolenbroek 			break;
1573*b636d99dSDavid van Moolenbroek 		case OFPAT_ENQUEUE:
1574*b636d99dSDavid van Moolenbroek 			/* port */
1575*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 2);
1576*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", port %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1577*b636d99dSDavid van Moolenbroek 			cp += 2;
1578*b636d99dSDavid van Moolenbroek 			/* pad */
1579*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 6);
1580*b636d99dSDavid van Moolenbroek 			cp += 6;
1581*b636d99dSDavid van Moolenbroek 			/* queue_id */
1582*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 4);
1583*b636d99dSDavid van Moolenbroek 			ND_PRINT((ndo, ", queue_id %s", tok2str(ofpq_str, "%u", EXTRACT_32BITS(cp))));
1584*b636d99dSDavid van Moolenbroek 			cp += 4;
1585*b636d99dSDavid van Moolenbroek 			break;
1586*b636d99dSDavid van Moolenbroek 		case OFPAT_VENDOR:
1587*b636d99dSDavid van Moolenbroek 			if (ep == (cp = of10_vendor_action_print(ndo, cp, ep, alen - 4)))
1588*b636d99dSDavid van Moolenbroek 				return ep; /* end of snapshot */
1589*b636d99dSDavid van Moolenbroek 			break;
1590*b636d99dSDavid van Moolenbroek 		case OFPAT_STRIP_VLAN:
1591*b636d99dSDavid van Moolenbroek 			/* pad */
1592*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, 4);
1593*b636d99dSDavid van Moolenbroek 			cp += 4;
1594*b636d99dSDavid van Moolenbroek 			break;
1595*b636d99dSDavid van Moolenbroek 		} /* switch */
1596*b636d99dSDavid van Moolenbroek next_action:
1597*b636d99dSDavid van Moolenbroek 		len -= alen;
1598*b636d99dSDavid van Moolenbroek 	} /* while */
1599*b636d99dSDavid van Moolenbroek 	return cp;
1600*b636d99dSDavid van Moolenbroek 
1601*b636d99dSDavid van Moolenbroek corrupt: /* skip the rest of actions */
1602*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1603*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1604*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1605*b636d99dSDavid van Moolenbroek trunc:
1606*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1607*b636d99dSDavid van Moolenbroek 	return ep;
1608*b636d99dSDavid van Moolenbroek }
1609*b636d99dSDavid van Moolenbroek 
1610*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.3.1 */
1611*b636d99dSDavid van Moolenbroek static const u_char *
of10_features_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1612*b636d99dSDavid van Moolenbroek of10_features_reply_print(netdissect_options *ndo,
1613*b636d99dSDavid van Moolenbroek                           const u_char *cp, const u_char *ep, const u_int len)
1614*b636d99dSDavid van Moolenbroek {
1615*b636d99dSDavid van Moolenbroek 	/* datapath_id */
1616*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
1617*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t dpid 0x%016" PRIx64, EXTRACT_64BITS(cp)));
1618*b636d99dSDavid van Moolenbroek 	cp += 8;
1619*b636d99dSDavid van Moolenbroek 	/* n_buffers */
1620*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1621*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", n_buffers %u", EXTRACT_32BITS(cp)));
1622*b636d99dSDavid van Moolenbroek 	cp += 4;
1623*b636d99dSDavid van Moolenbroek 	/* n_tables */
1624*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
1625*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", n_tables %u", *cp));
1626*b636d99dSDavid van Moolenbroek 	cp += 1;
1627*b636d99dSDavid van Moolenbroek 	/* pad */
1628*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 3);
1629*b636d99dSDavid van Moolenbroek 	cp += 3;
1630*b636d99dSDavid van Moolenbroek 	/* capabilities */
1631*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1632*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t capabilities 0x%08x", EXTRACT_32BITS(cp)));
1633*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofp_capabilities_bm, EXTRACT_32BITS(cp), OFPCAP_U);
1634*b636d99dSDavid van Moolenbroek 	cp += 4;
1635*b636d99dSDavid van Moolenbroek 	/* actions */
1636*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1637*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t actions 0x%08x", EXTRACT_32BITS(cp)));
1638*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofpat_bm, EXTRACT_32BITS(cp), OFPAT_U);
1639*b636d99dSDavid van Moolenbroek 	cp += 4;
1640*b636d99dSDavid van Moolenbroek 	/* ports */
1641*b636d99dSDavid van Moolenbroek 	return of10_phy_ports_print(ndo, cp, ep, len - OF_SWITCH_FEATURES_LEN);
1642*b636d99dSDavid van Moolenbroek 
1643*b636d99dSDavid van Moolenbroek trunc:
1644*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1645*b636d99dSDavid van Moolenbroek 	return ep;
1646*b636d99dSDavid van Moolenbroek }
1647*b636d99dSDavid van Moolenbroek 
1648*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.3.3 */
1649*b636d99dSDavid van Moolenbroek static const u_char *
of10_flow_mod_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1650*b636d99dSDavid van Moolenbroek of10_flow_mod_print(netdissect_options *ndo,
1651*b636d99dSDavid van Moolenbroek                     const u_char *cp, const u_char *ep, const u_int len)
1652*b636d99dSDavid van Moolenbroek {
1653*b636d99dSDavid van Moolenbroek 	uint16_t command;
1654*b636d99dSDavid van Moolenbroek 
1655*b636d99dSDavid van Moolenbroek 	/* match */
1656*b636d99dSDavid van Moolenbroek 	if (ep == (cp = of10_match_print(ndo, "\n\t ", cp, ep)))
1657*b636d99dSDavid van Moolenbroek 		return ep; /* end of snapshot */
1658*b636d99dSDavid van Moolenbroek 	/* cookie */
1659*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
1660*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t cookie 0x%016" PRIx64, EXTRACT_64BITS(cp)));
1661*b636d99dSDavid van Moolenbroek 	cp += 8;
1662*b636d99dSDavid van Moolenbroek 	/* command */
1663*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1664*b636d99dSDavid van Moolenbroek 	command = EXTRACT_16BITS(cp);
1665*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", command %s", tok2str(ofpfc_str, "invalid (0x%04x)", command)));
1666*b636d99dSDavid van Moolenbroek 	cp += 2;
1667*b636d99dSDavid van Moolenbroek 	/* idle_timeout */
1668*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1669*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
1670*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", idle_timeout %u", EXTRACT_16BITS(cp)));
1671*b636d99dSDavid van Moolenbroek 	cp += 2;
1672*b636d99dSDavid van Moolenbroek 	/* hard_timeout */
1673*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1674*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
1675*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", hard_timeout %u", EXTRACT_16BITS(cp)));
1676*b636d99dSDavid van Moolenbroek 	cp += 2;
1677*b636d99dSDavid van Moolenbroek 	/* priority */
1678*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1679*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
1680*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", priority %u", EXTRACT_16BITS(cp)));
1681*b636d99dSDavid van Moolenbroek 	cp += 2;
1682*b636d99dSDavid van Moolenbroek 	/* buffer_id */
1683*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1684*b636d99dSDavid van Moolenbroek 	if (command == OFPFC_ADD || command == OFPFC_MODIFY ||
1685*b636d99dSDavid van Moolenbroek 	    command == OFPFC_MODIFY_STRICT)
1686*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", buffer_id %s", tok2str(bufferid_str, "0x%08x", EXTRACT_32BITS(cp))));
1687*b636d99dSDavid van Moolenbroek 	cp += 4;
1688*b636d99dSDavid van Moolenbroek 	/* out_port */
1689*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1690*b636d99dSDavid van Moolenbroek 	if (command == OFPFC_DELETE || command == OFPFC_DELETE_STRICT)
1691*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", out_port %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1692*b636d99dSDavid van Moolenbroek 	cp += 2;
1693*b636d99dSDavid van Moolenbroek 	/* flags */
1694*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1695*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", flags 0x%04x", EXTRACT_16BITS(cp)));
1696*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofpff_bm, EXTRACT_16BITS(cp), OFPFF_U);
1697*b636d99dSDavid van Moolenbroek 	cp += 2;
1698*b636d99dSDavid van Moolenbroek 	/* actions */
1699*b636d99dSDavid van Moolenbroek 	return of10_actions_print(ndo, "\n\t ", cp, ep, len - OF_FLOW_MOD_LEN);
1700*b636d99dSDavid van Moolenbroek 
1701*b636d99dSDavid van Moolenbroek trunc:
1702*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1703*b636d99dSDavid van Moolenbroek 	return ep;
1704*b636d99dSDavid van Moolenbroek }
1705*b636d99dSDavid van Moolenbroek 
1706*b636d99dSDavid van Moolenbroek /* ibid */
1707*b636d99dSDavid van Moolenbroek static const u_char *
of10_port_mod_print(netdissect_options * ndo,const u_char * cp,const u_char * ep)1708*b636d99dSDavid van Moolenbroek of10_port_mod_print(netdissect_options *ndo,
1709*b636d99dSDavid van Moolenbroek                     const u_char *cp, const u_char *ep)
1710*b636d99dSDavid van Moolenbroek {
1711*b636d99dSDavid van Moolenbroek 	/* port_no */
1712*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1713*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1714*b636d99dSDavid van Moolenbroek 	cp += 2;
1715*b636d99dSDavid van Moolenbroek 	/* hw_addr */
1716*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, ETHER_ADDR_LEN);
1717*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", hw_addr %s", etheraddr_string(ndo, cp)));
1718*b636d99dSDavid van Moolenbroek 	cp += ETHER_ADDR_LEN;
1719*b636d99dSDavid van Moolenbroek 	/* config */
1720*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1721*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t config 0x%08x", EXTRACT_32BITS(cp)));
1722*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofppc_bm, EXTRACT_32BITS(cp), OFPPC_U);
1723*b636d99dSDavid van Moolenbroek 	cp += 4;
1724*b636d99dSDavid van Moolenbroek 	/* mask */
1725*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1726*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t mask 0x%08x", EXTRACT_32BITS(cp)));
1727*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofppc_bm, EXTRACT_32BITS(cp), OFPPC_U);
1728*b636d99dSDavid van Moolenbroek 	cp += 4;
1729*b636d99dSDavid van Moolenbroek 	/* advertise */
1730*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1731*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t advertise 0x%08x", EXTRACT_32BITS(cp)));
1732*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofppf_bm, EXTRACT_32BITS(cp), OFPPF_U);
1733*b636d99dSDavid van Moolenbroek 	cp += 4;
1734*b636d99dSDavid van Moolenbroek 	/* pad */
1735*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1736*b636d99dSDavid van Moolenbroek 	return cp + 4;
1737*b636d99dSDavid van Moolenbroek 
1738*b636d99dSDavid van Moolenbroek trunc:
1739*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1740*b636d99dSDavid van Moolenbroek 	return ep;
1741*b636d99dSDavid van Moolenbroek }
1742*b636d99dSDavid van Moolenbroek 
1743*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.3.5 */
1744*b636d99dSDavid van Moolenbroek static const u_char *
of10_stats_request_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1745*b636d99dSDavid van Moolenbroek of10_stats_request_print(netdissect_options *ndo,
1746*b636d99dSDavid van Moolenbroek                          const u_char *cp, const u_char *ep, u_int len)
1747*b636d99dSDavid van Moolenbroek {
1748*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1749*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1750*b636d99dSDavid van Moolenbroek 	uint16_t type;
1751*b636d99dSDavid van Moolenbroek 
1752*b636d99dSDavid van Moolenbroek 	/* type */
1753*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1754*b636d99dSDavid van Moolenbroek 	type = EXTRACT_16BITS(cp);
1755*b636d99dSDavid van Moolenbroek 	cp += 2;
1756*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t type %s", tok2str(ofpst_str, "invalid (0x%04x)", type)));
1757*b636d99dSDavid van Moolenbroek 	/* flags */
1758*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
1759*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", flags 0x%04x", EXTRACT_16BITS(cp)));
1760*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
1761*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, " (bogus)"));
1762*b636d99dSDavid van Moolenbroek 	cp += 2;
1763*b636d99dSDavid van Moolenbroek 	/* type-specific body of one of fixed lengths */
1764*b636d99dSDavid van Moolenbroek 	len -= OF_STATS_REQUEST_LEN;
1765*b636d99dSDavid van Moolenbroek 	switch(type) {
1766*b636d99dSDavid van Moolenbroek 	case OFPST_DESC:
1767*b636d99dSDavid van Moolenbroek 	case OFPST_TABLE:
1768*b636d99dSDavid van Moolenbroek 		if (len)
1769*b636d99dSDavid van Moolenbroek 			goto corrupt;
1770*b636d99dSDavid van Moolenbroek 		return cp;
1771*b636d99dSDavid van Moolenbroek 	case OFPST_FLOW:
1772*b636d99dSDavid van Moolenbroek 	case OFPST_AGGREGATE:
1773*b636d99dSDavid van Moolenbroek 		if (len != OF_FLOW_STATS_REQUEST_LEN)
1774*b636d99dSDavid van Moolenbroek 			goto corrupt;
1775*b636d99dSDavid van Moolenbroek 		/* match */
1776*b636d99dSDavid van Moolenbroek 		if (ep == (cp = of10_match_print(ndo, "\n\t ", cp, ep)))
1777*b636d99dSDavid van Moolenbroek 			return ep; /* end of snapshot */
1778*b636d99dSDavid van Moolenbroek 		/* table_id */
1779*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
1780*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t table_id %s", tok2str(tableid_str, "%u", *cp)));
1781*b636d99dSDavid van Moolenbroek 		cp += 1;
1782*b636d99dSDavid van Moolenbroek 		/* pad */
1783*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
1784*b636d99dSDavid van Moolenbroek 		cp += 1;
1785*b636d99dSDavid van Moolenbroek 		/* out_port */
1786*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1787*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", out_port %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1788*b636d99dSDavid van Moolenbroek 		return cp + 2;
1789*b636d99dSDavid van Moolenbroek 	case OFPST_PORT:
1790*b636d99dSDavid van Moolenbroek 		if (len != OF_PORT_STATS_REQUEST_LEN)
1791*b636d99dSDavid van Moolenbroek 			goto corrupt;
1792*b636d99dSDavid van Moolenbroek 		/* port_no */
1793*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1794*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1795*b636d99dSDavid van Moolenbroek 		cp += 2;
1796*b636d99dSDavid van Moolenbroek 		/* pad */
1797*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 6);
1798*b636d99dSDavid van Moolenbroek 		return cp + 6;
1799*b636d99dSDavid van Moolenbroek 	case OFPST_QUEUE:
1800*b636d99dSDavid van Moolenbroek 		if (len != OF_QUEUE_STATS_REQUEST_LEN)
1801*b636d99dSDavid van Moolenbroek 			goto corrupt;
1802*b636d99dSDavid van Moolenbroek 		/* port_no */
1803*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1804*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
1805*b636d99dSDavid van Moolenbroek 		cp += 2;
1806*b636d99dSDavid van Moolenbroek 		/* pad */
1807*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1808*b636d99dSDavid van Moolenbroek 		cp += 2;
1809*b636d99dSDavid van Moolenbroek 		/* queue_id */
1810*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1811*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", queue_id %s", tok2str(ofpq_str, "%u", EXTRACT_32BITS(cp))));
1812*b636d99dSDavid van Moolenbroek 		return cp + 4;
1813*b636d99dSDavid van Moolenbroek 	case OFPST_VENDOR:
1814*b636d99dSDavid van Moolenbroek 		return of10_vendor_data_print(ndo, cp, ep, len);
1815*b636d99dSDavid van Moolenbroek 	}
1816*b636d99dSDavid van Moolenbroek 	return cp;
1817*b636d99dSDavid van Moolenbroek 
1818*b636d99dSDavid van Moolenbroek corrupt: /* skip the message body */
1819*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1820*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1821*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1822*b636d99dSDavid van Moolenbroek trunc:
1823*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1824*b636d99dSDavid van Moolenbroek 	return ep;
1825*b636d99dSDavid van Moolenbroek }
1826*b636d99dSDavid van Moolenbroek 
1827*b636d99dSDavid van Moolenbroek /* ibid */
1828*b636d99dSDavid van Moolenbroek static const u_char *
of10_desc_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1829*b636d99dSDavid van Moolenbroek of10_desc_stats_reply_print(netdissect_options *ndo,
1830*b636d99dSDavid van Moolenbroek                             const u_char *cp, const u_char *ep, const u_int len)
1831*b636d99dSDavid van Moolenbroek {
1832*b636d99dSDavid van Moolenbroek 	if (len != OF_DESC_STATS_LEN)
1833*b636d99dSDavid van Moolenbroek 		goto corrupt;
1834*b636d99dSDavid van Moolenbroek 	/* mfr_desc */
1835*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, DESC_STR_LEN);
1836*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  mfr_desc '"));
1837*b636d99dSDavid van Moolenbroek 	fn_print(ndo, cp, cp + DESC_STR_LEN);
1838*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "'"));
1839*b636d99dSDavid van Moolenbroek 	cp += DESC_STR_LEN;
1840*b636d99dSDavid van Moolenbroek 	/* hw_desc */
1841*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, DESC_STR_LEN);
1842*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  hw_desc '"));
1843*b636d99dSDavid van Moolenbroek 	fn_print(ndo, cp, cp + DESC_STR_LEN);
1844*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "'"));
1845*b636d99dSDavid van Moolenbroek 	cp += DESC_STR_LEN;
1846*b636d99dSDavid van Moolenbroek 	/* sw_desc */
1847*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, DESC_STR_LEN);
1848*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  sw_desc '"));
1849*b636d99dSDavid van Moolenbroek 	fn_print(ndo, cp, cp + DESC_STR_LEN);
1850*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "'"));
1851*b636d99dSDavid van Moolenbroek 	cp += DESC_STR_LEN;
1852*b636d99dSDavid van Moolenbroek 	/* serial_num */
1853*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, SERIAL_NUM_LEN);
1854*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  serial_num '"));
1855*b636d99dSDavid van Moolenbroek 	fn_print(ndo, cp, cp + SERIAL_NUM_LEN);
1856*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "'"));
1857*b636d99dSDavid van Moolenbroek 	cp += SERIAL_NUM_LEN;
1858*b636d99dSDavid van Moolenbroek 	/* dp_desc */
1859*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, DESC_STR_LEN);
1860*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t  dp_desc '"));
1861*b636d99dSDavid van Moolenbroek 	fn_print(ndo, cp, cp + DESC_STR_LEN);
1862*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "'"));
1863*b636d99dSDavid van Moolenbroek 	return cp + DESC_STR_LEN;
1864*b636d99dSDavid van Moolenbroek 
1865*b636d99dSDavid van Moolenbroek corrupt: /* skip the message body */
1866*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1867*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1868*b636d99dSDavid van Moolenbroek 	return cp + len;
1869*b636d99dSDavid van Moolenbroek trunc:
1870*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1871*b636d99dSDavid van Moolenbroek 	return ep;
1872*b636d99dSDavid van Moolenbroek }
1873*b636d99dSDavid van Moolenbroek 
1874*b636d99dSDavid van Moolenbroek /* ibid */
1875*b636d99dSDavid van Moolenbroek static const u_char *
of10_flow_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1876*b636d99dSDavid van Moolenbroek of10_flow_stats_reply_print(netdissect_options *ndo,
1877*b636d99dSDavid van Moolenbroek                             const u_char *cp, const u_char *ep, u_int len)
1878*b636d99dSDavid van Moolenbroek {
1879*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1880*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1881*b636d99dSDavid van Moolenbroek 	uint16_t entry_len;
1882*b636d99dSDavid van Moolenbroek 
1883*b636d99dSDavid van Moolenbroek 	while (len) {
1884*b636d99dSDavid van Moolenbroek 		if (len < OF_FLOW_STATS_LEN)
1885*b636d99dSDavid van Moolenbroek 			goto corrupt;
1886*b636d99dSDavid van Moolenbroek 		/* length */
1887*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1888*b636d99dSDavid van Moolenbroek 		entry_len = EXTRACT_16BITS(cp);
1889*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t length %u", entry_len));
1890*b636d99dSDavid van Moolenbroek 		if (entry_len < OF_FLOW_STATS_LEN || entry_len > len)
1891*b636d99dSDavid van Moolenbroek 			goto corrupt;
1892*b636d99dSDavid van Moolenbroek 		cp += 2;
1893*b636d99dSDavid van Moolenbroek 		/* table_id */
1894*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
1895*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", table_id %s", tok2str(tableid_str, "%u", *cp)));
1896*b636d99dSDavid van Moolenbroek 		cp += 1;
1897*b636d99dSDavid van Moolenbroek 		/* pad */
1898*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
1899*b636d99dSDavid van Moolenbroek 		cp += 1;
1900*b636d99dSDavid van Moolenbroek 		/* match */
1901*b636d99dSDavid van Moolenbroek 		if (ep == (cp = of10_match_print(ndo, "\n\t  ", cp, ep)))
1902*b636d99dSDavid van Moolenbroek 			return ep; /* end of snapshot */
1903*b636d99dSDavid van Moolenbroek 		/* duration_sec */
1904*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1905*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  duration_sec %u", EXTRACT_32BITS(cp)));
1906*b636d99dSDavid van Moolenbroek 		cp += 4;
1907*b636d99dSDavid van Moolenbroek 		/* duration_nsec */
1908*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
1909*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", duration_nsec %u", EXTRACT_32BITS(cp)));
1910*b636d99dSDavid van Moolenbroek 		cp += 4;
1911*b636d99dSDavid van Moolenbroek 		/* priority */
1912*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1913*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", priority %u", EXTRACT_16BITS(cp)));
1914*b636d99dSDavid van Moolenbroek 		cp += 2;
1915*b636d99dSDavid van Moolenbroek 		/* idle_timeout */
1916*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1917*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", idle_timeout %u", EXTRACT_16BITS(cp)));
1918*b636d99dSDavid van Moolenbroek 		cp += 2;
1919*b636d99dSDavid van Moolenbroek 		/* hard_timeout */
1920*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
1921*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", hard_timeout %u", EXTRACT_16BITS(cp)));
1922*b636d99dSDavid van Moolenbroek 		cp += 2;
1923*b636d99dSDavid van Moolenbroek 		/* pad2 */
1924*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 6);
1925*b636d99dSDavid van Moolenbroek 		cp += 6;
1926*b636d99dSDavid van Moolenbroek 		/* cookie */
1927*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
1928*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", cookie 0x%016" PRIx64, EXTRACT_64BITS(cp)));
1929*b636d99dSDavid van Moolenbroek 		cp += 8;
1930*b636d99dSDavid van Moolenbroek 		/* packet_count */
1931*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
1932*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", packet_count %" PRIu64, EXTRACT_64BITS(cp)));
1933*b636d99dSDavid van Moolenbroek 		cp += 8;
1934*b636d99dSDavid van Moolenbroek 		/* byte_count */
1935*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
1936*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", byte_count %" PRIu64, EXTRACT_64BITS(cp)));
1937*b636d99dSDavid van Moolenbroek 		cp += 8;
1938*b636d99dSDavid van Moolenbroek 		/* actions */
1939*b636d99dSDavid van Moolenbroek 		if (ep == (cp = of10_actions_print(ndo, "\n\t  ", cp, ep, entry_len - OF_FLOW_STATS_LEN)))
1940*b636d99dSDavid van Moolenbroek 			return ep; /* end of snapshot */
1941*b636d99dSDavid van Moolenbroek 
1942*b636d99dSDavid van Moolenbroek 		len -= entry_len;
1943*b636d99dSDavid van Moolenbroek 	} /* while */
1944*b636d99dSDavid van Moolenbroek 	return cp;
1945*b636d99dSDavid van Moolenbroek 
1946*b636d99dSDavid van Moolenbroek corrupt: /* skip the rest of flow statistics entries */
1947*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1948*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
1949*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
1950*b636d99dSDavid van Moolenbroek trunc:
1951*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1952*b636d99dSDavid van Moolenbroek 	return ep;
1953*b636d99dSDavid van Moolenbroek }
1954*b636d99dSDavid van Moolenbroek 
1955*b636d99dSDavid van Moolenbroek /* ibid */
1956*b636d99dSDavid van Moolenbroek static const u_char *
of10_aggregate_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)1957*b636d99dSDavid van Moolenbroek of10_aggregate_stats_reply_print(netdissect_options *ndo,
1958*b636d99dSDavid van Moolenbroek                                  const u_char *cp, const u_char *ep,
1959*b636d99dSDavid van Moolenbroek                                  const u_int len)
1960*b636d99dSDavid van Moolenbroek {
1961*b636d99dSDavid van Moolenbroek 	if (len != OF_AGGREGATE_STATS_REPLY_LEN)
1962*b636d99dSDavid van Moolenbroek 		goto corrupt;
1963*b636d99dSDavid van Moolenbroek 	/* packet_count */
1964*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
1965*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t packet_count %" PRIu64, EXTRACT_64BITS(cp)));
1966*b636d99dSDavid van Moolenbroek 	cp += 8;
1967*b636d99dSDavid van Moolenbroek 	/* byte_count */
1968*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
1969*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", byte_count %" PRIu64, EXTRACT_64BITS(cp)));
1970*b636d99dSDavid van Moolenbroek 	cp += 8;
1971*b636d99dSDavid van Moolenbroek 	/* flow_count */
1972*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1973*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", flow_count %u", EXTRACT_32BITS(cp)));
1974*b636d99dSDavid van Moolenbroek 	cp += 4;
1975*b636d99dSDavid van Moolenbroek 	/* pad */
1976*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
1977*b636d99dSDavid van Moolenbroek 	return cp + 4;
1978*b636d99dSDavid van Moolenbroek 
1979*b636d99dSDavid van Moolenbroek corrupt: /* skip the message body */
1980*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
1981*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, len);
1982*b636d99dSDavid van Moolenbroek 	return cp + len;
1983*b636d99dSDavid van Moolenbroek trunc:
1984*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
1985*b636d99dSDavid van Moolenbroek 	return ep;
1986*b636d99dSDavid van Moolenbroek }
1987*b636d99dSDavid van Moolenbroek 
1988*b636d99dSDavid van Moolenbroek /* ibid */
1989*b636d99dSDavid van Moolenbroek static const u_char *
of10_table_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)1990*b636d99dSDavid van Moolenbroek of10_table_stats_reply_print(netdissect_options *ndo,
1991*b636d99dSDavid van Moolenbroek                              const u_char *cp, const u_char *ep, u_int len)
1992*b636d99dSDavid van Moolenbroek {
1993*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
1994*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
1995*b636d99dSDavid van Moolenbroek 
1996*b636d99dSDavid van Moolenbroek 	while (len) {
1997*b636d99dSDavid van Moolenbroek 		if (len < OF_TABLE_STATS_LEN)
1998*b636d99dSDavid van Moolenbroek 			goto corrupt;
1999*b636d99dSDavid van Moolenbroek 		/* table_id */
2000*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
2001*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t table_id %s", tok2str(tableid_str, "%u", *cp)));
2002*b636d99dSDavid van Moolenbroek 		cp += 1;
2003*b636d99dSDavid van Moolenbroek 		/* pad */
2004*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 3);
2005*b636d99dSDavid van Moolenbroek 		cp += 3;
2006*b636d99dSDavid van Moolenbroek 		/* name */
2007*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, OFP_MAX_TABLE_NAME_LEN);
2008*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", name '"));
2009*b636d99dSDavid van Moolenbroek 		fn_print(ndo, cp, cp + OFP_MAX_TABLE_NAME_LEN);
2010*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "'"));
2011*b636d99dSDavid van Moolenbroek 		cp += OFP_MAX_TABLE_NAME_LEN;
2012*b636d99dSDavid van Moolenbroek 		/* wildcards */
2013*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
2014*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  wildcards 0x%08x", EXTRACT_32BITS(cp)));
2015*b636d99dSDavid van Moolenbroek 		of10_bitmap_print(ndo, ofpfw_bm, EXTRACT_32BITS(cp), OFPFW_U);
2016*b636d99dSDavid van Moolenbroek 		cp += 4;
2017*b636d99dSDavid van Moolenbroek 		/* max_entries */
2018*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
2019*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  max_entries %u", EXTRACT_32BITS(cp)));
2020*b636d99dSDavid van Moolenbroek 		cp += 4;
2021*b636d99dSDavid van Moolenbroek 		/* active_count */
2022*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
2023*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", active_count %u", EXTRACT_32BITS(cp)));
2024*b636d99dSDavid van Moolenbroek 		cp += 4;
2025*b636d99dSDavid van Moolenbroek 		/* lookup_count */
2026*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2027*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", lookup_count %" PRIu64, EXTRACT_64BITS(cp)));
2028*b636d99dSDavid van Moolenbroek 		cp += 8;
2029*b636d99dSDavid van Moolenbroek 		/* matched_count */
2030*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2031*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", matched_count %" PRIu64, EXTRACT_64BITS(cp)));
2032*b636d99dSDavid van Moolenbroek 		cp += 8;
2033*b636d99dSDavid van Moolenbroek 
2034*b636d99dSDavid van Moolenbroek 		len -= OF_TABLE_STATS_LEN;
2035*b636d99dSDavid van Moolenbroek 	} /* while */
2036*b636d99dSDavid van Moolenbroek 	return cp;
2037*b636d99dSDavid van Moolenbroek 
2038*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized trailing data */
2039*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
2040*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
2041*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
2042*b636d99dSDavid van Moolenbroek trunc:
2043*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2044*b636d99dSDavid van Moolenbroek 	return ep;
2045*b636d99dSDavid van Moolenbroek }
2046*b636d99dSDavid van Moolenbroek 
2047*b636d99dSDavid van Moolenbroek /* ibid */
2048*b636d99dSDavid van Moolenbroek static const u_char *
of10_port_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)2049*b636d99dSDavid van Moolenbroek of10_port_stats_reply_print(netdissect_options *ndo,
2050*b636d99dSDavid van Moolenbroek                             const u_char *cp, const u_char *ep, u_int len)
2051*b636d99dSDavid van Moolenbroek {
2052*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
2053*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
2054*b636d99dSDavid van Moolenbroek 
2055*b636d99dSDavid van Moolenbroek 	while (len) {
2056*b636d99dSDavid van Moolenbroek 		if (len < OF_PORT_STATS_LEN)
2057*b636d99dSDavid van Moolenbroek 			goto corrupt;
2058*b636d99dSDavid van Moolenbroek 		/* port_no */
2059*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2060*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2061*b636d99dSDavid van Moolenbroek 		cp += 2;
2062*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 2) {
2063*b636d99dSDavid van Moolenbroek 			ND_TCHECK2(*cp, OF_PORT_STATS_LEN - 2);
2064*b636d99dSDavid van Moolenbroek 			cp += OF_PORT_STATS_LEN - 2;
2065*b636d99dSDavid van Moolenbroek 			goto next_port;
2066*b636d99dSDavid van Moolenbroek 		}
2067*b636d99dSDavid van Moolenbroek 		/* pad */
2068*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 6);
2069*b636d99dSDavid van Moolenbroek 		cp += 6;
2070*b636d99dSDavid van Moolenbroek 		/* rx_packets */
2071*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2072*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_packets %" PRIu64, EXTRACT_64BITS(cp)));
2073*b636d99dSDavid van Moolenbroek 		cp += 8;
2074*b636d99dSDavid van Moolenbroek 		/* tx_packets */
2075*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2076*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_packets %" PRIu64, EXTRACT_64BITS(cp)));
2077*b636d99dSDavid van Moolenbroek 		cp += 8;
2078*b636d99dSDavid van Moolenbroek 		/* rx_bytes */
2079*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2080*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_bytes %" PRIu64, EXTRACT_64BITS(cp)));
2081*b636d99dSDavid van Moolenbroek 		cp += 8;
2082*b636d99dSDavid van Moolenbroek 		/* tx_bytes */
2083*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2084*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_bytes %" PRIu64, EXTRACT_64BITS(cp)));
2085*b636d99dSDavid van Moolenbroek 		cp += 8;
2086*b636d99dSDavid van Moolenbroek 		/* rx_dropped */
2087*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2088*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_dropped %" PRIu64, EXTRACT_64BITS(cp)));
2089*b636d99dSDavid van Moolenbroek 		cp += 8;
2090*b636d99dSDavid van Moolenbroek 		/* tx_dropped */
2091*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2092*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_dropped %" PRIu64, EXTRACT_64BITS(cp)));
2093*b636d99dSDavid van Moolenbroek 		cp += 8;
2094*b636d99dSDavid van Moolenbroek 		/* rx_errors */
2095*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2096*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_errors %" PRIu64, EXTRACT_64BITS(cp)));
2097*b636d99dSDavid van Moolenbroek 		cp += 8;
2098*b636d99dSDavid van Moolenbroek 		/* tx_errors */
2099*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2100*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_errors %" PRIu64, EXTRACT_64BITS(cp)));
2101*b636d99dSDavid van Moolenbroek 		cp += 8;
2102*b636d99dSDavid van Moolenbroek 		/* rx_frame_err */
2103*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2104*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_frame_err %" PRIu64, EXTRACT_64BITS(cp)));
2105*b636d99dSDavid van Moolenbroek 		cp += 8;
2106*b636d99dSDavid van Moolenbroek 		/* rx_over_err */
2107*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2108*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_over_err %" PRIu64, EXTRACT_64BITS(cp)));
2109*b636d99dSDavid van Moolenbroek 		cp += 8;
2110*b636d99dSDavid van Moolenbroek 		/* rx_crc_err */
2111*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2112*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", rx_crc_err %" PRIu64, EXTRACT_64BITS(cp)));
2113*b636d99dSDavid van Moolenbroek 		cp += 8;
2114*b636d99dSDavid van Moolenbroek 		/* collisions */
2115*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2116*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", collisions %" PRIu64, EXTRACT_64BITS(cp)));
2117*b636d99dSDavid van Moolenbroek 		cp += 8;
2118*b636d99dSDavid van Moolenbroek next_port:
2119*b636d99dSDavid van Moolenbroek 		len -= OF_PORT_STATS_LEN;
2120*b636d99dSDavid van Moolenbroek 	} /* while */
2121*b636d99dSDavid van Moolenbroek 	return cp;
2122*b636d99dSDavid van Moolenbroek 
2123*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized trailing data */
2124*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
2125*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
2126*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
2127*b636d99dSDavid van Moolenbroek trunc:
2128*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2129*b636d99dSDavid van Moolenbroek 	return ep;
2130*b636d99dSDavid van Moolenbroek }
2131*b636d99dSDavid van Moolenbroek 
2132*b636d99dSDavid van Moolenbroek /* ibid */
2133*b636d99dSDavid van Moolenbroek static const u_char *
of10_queue_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,u_int len)2134*b636d99dSDavid van Moolenbroek of10_queue_stats_reply_print(netdissect_options *ndo,
2135*b636d99dSDavid van Moolenbroek                              const u_char *cp, const u_char *ep, u_int len)
2136*b636d99dSDavid van Moolenbroek {
2137*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
2138*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
2139*b636d99dSDavid van Moolenbroek 
2140*b636d99dSDavid van Moolenbroek 	while (len) {
2141*b636d99dSDavid van Moolenbroek 		if (len < OF_QUEUE_STATS_LEN)
2142*b636d99dSDavid van Moolenbroek 			goto corrupt;
2143*b636d99dSDavid van Moolenbroek 		/* port_no */
2144*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2145*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t  port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2146*b636d99dSDavid van Moolenbroek 		cp += 2;
2147*b636d99dSDavid van Moolenbroek 		/* pad */
2148*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2149*b636d99dSDavid van Moolenbroek 		cp += 2;
2150*b636d99dSDavid van Moolenbroek 		/* queue_id */
2151*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 4);
2152*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", queue_id %u", EXTRACT_32BITS(cp)));
2153*b636d99dSDavid van Moolenbroek 		cp += 4;
2154*b636d99dSDavid van Moolenbroek 		/* tx_bytes */
2155*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2156*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_bytes %" PRIu64, EXTRACT_64BITS(cp)));
2157*b636d99dSDavid van Moolenbroek 		cp += 8;
2158*b636d99dSDavid van Moolenbroek 		/* tx_packets */
2159*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2160*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_packets %" PRIu64, EXTRACT_64BITS(cp)));
2161*b636d99dSDavid van Moolenbroek 		cp += 8;
2162*b636d99dSDavid van Moolenbroek 		/* tx_errors */
2163*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 8);
2164*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", tx_errors %" PRIu64, EXTRACT_64BITS(cp)));
2165*b636d99dSDavid van Moolenbroek 		cp += 8;
2166*b636d99dSDavid van Moolenbroek 
2167*b636d99dSDavid van Moolenbroek 		len -= OF_QUEUE_STATS_LEN;
2168*b636d99dSDavid van Moolenbroek 	} /* while */
2169*b636d99dSDavid van Moolenbroek 	return cp;
2170*b636d99dSDavid van Moolenbroek 
2171*b636d99dSDavid van Moolenbroek corrupt: /* skip the undersized trailing data */
2172*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
2173*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
2174*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
2175*b636d99dSDavid van Moolenbroek trunc:
2176*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2177*b636d99dSDavid van Moolenbroek 	return ep;
2178*b636d99dSDavid van Moolenbroek }
2179*b636d99dSDavid van Moolenbroek 
2180*b636d99dSDavid van Moolenbroek /* ibid */
2181*b636d99dSDavid van Moolenbroek static const u_char *
of10_stats_reply_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)2182*b636d99dSDavid van Moolenbroek of10_stats_reply_print(netdissect_options *ndo,
2183*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const u_int len)
2184*b636d99dSDavid van Moolenbroek {
2185*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
2186*b636d99dSDavid van Moolenbroek 	uint16_t type;
2187*b636d99dSDavid van Moolenbroek 
2188*b636d99dSDavid van Moolenbroek 	/* type */
2189*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2190*b636d99dSDavid van Moolenbroek 	type = EXTRACT_16BITS(cp);
2191*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t type %s", tok2str(ofpst_str, "invalid (0x%04x)", type)));
2192*b636d99dSDavid van Moolenbroek 	cp += 2;
2193*b636d99dSDavid van Moolenbroek 	/* flags */
2194*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2195*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", flags 0x%04x", EXTRACT_16BITS(cp)));
2196*b636d99dSDavid van Moolenbroek 	of10_bitmap_print(ndo, ofpsf_reply_bm, EXTRACT_16BITS(cp), OFPSF_REPLY_U);
2197*b636d99dSDavid van Moolenbroek 	cp += 2;
2198*b636d99dSDavid van Moolenbroek 
2199*b636d99dSDavid van Moolenbroek 	if (ndo->ndo_vflag > 0) {
2200*b636d99dSDavid van Moolenbroek 		const u_char *(*decoder)(netdissect_options *, const u_char *, const u_char *, u_int) =
2201*b636d99dSDavid van Moolenbroek 			type == OFPST_DESC      ? of10_desc_stats_reply_print      :
2202*b636d99dSDavid van Moolenbroek 			type == OFPST_FLOW      ? of10_flow_stats_reply_print      :
2203*b636d99dSDavid van Moolenbroek 			type == OFPST_AGGREGATE ? of10_aggregate_stats_reply_print :
2204*b636d99dSDavid van Moolenbroek 			type == OFPST_TABLE     ? of10_table_stats_reply_print     :
2205*b636d99dSDavid van Moolenbroek 			type == OFPST_PORT      ? of10_port_stats_reply_print      :
2206*b636d99dSDavid van Moolenbroek 			type == OFPST_QUEUE     ? of10_queue_stats_reply_print     :
2207*b636d99dSDavid van Moolenbroek 			type == OFPST_VENDOR    ? of10_vendor_data_print           :
2208*b636d99dSDavid van Moolenbroek 			NULL;
2209*b636d99dSDavid van Moolenbroek 		if (decoder != NULL)
2210*b636d99dSDavid van Moolenbroek 			return decoder(ndo, cp, ep, len - OF_STATS_REPLY_LEN);
2211*b636d99dSDavid van Moolenbroek 	}
2212*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len);
2213*b636d99dSDavid van Moolenbroek 	return cp0 + len;
2214*b636d99dSDavid van Moolenbroek 
2215*b636d99dSDavid van Moolenbroek trunc:
2216*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2217*b636d99dSDavid van Moolenbroek 	return ep;
2218*b636d99dSDavid van Moolenbroek }
2219*b636d99dSDavid van Moolenbroek 
2220*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.3.6 */
2221*b636d99dSDavid van Moolenbroek static const u_char *
of10_packet_out_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)2222*b636d99dSDavid van Moolenbroek of10_packet_out_print(netdissect_options *ndo,
2223*b636d99dSDavid van Moolenbroek                       const u_char *cp, const u_char *ep, const u_int len)
2224*b636d99dSDavid van Moolenbroek {
2225*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
2226*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
2227*b636d99dSDavid van Moolenbroek 	uint16_t actions_len;
2228*b636d99dSDavid van Moolenbroek 
2229*b636d99dSDavid van Moolenbroek 	/* buffer_id */
2230*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
2231*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t buffer_id 0x%08x", EXTRACT_32BITS(cp)));
2232*b636d99dSDavid van Moolenbroek 	cp += 4;
2233*b636d99dSDavid van Moolenbroek 	/* in_port */
2234*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2235*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", in_port %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2236*b636d99dSDavid van Moolenbroek 	cp += 2;
2237*b636d99dSDavid van Moolenbroek 	/* actions_len */
2238*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2239*b636d99dSDavid van Moolenbroek 	actions_len = EXTRACT_16BITS(cp);
2240*b636d99dSDavid van Moolenbroek 	cp += 2;
2241*b636d99dSDavid van Moolenbroek 	if (actions_len > len - OF_PACKET_OUT_LEN)
2242*b636d99dSDavid van Moolenbroek 		goto corrupt;
2243*b636d99dSDavid van Moolenbroek 	/* actions */
2244*b636d99dSDavid van Moolenbroek 	if (ep == (cp = of10_actions_print(ndo, "\n\t ", cp, ep, actions_len)))
2245*b636d99dSDavid van Moolenbroek 		return ep; /* end of snapshot */
2246*b636d99dSDavid van Moolenbroek 	/* data */
2247*b636d99dSDavid van Moolenbroek 	return of10_packet_data_print(ndo, cp, ep, len - OF_PACKET_OUT_LEN - actions_len);
2248*b636d99dSDavid van Moolenbroek 
2249*b636d99dSDavid van Moolenbroek corrupt: /* skip the rest of the message body */
2250*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
2251*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0);
2252*b636d99dSDavid van Moolenbroek 	return cp0 + len0;
2253*b636d99dSDavid van Moolenbroek trunc:
2254*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2255*b636d99dSDavid van Moolenbroek 	return ep;
2256*b636d99dSDavid van Moolenbroek }
2257*b636d99dSDavid van Moolenbroek 
2258*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.4.1 */
2259*b636d99dSDavid van Moolenbroek static const u_char *
of10_packet_in_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)2260*b636d99dSDavid van Moolenbroek of10_packet_in_print(netdissect_options *ndo,
2261*b636d99dSDavid van Moolenbroek                      const u_char *cp, const u_char *ep, const u_int len)
2262*b636d99dSDavid van Moolenbroek {
2263*b636d99dSDavid van Moolenbroek 	/* buffer_id */
2264*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
2265*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t buffer_id %s", tok2str(bufferid_str, "0x%08x", EXTRACT_32BITS(cp))));
2266*b636d99dSDavid van Moolenbroek 	cp += 4;
2267*b636d99dSDavid van Moolenbroek 	/* total_len */
2268*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2269*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", total_len %u", EXTRACT_16BITS(cp)));
2270*b636d99dSDavid van Moolenbroek 	cp += 2;
2271*b636d99dSDavid van Moolenbroek 	/* in_port */
2272*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2273*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", in_port %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2274*b636d99dSDavid van Moolenbroek 	cp += 2;
2275*b636d99dSDavid van Moolenbroek 	/* reason */
2276*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
2277*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", reason %s", tok2str(ofpr_str, "invalid (0x%02x)", *cp)));
2278*b636d99dSDavid van Moolenbroek 	cp += 1;
2279*b636d99dSDavid van Moolenbroek 	/* pad */
2280*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
2281*b636d99dSDavid van Moolenbroek 	cp += 1;
2282*b636d99dSDavid van Moolenbroek 	/* data */
2283*b636d99dSDavid van Moolenbroek 	/* 2 mock octets count in OF_PACKET_IN_LEN but not in len */
2284*b636d99dSDavid van Moolenbroek 	return of10_packet_data_print(ndo, cp, ep, len - (OF_PACKET_IN_LEN - 2));
2285*b636d99dSDavid van Moolenbroek 
2286*b636d99dSDavid van Moolenbroek trunc:
2287*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2288*b636d99dSDavid van Moolenbroek 	return ep;
2289*b636d99dSDavid van Moolenbroek }
2290*b636d99dSDavid van Moolenbroek 
2291*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.4.2 */
2292*b636d99dSDavid van Moolenbroek static const u_char *
of10_flow_removed_print(netdissect_options * ndo,const u_char * cp,const u_char * ep)2293*b636d99dSDavid van Moolenbroek of10_flow_removed_print(netdissect_options *ndo,
2294*b636d99dSDavid van Moolenbroek                         const u_char *cp, const u_char *ep)
2295*b636d99dSDavid van Moolenbroek {
2296*b636d99dSDavid van Moolenbroek 	/* match */
2297*b636d99dSDavid van Moolenbroek 	if (ep == (cp = of10_match_print(ndo, "\n\t ", cp, ep)))
2298*b636d99dSDavid van Moolenbroek 		return ep; /* end of snapshot */
2299*b636d99dSDavid van Moolenbroek 	/* cookie */
2300*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
2301*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t cookie 0x%016" PRIx64, EXTRACT_64BITS(cp)));
2302*b636d99dSDavid van Moolenbroek 	cp += 8;
2303*b636d99dSDavid van Moolenbroek 	/* priority */
2304*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2305*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
2306*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", priority %u", EXTRACT_16BITS(cp)));
2307*b636d99dSDavid van Moolenbroek 	cp += 2;
2308*b636d99dSDavid van Moolenbroek 	/* reason */
2309*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
2310*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", reason %s", tok2str(ofprr_str, "unknown (0x%02x)", *cp)));
2311*b636d99dSDavid van Moolenbroek 	cp += 1;
2312*b636d99dSDavid van Moolenbroek 	/* pad */
2313*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 1);
2314*b636d99dSDavid van Moolenbroek 	cp += 1;
2315*b636d99dSDavid van Moolenbroek 	/* duration_sec */
2316*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
2317*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", duration_sec %u", EXTRACT_32BITS(cp)));
2318*b636d99dSDavid van Moolenbroek 	cp += 4;
2319*b636d99dSDavid van Moolenbroek 	/* duration_nsec */
2320*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 4);
2321*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", duration_nsec %u", EXTRACT_32BITS(cp)));
2322*b636d99dSDavid van Moolenbroek 	cp += 4;
2323*b636d99dSDavid van Moolenbroek 	/* idle_timeout */
2324*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2325*b636d99dSDavid van Moolenbroek 	if (EXTRACT_16BITS(cp))
2326*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", idle_timeout %u", EXTRACT_16BITS(cp)));
2327*b636d99dSDavid van Moolenbroek 	cp += 2;
2328*b636d99dSDavid van Moolenbroek 	/* pad2 */
2329*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2330*b636d99dSDavid van Moolenbroek 	cp += 2;
2331*b636d99dSDavid van Moolenbroek 	/* packet_count */
2332*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
2333*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", packet_count %" PRIu64, EXTRACT_64BITS(cp)));
2334*b636d99dSDavid van Moolenbroek 	cp += 8;
2335*b636d99dSDavid van Moolenbroek 	/* byte_count */
2336*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 8);
2337*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", byte_count %" PRIu64, EXTRACT_64BITS(cp)));
2338*b636d99dSDavid van Moolenbroek 	return cp + 8;
2339*b636d99dSDavid van Moolenbroek 
2340*b636d99dSDavid van Moolenbroek trunc:
2341*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2342*b636d99dSDavid van Moolenbroek 	return ep;
2343*b636d99dSDavid van Moolenbroek }
2344*b636d99dSDavid van Moolenbroek 
2345*b636d99dSDavid van Moolenbroek /* [OF10] Section 5.4.4 */
2346*b636d99dSDavid van Moolenbroek static const u_char *
of10_error_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const u_int len)2347*b636d99dSDavid van Moolenbroek of10_error_print(netdissect_options *ndo,
2348*b636d99dSDavid van Moolenbroek                  const u_char *cp, const u_char *ep, const u_int len)
2349*b636d99dSDavid van Moolenbroek {
2350*b636d99dSDavid van Moolenbroek 	uint16_t type;
2351*b636d99dSDavid van Moolenbroek 	const struct tok *code_str;
2352*b636d99dSDavid van Moolenbroek 
2353*b636d99dSDavid van Moolenbroek 	/* type */
2354*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2355*b636d99dSDavid van Moolenbroek 	type = EXTRACT_16BITS(cp);
2356*b636d99dSDavid van Moolenbroek 	cp += 2;
2357*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\t type %s", tok2str(ofpet_str, "invalid (0x%04x)", type)));
2358*b636d99dSDavid van Moolenbroek 	/* code */
2359*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp, 2);
2360*b636d99dSDavid van Moolenbroek 	code_str =
2361*b636d99dSDavid van Moolenbroek 		type == OFPET_HELLO_FAILED    ? ofphfc_str  :
2362*b636d99dSDavid van Moolenbroek 		type == OFPET_BAD_REQUEST     ? ofpbrc_str  :
2363*b636d99dSDavid van Moolenbroek 		type == OFPET_BAD_ACTION      ? ofpbac_str  :
2364*b636d99dSDavid van Moolenbroek 		type == OFPET_FLOW_MOD_FAILED ? ofpfmfc_str :
2365*b636d99dSDavid van Moolenbroek 		type == OFPET_PORT_MOD_FAILED ? ofppmfc_str :
2366*b636d99dSDavid van Moolenbroek 		type == OFPET_QUEUE_OP_FAILED ? ofpqofc_str :
2367*b636d99dSDavid van Moolenbroek 		empty_str;
2368*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, ", code %s", tok2str(code_str, "invalid (0x%04x)", EXTRACT_16BITS(cp))));
2369*b636d99dSDavid van Moolenbroek 	cp += 2;
2370*b636d99dSDavid van Moolenbroek 	/* data */
2371*b636d99dSDavid van Moolenbroek 	return of10_data_print(ndo, cp, ep, len - OF_ERROR_MSG_LEN);
2372*b636d99dSDavid van Moolenbroek 
2373*b636d99dSDavid van Moolenbroek trunc:
2374*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2375*b636d99dSDavid van Moolenbroek 	return ep;
2376*b636d99dSDavid van Moolenbroek }
2377*b636d99dSDavid van Moolenbroek 
2378*b636d99dSDavid van Moolenbroek const u_char *
of10_header_body_print(netdissect_options * ndo,const u_char * cp,const u_char * ep,const uint8_t type,const uint16_t len,const uint32_t xid)2379*b636d99dSDavid van Moolenbroek of10_header_body_print(netdissect_options *ndo,
2380*b636d99dSDavid van Moolenbroek                        const u_char *cp, const u_char *ep, const uint8_t type,
2381*b636d99dSDavid van Moolenbroek                        const uint16_t len, const uint32_t xid)
2382*b636d99dSDavid van Moolenbroek {
2383*b636d99dSDavid van Moolenbroek 	const u_char *cp0 = cp;
2384*b636d99dSDavid van Moolenbroek 	const u_int len0 = len;
2385*b636d99dSDavid van Moolenbroek 	/* Thus far message length is not less than the basic header size, but most
2386*b636d99dSDavid van Moolenbroek 	 * message types have additional assorted constraints on the length. Wherever
2387*b636d99dSDavid van Moolenbroek 	 * possible, check that message length meets the constraint, in remaining
2388*b636d99dSDavid van Moolenbroek 	 * cases check that the length is OK to begin decoding and leave any final
2389*b636d99dSDavid van Moolenbroek 	 * verification up to a lower-layer function. When the current message is
2390*b636d99dSDavid van Moolenbroek 	 * corrupt, proceed to the next message. */
2391*b636d99dSDavid van Moolenbroek 
2392*b636d99dSDavid van Moolenbroek 	/* [OF10] Section 5.1 */
2393*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "\n\tversion 1.0, type %s, length %u, xid 0x%08x",
2394*b636d99dSDavid van Moolenbroek 	       tok2str(ofpt_str, "invalid (0x%02x)", type), len, xid));
2395*b636d99dSDavid van Moolenbroek 	switch (type) {
2396*b636d99dSDavid van Moolenbroek 	/* OpenFlow header only. */
2397*b636d99dSDavid van Moolenbroek 	case OFPT_FEATURES_REQUEST: /* [OF10] Section 5.3.1 */
2398*b636d99dSDavid van Moolenbroek 	case OFPT_GET_CONFIG_REQUEST: /* [OF10] Section 5.3.2 */
2399*b636d99dSDavid van Moolenbroek 	case OFPT_BARRIER_REQUEST: /* [OF10] Section 5.3.7 */
2400*b636d99dSDavid van Moolenbroek 	case OFPT_BARRIER_REPLY: /* ibid */
2401*b636d99dSDavid van Moolenbroek 		if (len != OF_HEADER_LEN)
2402*b636d99dSDavid van Moolenbroek 			goto corrupt;
2403*b636d99dSDavid van Moolenbroek 		break;
2404*b636d99dSDavid van Moolenbroek 
2405*b636d99dSDavid van Moolenbroek 	/* OpenFlow header and fixed-size message body. */
2406*b636d99dSDavid van Moolenbroek 	case OFPT_SET_CONFIG: /* [OF10] Section 5.3.2 */
2407*b636d99dSDavid van Moolenbroek 	case OFPT_GET_CONFIG_REPLY: /* ibid */
2408*b636d99dSDavid van Moolenbroek 		if (len != OF_SWITCH_CONFIG_LEN)
2409*b636d99dSDavid van Moolenbroek 			goto corrupt;
2410*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2411*b636d99dSDavid van Moolenbroek 			goto next_message;
2412*b636d99dSDavid van Moolenbroek 		/* flags */
2413*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2414*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t flags %s", tok2str(ofp_config_str, "invalid (0x%04x)", EXTRACT_16BITS(cp))));
2415*b636d99dSDavid van Moolenbroek 		cp += 2;
2416*b636d99dSDavid van Moolenbroek 		/* miss_send_len */
2417*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2418*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, ", miss_send_len %u", EXTRACT_16BITS(cp)));
2419*b636d99dSDavid van Moolenbroek 		return cp + 2;
2420*b636d99dSDavid van Moolenbroek 	case OFPT_PORT_MOD:
2421*b636d99dSDavid van Moolenbroek 		if (len != OF_PORT_MOD_LEN)
2422*b636d99dSDavid van Moolenbroek 			goto corrupt;
2423*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2424*b636d99dSDavid van Moolenbroek 			goto next_message;
2425*b636d99dSDavid van Moolenbroek 		return of10_port_mod_print(ndo, cp, ep);
2426*b636d99dSDavid van Moolenbroek 	case OFPT_QUEUE_GET_CONFIG_REQUEST: /* [OF10] Section 5.3.4 */
2427*b636d99dSDavid van Moolenbroek 		if (len != OF_QUEUE_GET_CONFIG_REQUEST_LEN)
2428*b636d99dSDavid van Moolenbroek 			goto corrupt;
2429*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2430*b636d99dSDavid van Moolenbroek 			goto next_message;
2431*b636d99dSDavid van Moolenbroek 		/* port */
2432*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2433*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2434*b636d99dSDavid van Moolenbroek 		cp += 2;
2435*b636d99dSDavid van Moolenbroek 		/* pad */
2436*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2437*b636d99dSDavid van Moolenbroek 		return cp + 2;
2438*b636d99dSDavid van Moolenbroek 	case OFPT_FLOW_REMOVED:
2439*b636d99dSDavid van Moolenbroek 		if (len != OF_FLOW_REMOVED_LEN)
2440*b636d99dSDavid van Moolenbroek 			goto corrupt;
2441*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2442*b636d99dSDavid van Moolenbroek 			goto next_message;
2443*b636d99dSDavid van Moolenbroek 		return of10_flow_removed_print(ndo, cp, ep);
2444*b636d99dSDavid van Moolenbroek 	case OFPT_PORT_STATUS: /* [OF10] Section 5.4.3 */
2445*b636d99dSDavid van Moolenbroek 		if (len != OF_PORT_STATUS_LEN)
2446*b636d99dSDavid van Moolenbroek 			goto corrupt;
2447*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2448*b636d99dSDavid van Moolenbroek 			goto next_message;
2449*b636d99dSDavid van Moolenbroek 		/* reason */
2450*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 1);
2451*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t reason %s", tok2str(ofppr_str, "invalid (0x%02x)", *cp)));
2452*b636d99dSDavid van Moolenbroek 		cp += 1;
2453*b636d99dSDavid van Moolenbroek 		/* pad */
2454*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 7);
2455*b636d99dSDavid van Moolenbroek 		cp += 7;
2456*b636d99dSDavid van Moolenbroek 		/* desc */
2457*b636d99dSDavid van Moolenbroek 		return of10_phy_ports_print(ndo, cp, ep, OF_PHY_PORT_LEN);
2458*b636d99dSDavid van Moolenbroek 
2459*b636d99dSDavid van Moolenbroek 	/* OpenFlow header, fixed-size message body and n * fixed-size data units. */
2460*b636d99dSDavid van Moolenbroek 	case OFPT_FEATURES_REPLY:
2461*b636d99dSDavid van Moolenbroek 		if (len < OF_SWITCH_FEATURES_LEN)
2462*b636d99dSDavid van Moolenbroek 			goto corrupt;
2463*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2464*b636d99dSDavid van Moolenbroek 			goto next_message;
2465*b636d99dSDavid van Moolenbroek 		return of10_features_reply_print(ndo, cp, ep, len);
2466*b636d99dSDavid van Moolenbroek 
2467*b636d99dSDavid van Moolenbroek 	/* OpenFlow header and variable-size data. */
2468*b636d99dSDavid van Moolenbroek 	case OFPT_HELLO: /* [OF10] Section 5.5.1 */
2469*b636d99dSDavid van Moolenbroek 	case OFPT_ECHO_REQUEST: /* [OF10] Section 5.5.2 */
2470*b636d99dSDavid van Moolenbroek 	case OFPT_ECHO_REPLY: /* [OF10] Section 5.5.3 */
2471*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2472*b636d99dSDavid van Moolenbroek 			goto next_message;
2473*b636d99dSDavid van Moolenbroek 		return of10_data_print(ndo, cp, ep, len - OF_HEADER_LEN);
2474*b636d99dSDavid van Moolenbroek 
2475*b636d99dSDavid van Moolenbroek 	/* OpenFlow header, fixed-size message body and variable-size data. */
2476*b636d99dSDavid van Moolenbroek 	case OFPT_ERROR:
2477*b636d99dSDavid van Moolenbroek 		if (len < OF_ERROR_MSG_LEN)
2478*b636d99dSDavid van Moolenbroek 			goto corrupt;
2479*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2480*b636d99dSDavid van Moolenbroek 			goto next_message;
2481*b636d99dSDavid van Moolenbroek 		return of10_error_print(ndo, cp, ep, len);
2482*b636d99dSDavid van Moolenbroek 	case OFPT_VENDOR:
2483*b636d99dSDavid van Moolenbroek 	  /* [OF10] Section 5.5.4 */
2484*b636d99dSDavid van Moolenbroek 		if (len < OF_VENDOR_HEADER_LEN)
2485*b636d99dSDavid van Moolenbroek 			goto corrupt;
2486*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2487*b636d99dSDavid van Moolenbroek 			goto next_message;
2488*b636d99dSDavid van Moolenbroek 		return of10_vendor_message_print(ndo, cp, ep, len - OF_HEADER_LEN);
2489*b636d99dSDavid van Moolenbroek 	case OFPT_PACKET_IN:
2490*b636d99dSDavid van Moolenbroek 		/* 2 mock octets count in OF_PACKET_IN_LEN but not in len */
2491*b636d99dSDavid van Moolenbroek 		if (len < OF_PACKET_IN_LEN - 2)
2492*b636d99dSDavid van Moolenbroek 			goto corrupt;
2493*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2494*b636d99dSDavid van Moolenbroek 			goto next_message;
2495*b636d99dSDavid van Moolenbroek 		return of10_packet_in_print(ndo, cp, ep, len);
2496*b636d99dSDavid van Moolenbroek 
2497*b636d99dSDavid van Moolenbroek 	/* a. OpenFlow header. */
2498*b636d99dSDavid van Moolenbroek 	/* b. OpenFlow header and one of the fixed-size message bodies. */
2499*b636d99dSDavid van Moolenbroek 	/* c. OpenFlow header, fixed-size message body and variable-size data. */
2500*b636d99dSDavid van Moolenbroek 	case OFPT_STATS_REQUEST:
2501*b636d99dSDavid van Moolenbroek 		if (len < OF_STATS_REQUEST_LEN)
2502*b636d99dSDavid van Moolenbroek 			goto corrupt;
2503*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2504*b636d99dSDavid van Moolenbroek 			goto next_message;
2505*b636d99dSDavid van Moolenbroek 		return of10_stats_request_print(ndo, cp, ep, len);
2506*b636d99dSDavid van Moolenbroek 
2507*b636d99dSDavid van Moolenbroek 	/* a. OpenFlow header and fixed-size message body. */
2508*b636d99dSDavid van Moolenbroek 	/* b. OpenFlow header and n * fixed-size data units. */
2509*b636d99dSDavid van Moolenbroek 	/* c. OpenFlow header and n * variable-size data units. */
2510*b636d99dSDavid van Moolenbroek 	/* d. OpenFlow header, fixed-size message body and variable-size data. */
2511*b636d99dSDavid van Moolenbroek 	case OFPT_STATS_REPLY:
2512*b636d99dSDavid van Moolenbroek 		if (len < OF_STATS_REPLY_LEN)
2513*b636d99dSDavid van Moolenbroek 			goto corrupt;
2514*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2515*b636d99dSDavid van Moolenbroek 			goto next_message;
2516*b636d99dSDavid van Moolenbroek 		return of10_stats_reply_print(ndo, cp, ep, len);
2517*b636d99dSDavid van Moolenbroek 
2518*b636d99dSDavid van Moolenbroek 	/* OpenFlow header and n * variable-size data units and variable-size data. */
2519*b636d99dSDavid van Moolenbroek 	case OFPT_PACKET_OUT:
2520*b636d99dSDavid van Moolenbroek 		if (len < OF_PACKET_OUT_LEN)
2521*b636d99dSDavid van Moolenbroek 			goto corrupt;
2522*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2523*b636d99dSDavid van Moolenbroek 			goto next_message;
2524*b636d99dSDavid van Moolenbroek 		return of10_packet_out_print(ndo, cp, ep, len);
2525*b636d99dSDavid van Moolenbroek 
2526*b636d99dSDavid van Moolenbroek 	/* OpenFlow header, fixed-size message body and n * variable-size data units. */
2527*b636d99dSDavid van Moolenbroek 	case OFPT_FLOW_MOD:
2528*b636d99dSDavid van Moolenbroek 		if (len < OF_FLOW_MOD_LEN)
2529*b636d99dSDavid van Moolenbroek 			goto corrupt;
2530*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2531*b636d99dSDavid van Moolenbroek 			goto next_message;
2532*b636d99dSDavid van Moolenbroek 		return of10_flow_mod_print(ndo, cp, ep, len);
2533*b636d99dSDavid van Moolenbroek 
2534*b636d99dSDavid van Moolenbroek 	/* OpenFlow header, fixed-size message body and n * variable-size data units. */
2535*b636d99dSDavid van Moolenbroek 	case OFPT_QUEUE_GET_CONFIG_REPLY: /* [OF10] Section 5.3.4 */
2536*b636d99dSDavid van Moolenbroek 		if (len < OF_QUEUE_GET_CONFIG_REPLY_LEN)
2537*b636d99dSDavid van Moolenbroek 			goto corrupt;
2538*b636d99dSDavid van Moolenbroek 		if (ndo->ndo_vflag < 1)
2539*b636d99dSDavid van Moolenbroek 			goto next_message;
2540*b636d99dSDavid van Moolenbroek 		/* port */
2541*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 2);
2542*b636d99dSDavid van Moolenbroek 		ND_PRINT((ndo, "\n\t port_no %s", tok2str(ofpp_str, "%u", EXTRACT_16BITS(cp))));
2543*b636d99dSDavid van Moolenbroek 		cp += 2;
2544*b636d99dSDavid van Moolenbroek 		/* pad */
2545*b636d99dSDavid van Moolenbroek 		ND_TCHECK2(*cp, 6);
2546*b636d99dSDavid van Moolenbroek 		cp += 6;
2547*b636d99dSDavid van Moolenbroek 		/* queues */
2548*b636d99dSDavid van Moolenbroek 		return of10_queues_print(ndo, cp, ep, len - OF_QUEUE_GET_CONFIG_REPLY_LEN);
2549*b636d99dSDavid van Moolenbroek 	} /* switch (type) */
2550*b636d99dSDavid van Moolenbroek 	goto next_message;
2551*b636d99dSDavid van Moolenbroek 
2552*b636d99dSDavid van Moolenbroek corrupt: /* skip the message body */
2553*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", cstr));
2554*b636d99dSDavid van Moolenbroek next_message:
2555*b636d99dSDavid van Moolenbroek 	ND_TCHECK2(*cp0, len0 - OF_HEADER_LEN);
2556*b636d99dSDavid van Moolenbroek 	return cp0 + len0 - OF_HEADER_LEN;
2557*b636d99dSDavid van Moolenbroek trunc:
2558*b636d99dSDavid van Moolenbroek 	ND_PRINT((ndo, "%s", tstr));
2559*b636d99dSDavid van Moolenbroek 	return ep;
2560*b636d99dSDavid van Moolenbroek }
2561