Coverage Report

Created: 2026-02-26 06:41

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/openvswitch/lib/ofp-print.c
Line
Count
Source
1
/*
2
 * Copyright (c) 2008-2017 Nicira, Inc.
3
 *
4
 * Licensed under the Apache License, Version 2.0 (the "License");
5
 * you may not use this file except in compliance with the License.
6
 * You may obtain a copy of the License at:
7
 *
8
 *     http://www.apache.org/licenses/LICENSE-2.0
9
 *
10
 * Unless required by applicable law or agreed to in writing, software
11
 * distributed under the License is distributed on an "AS IS" BASIS,
12
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
 * See the License for the specific language governing permissions and
14
 * limitations under the License.
15
 */
16
17
#include <config.h>
18
19
#include "openvswitch/ofp-print.h"
20
21
#include <errno.h>
22
#include <inttypes.h>
23
#include <sys/types.h>
24
#include <netinet/in.h>
25
#include <sys/wait.h>
26
#include <stdarg.h>
27
#include <stdlib.h>
28
#include <ctype.h>
29
30
#include "bundle.h"
31
#include "byte-order.h"
32
#include "colors.h"
33
#include "compiler.h"
34
#include "dp-packet.h"
35
#include "flow.h"
36
#include "learn.h"
37
#include "multipath.h"
38
#include "netdev.h"
39
#include "nx-match.h"
40
#include "odp-util.h"
41
#include "openflow/nicira-ext.h"
42
#include "openflow/openflow.h"
43
#include "openvswitch/dynamic-string.h"
44
#include "openvswitch/meta-flow.h"
45
#include "openvswitch/ofp-actions.h"
46
#include "openvswitch/ofp-bundle.h"
47
#include "openvswitch/ofp-connection.h"
48
#include "openvswitch/ofp-ct.h"
49
#include "openvswitch/ofp-errors.h"
50
#include "openvswitch/ofp-group.h"
51
#include "openvswitch/ofp-ipfix.h"
52
#include "openvswitch/ofp-match.h"
53
#include "openvswitch/ofp-meter.h"
54
#include "openvswitch/ofp-monitor.h"
55
#include "openvswitch/ofp-msgs.h"
56
#include "openvswitch/ofp-port.h"
57
#include "openvswitch/ofp-queue.h"
58
#include "openvswitch/ofp-switch.h"
59
#include "openvswitch/ofp-table.h"
60
#include "openvswitch/ofp-util.h"
61
#include "openvswitch/ofpbuf.h"
62
#include "openvswitch/type-props.h"
63
#include "packets.h"
64
#include "unaligned.h"
65
#include "util.h"
66
#include "uuid.h"
67
68
static void ofp_print_error(struct ds *, enum ofperr);
69
70
/* Returns a string that represents the contents of the Ethernet frame in the
71
 * 'len' bytes starting at 'data'.  The caller must free the returned string.*/
72
char *
73
ofp_packet_to_string(const void *data, size_t len, ovs_be32 packet_type)
74
57.9k
{
75
57.9k
    struct ds ds = DS_EMPTY_INITIALIZER;
76
57.9k
    struct dp_packet buf;
77
57.9k
    struct flow flow;
78
57.9k
    size_t l4_size;
79
80
57.9k
    dp_packet_use_const(&buf, data, len);
81
57.9k
    buf.packet_type = packet_type;
82
57.9k
    flow_extract(&buf, &flow);
83
57.9k
    flow_format(&ds, &flow, NULL);
84
85
57.9k
    l4_size = dp_packet_l4_size(&buf);
86
87
57.9k
    if (flow.nw_proto == IPPROTO_TCP && l4_size >= TCP_HEADER_LEN) {
88
9.98k
        struct tcp_header *th = dp_packet_l4(&buf);
89
9.98k
        ds_put_format(&ds, " tcp_csum:%"PRIx16, ntohs(th->tcp_csum));
90
47.9k
    } else if (flow.nw_proto == IPPROTO_UDP && l4_size >= UDP_HEADER_LEN) {
91
928
        struct udp_header *uh = dp_packet_l4(&buf);
92
928
        ds_put_format(&ds, " udp_csum:%"PRIx16, ntohs(uh->udp_csum));
93
47.0k
    } else if (flow.nw_proto == IPPROTO_SCTP && l4_size >= SCTP_HEADER_LEN) {
94
615
        struct sctp_header *sh = dp_packet_l4(&buf);
95
615
        ds_put_format(&ds, " sctp_csum:%"PRIx32,
96
615
                      ntohl(get_16aligned_be32(&sh->sctp_csum)));
97
46.4k
    } else if (flow.nw_proto == IPPROTO_ICMP && l4_size >= ICMP_HEADER_LEN) {
98
2.82k
        struct icmp_header *icmph = dp_packet_l4(&buf);
99
2.82k
        ds_put_format(&ds, " icmp_csum:%"PRIx16,
100
2.82k
                      ntohs(icmph->icmp_csum));
101
43.6k
    } else if (flow.nw_proto == IPPROTO_ICMPV6 && l4_size >= ICMP6_HEADER_LEN) {
102
1.15k
        struct icmp6_header *icmp6h = dp_packet_l4(&buf);
103
1.15k
        ds_put_format(&ds, " icmp6_csum:%"PRIx16,
104
1.15k
                      ntohs(icmp6h->icmp6_cksum));
105
1.15k
    }
106
107
57.9k
    ds_put_char(&ds, '\n');
108
109
57.9k
    return ds_cstr(&ds);
110
57.9k
}
111
112
char *
113
ofp_dp_packet_to_string(const struct dp_packet *packet)
114
0
{
115
0
    return ofp_packet_to_string(dp_packet_data(packet),
116
0
                                dp_packet_size(packet),
117
0
                                packet->packet_type);
118
0
}
119
120
static enum ofperr
121
ofp_print_packet_in(struct ds *string, const struct ofp_header *oh,
122
                    const struct ofputil_port_map *port_map,
123
                    const struct ofputil_table_map *table_map, int verbosity)
124
91.9k
{
125
91.9k
    struct ofputil_packet_in_private pin;
126
91.9k
    uint32_t buffer_id;
127
91.9k
    size_t total_len;
128
91.9k
    enum ofperr error = ofputil_decode_packet_in_private(oh, true, NULL, NULL,
129
91.9k
                                                         &pin, &total_len,
130
91.9k
                                                         &buffer_id);
131
91.9k
    if (!error) {
132
55.3k
        ofputil_packet_in_private_format(string, &pin, total_len, buffer_id,
133
55.3k
                                         port_map, table_map, verbosity);
134
55.3k
        ofputil_packet_in_private_destroy(&pin);
135
55.3k
    }
136
91.9k
    return error;
137
91.9k
}
138
139
static enum ofperr
140
ofp_print_packet_out(struct ds *string, const struct ofp_header *oh,
141
                     const struct ofputil_port_map *port_map,
142
                     const struct ofputil_table_map *table_map, int verbosity)
143
125k
{
144
125k
    struct ofputil_packet_out po;
145
125k
    struct ofpbuf ofpacts;
146
125k
    enum ofperr error;
147
148
125k
    ofpbuf_init(&ofpacts, 64);
149
125k
    error = ofputil_decode_packet_out(&po, oh, NULL, &ofpacts);
150
125k
    if (!error) {
151
17.9k
        ofputil_packet_out_format(string, &po, port_map, table_map, verbosity);
152
17.9k
    }
153
125k
    ofpbuf_uninit(&ofpacts);
154
125k
    return error;
155
125k
}
156
157
void
158
ofp_print_bit_names(struct ds *string, uint32_t bits,
159
                    const char *(*bit_to_name)(uint32_t bit),
160
                    char separator)
161
99.0k
{
162
99.0k
    int n = 0;
163
99.0k
    int i;
164
165
99.0k
    if (!bits) {
166
13.5k
        ds_put_cstr(string, "0");
167
13.5k
        return;
168
13.5k
    }
169
170
2.82M
    for (i = 0; i < 32; i++) {
171
2.73M
        uint32_t bit = UINT32_C(1) << i;
172
173
2.73M
        if (bits & bit) {
174
479k
            const char *name = bit_to_name(bit);
175
479k
            if (name) {
176
461k
                if (n++) {
177
376k
                    ds_put_char(string, separator);
178
376k
                }
179
461k
                ds_put_cstr(string, name);
180
461k
                bits &= ~bit;
181
461k
            }
182
479k
        }
183
2.73M
    }
184
185
85.4k
    if (bits) {
186
1.81k
        if (n) {
187
871
            ds_put_char(string, separator);
188
871
        }
189
1.81k
        ds_put_format(string, "0x%"PRIx32, bits);
190
1.81k
    }
191
85.4k
}
192
193
static enum ofperr
194
ofp_print_switch_features(struct ds *string, const struct ofp_header *oh)
195
3.63k
{
196
3.63k
    struct ofputil_switch_features features;
197
3.63k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
198
3.63k
    enum ofperr error = ofputil_pull_switch_features(&b, &features);
199
3.63k
    if (!error) {
200
3.63k
        ofputil_switch_features_format(string, &features);
201
3.63k
        error = ofputil_phy_ports_format(string, oh->version, &b);
202
3.63k
    }
203
3.63k
    return error;
204
3.63k
}
205
206
static enum ofperr
207
ofp_print_set_config(struct ds *string, const struct ofp_header *oh)
208
2.86k
{
209
2.86k
    struct ofputil_switch_config config;
210
2.86k
    enum ofperr error;
211
212
2.86k
    error = ofputil_decode_set_config(oh, &config);
213
2.86k
    if (error) {
214
2.35k
        return error;
215
2.35k
    }
216
516
    ofputil_switch_config_format(string, &config);
217
516
    return 0;
218
2.86k
}
219
220
static enum ofperr
221
ofp_print_get_config_reply(struct ds *string, const struct ofp_header *oh)
222
2.60k
{
223
2.60k
    struct ofputil_switch_config config;
224
2.60k
    ofputil_decode_get_config_reply(oh, &config);
225
2.60k
    ofputil_switch_config_format(string, &config);
226
2.60k
    return 0;
227
2.60k
}
228
229
static enum ofperr
230
ofp_print_table_features_reply(struct ds *s, const struct ofp_header *oh)
231
6.70k
{
232
6.70k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
233
234
6.70k
    struct ofputil_table_features prev;
235
6.70k
    int first_ditto = -1, last_ditto = -1;
236
8.07k
    for (int i = 0; ; i++) {
237
8.07k
        struct ofputil_table_features tf;
238
8.07k
        struct ofpbuf raw_properties;
239
8.07k
        int retval = ofputil_decode_table_features(&b, &tf, &raw_properties);
240
8.07k
        if (retval) {
241
6.70k
            ofputil_table_features_format_finish(s, first_ditto, last_ditto);
242
6.70k
            return retval != EOF ? retval : 0;
243
6.70k
        }
244
245
1.36k
        ofputil_table_features_format(s, &tf, i ? &prev : NULL, NULL, NULL,
246
1.36k
                                      &first_ditto, &last_ditto);
247
1.36k
        prev = tf;
248
1.36k
    }
249
6.70k
}
250
251
void
252
ofp_print_duration(struct ds *string, unsigned int sec, unsigned int nsec)
253
19.5k
{
254
19.5k
    ds_put_format(string, "%u", sec);
255
256
    /* If there are no fractional seconds, don't print any decimals.
257
     *
258
     * If the fractional seconds can be expressed exactly as milliseconds,
259
     * print 3 decimals.  Open vSwitch provides millisecond precision for most
260
     * time measurements, so printing 3 decimals every time makes it easier to
261
     * spot real changes in flow dumps that refresh themselves quickly.
262
     *
263
     * If the fractional seconds are more precise than milliseconds, print the
264
     * number of decimals needed to express them exactly.
265
     */
266
19.5k
    if (nsec > 0) {
267
16.4k
        unsigned int msec = nsec / 1000000;
268
16.4k
        if (msec * 1000000 == nsec) {
269
465
            ds_put_format(string, ".%03u", msec);
270
15.9k
        } else {
271
15.9k
            ds_put_format(string, ".%09u", nsec);
272
18.1k
            while (string->string[string->length - 1] == '0') {
273
2.20k
                string->length--;
274
2.20k
            }
275
15.9k
        }
276
16.4k
    }
277
19.5k
    ds_put_char(string, 's');
278
19.5k
}
279
280
static enum ofperr
281
ofp_print_flow_removed(struct ds *string, const struct ofp_header *oh,
282
                       const struct ofputil_port_map *port_map,
283
                       const struct ofputil_table_map *table_map)
284
9.10k
{
285
9.10k
    struct ofputil_flow_removed fr;
286
9.10k
    enum ofperr error = ofputil_decode_flow_removed(&fr, oh);
287
9.10k
    if (!error) {
288
2.00k
        ofputil_flow_removed_format(string, &fr, port_map, table_map);
289
2.00k
    }
290
9.10k
    return error;
291
9.10k
}
292
293
static enum ofperr
294
ofp_print_port_mod(struct ds *string, const struct ofp_header *oh,
295
                   const struct ofputil_port_map *port_map)
296
3.31k
{
297
3.31k
    struct ofputil_port_mod pm;
298
3.31k
    enum ofperr error = ofputil_decode_port_mod(oh, &pm, true);
299
3.31k
    if (!error) {
300
1.09k
        ofputil_port_mod_format(string, &pm, port_map);
301
1.09k
    }
302
3.31k
    return error;
303
3.31k
}
304
305
static enum ofperr
306
ofp_print_table_mod(struct ds *string, const struct ofp_header *oh,
307
                  const struct ofputil_table_map *table_map)
308
6.46k
{
309
6.46k
    struct ofputil_table_mod tm;
310
6.46k
    enum ofperr error = ofputil_decode_table_mod(oh, &tm);
311
6.46k
    if (!error) {
312
3.02k
        ofputil_table_mod_format(string, &tm, table_map);
313
3.02k
    }
314
6.46k
    return error;
315
6.46k
}
316
317
static enum ofperr
318
ofp_print_table_status_message(struct ds *string, const struct ofp_header *oh,
319
                               const struct ofputil_table_map *table_map)
320
866
{
321
866
    struct ofputil_table_status ts;
322
866
    enum ofperr error = ofputil_decode_table_status(oh, &ts);
323
866
    if (!error) {
324
155
        ofputil_format_table_status(string, &ts, table_map);
325
155
    }
326
866
    return error;
327
866
}
328
329
static enum ofperr
330
ofp_print_meter_mod(struct ds *s, const struct ofp_header *oh)
331
5.12k
{
332
5.12k
    struct ofputil_meter_mod mm;
333
5.12k
    struct ofpbuf bands;
334
335
5.12k
    ofpbuf_init(&bands, 64);
336
5.12k
    enum ofperr error = ofputil_decode_meter_mod(oh, &mm, &bands);
337
5.12k
    if (!error) {
338
1.75k
        ofputil_format_meter_mod(s, &mm);
339
1.75k
    }
340
5.12k
    ofpbuf_uninit(&bands);
341
342
5.12k
    return error;
343
5.12k
}
344
345
static enum ofperr
346
ofp_print_meter_stats_request(struct ds *s, const struct ofp_header *oh)
347
1.28k
{
348
1.28k
    uint32_t meter_id;
349
350
1.28k
    ofputil_decode_meter_request(oh, &meter_id);
351
1.28k
    ds_put_char(s, ' ');
352
353
1.28k
    ofputil_format_meter_id(s, meter_id, '=');
354
355
1.28k
    return 0;
356
1.28k
}
357
358
static enum ofperr
359
ofp_print_meter_features_reply(struct ds *s, const struct ofp_header *oh)
360
197
{
361
197
    struct ofputil_meter_features mf;
362
197
    ofputil_decode_meter_features(oh, &mf);
363
197
    ofputil_format_meter_features(s, &mf);
364
197
    return 0;
365
197
}
366
367
static enum ofperr
368
ofp_print_meter_config_reply(struct ds *s, const struct ofp_header *oh)
369
1.09k
{
370
1.09k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
371
1.09k
    struct ofpbuf bands;
372
1.09k
    int retval;
373
374
1.09k
    ofpbuf_init(&bands, 64);
375
2.22k
    for (;;) {
376
2.22k
        struct ofputil_meter_config mc;
377
378
2.22k
        retval = ofputil_decode_meter_config(&b, &mc, &bands);
379
2.22k
        if (retval) {
380
1.09k
            break;
381
1.09k
        }
382
1.12k
        ds_put_char(s, '\n');
383
1.12k
        ofputil_format_meter_config(s, &mc);
384
1.12k
    }
385
1.09k
    ofpbuf_uninit(&bands);
386
387
1.09k
    return retval != EOF ? retval : 0;
388
1.09k
}
389
390
static enum ofperr
391
ofp_print_meter_stats_reply(struct ds *s, const struct ofp_header *oh)
392
2.84k
{
393
2.84k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
394
2.84k
    struct ofpbuf bands;
395
2.84k
    int retval;
396
397
2.84k
    ofpbuf_init(&bands, 64);
398
4.31k
    for (;;) {
399
4.31k
        struct ofputil_meter_stats ms;
400
401
4.31k
        retval = ofputil_decode_meter_stats(&b, &ms, &bands);
402
4.31k
        if (retval) {
403
2.84k
            break;
404
2.84k
        }
405
1.46k
        ds_put_char(s, '\n');
406
1.46k
        ofputil_format_meter_stats(s, &ms);
407
1.46k
    }
408
2.84k
    ofpbuf_uninit(&bands);
409
410
2.84k
    return retval != EOF ? retval : 0;
411
2.84k
}
412
413
static void
414
ofp_print_error(struct ds *string, enum ofperr error)
415
1.04M
{
416
1.04M
    ds_put_format(string, "***decode error: %s***\n", ofperr_get_name(error));
417
1.04M
}
418
419
static enum ofperr
420
ofp_print_hello(struct ds *string, const struct ofp_header *oh)
421
37.8k
{
422
37.8k
    ofputil_hello_format(string, oh);
423
37.8k
    return 0;
424
37.8k
}
425
426
static enum ofperr
427
ofp_print_error_msg(struct ds *string, const struct ofp_header *oh,
428
                    const struct ofputil_port_map *port_map,
429
                    const struct ofputil_table_map *table_map)
430
520k
{
431
520k
    struct ofpbuf payload;
432
520k
    enum ofperr error = ofperr_decode_msg(oh, &payload);
433
520k
    if (!error) {
434
184k
        return OFPERR_OFPBRC_BAD_LEN;
435
184k
    }
436
335k
    ofperr_msg_format(string, error, &payload, port_map, table_map);
437
335k
    ofpbuf_uninit(&payload);
438
439
335k
    return 0;
440
520k
}
441
442
static enum ofperr
443
ofp_print_port_status(struct ds *string, const struct ofp_header *oh)
444
1.87k
{
445
1.87k
    struct ofputil_port_status ps;
446
1.87k
    enum ofperr error = ofputil_decode_port_status(oh, &ps);
447
1.87k
    if (!error) {
448
1.20k
        ofputil_port_status_format(string, &ps);
449
1.20k
    }
450
1.87k
    return error;
451
1.87k
}
452
453
static enum ofperr
454
ofp_print_ofpst_desc_reply(struct ds *string, const struct ofp_header *oh)
455
24
{
456
24
    const struct ofp_desc_stats *ods = ofpmsg_body(oh);
457
458
24
    ds_put_char(string, '\n');
459
24
    ds_put_format(string, "Manufacturer: %.*s\n",
460
24
            (int) sizeof ods->mfr_desc, ods->mfr_desc);
461
24
    ds_put_format(string, "Hardware: %.*s\n",
462
24
            (int) sizeof ods->hw_desc, ods->hw_desc);
463
24
    ds_put_format(string, "Software: %.*s\n",
464
24
            (int) sizeof ods->sw_desc, ods->sw_desc);
465
24
    ds_put_format(string, "Serial Num: %.*s\n",
466
24
            (int) sizeof ods->serial_num, ods->serial_num);
467
24
    ds_put_format(string, "DP Description: %.*s\n",
468
24
            (int) sizeof ods->dp_desc, ods->dp_desc);
469
470
24
    return 0;
471
24
}
472
473
static enum ofperr
474
ofp_print_flow_stats_request(struct ds *string, const struct ofp_header *oh,
475
                             const struct ofputil_port_map *port_map,
476
                             const struct ofputil_table_map *table_map)
477
5.72k
{
478
5.72k
    struct ofputil_flow_stats_request fsr;
479
5.72k
    enum ofperr error = ofputil_decode_flow_stats_request(&fsr, oh, NULL,
480
5.72k
                                                          NULL);
481
5.72k
    if (!error) {
482
2.36k
        ofputil_flow_stats_request_format(string, &fsr, port_map, table_map);
483
2.36k
    }
484
5.72k
    return error;
485
5.72k
}
486
487
static enum ofperr
488
ofp_print_flow_stats_reply(struct ds *string, const struct ofp_header *oh,
489
                           const struct ofputil_port_map *port_map,
490
                           const struct ofputil_table_map *table_map)
491
19.9k
{
492
19.9k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
493
19.9k
    struct ofpbuf ofpacts;
494
19.9k
    int retval;
495
496
19.9k
    ofpbuf_init(&ofpacts, 64);
497
26.7k
    for (;;) {
498
26.7k
        struct ofputil_flow_stats fs;
499
500
26.7k
        retval = ofputil_decode_flow_stats_reply(&fs, &b, true, &ofpacts);
501
26.7k
        if (retval) {
502
19.9k
            break;
503
19.9k
        }
504
6.89k
        ds_put_cstr(string, "\n ");
505
6.89k
        ofputil_flow_stats_format(string, &fs, port_map, table_map, true);
506
6.89k
     }
507
19.9k
    ofpbuf_uninit(&ofpacts);
508
509
19.9k
    return retval != EOF ? retval : 0;
510
19.9k
}
511
512
static enum ofperr
513
ofp_print_aggregate_stats_reply(struct ds *string, const struct ofp_header *oh)
514
1.47k
{
515
1.47k
    struct ofputil_aggregate_stats as;
516
1.47k
    enum ofperr error;
517
518
1.47k
    error = ofputil_decode_aggregate_stats_reply(&as, oh);
519
1.47k
    if (!error) {
520
154
        ofputil_aggregate_stats_format(string, &as);
521
154
    }
522
1.47k
    return error;
523
1.47k
}
524
525
static enum ofperr
526
ofp_print_ofpst_port_request(struct ds *string, const struct ofp_header *oh,
527
                             const struct ofputil_port_map *port_map)
528
1.11k
{
529
1.11k
    ofp_port_t ofp10_port;
530
1.11k
    enum ofperr error;
531
532
1.11k
    error = ofputil_decode_port_stats_request(oh, &ofp10_port);
533
1.11k
    if (error) {
534
979
        return error;
535
979
    }
536
537
132
    ds_put_cstr(string, " port_no=");
538
132
    ofputil_format_port(ofp10_port, port_map, string);
539
540
132
    return 0;
541
1.11k
}
542
543
static enum ofperr
544
ofp_print_ofpst_port_reply(struct ds *string, const struct ofp_header *oh,
545
                           const struct ofputil_port_map *port_map,
546
                           int verbosity)
547
11.2k
{
548
11.2k
    ds_put_format(string, " %"PRIuSIZE" ports\n",
549
11.2k
                  ofputil_count_port_stats(oh));
550
11.2k
    if (verbosity < 1) {
551
0
        return 0;
552
0
    }
553
554
11.2k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
555
20.3k
    for (;;) {
556
20.3k
        struct ofputil_port_stats ps;
557
20.3k
        int retval;
558
559
20.3k
        retval = ofputil_decode_port_stats(&ps, &b);
560
20.3k
        if (retval) {
561
11.2k
            return retval != EOF ? retval : 0;
562
11.2k
        }
563
9.10k
        ofputil_format_port_stats(string, &ps, port_map);
564
9.10k
        netdev_free_custom_stats_counters(&ps.custom_stats);
565
9.10k
    }
566
11.2k
}
567
568
static enum ofperr
569
ofp_print_table_stats_reply(struct ds *string, const struct ofp_header *oh)
570
3.95k
{
571
3.95k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
572
3.95k
    ofpraw_pull_assert(&b);
573
574
3.95k
    struct ofputil_table_features prev_features;
575
3.95k
    struct ofputil_table_stats prev_stats;
576
3.95k
    int first_ditto = -1, last_ditto = -1;
577
35.8k
    for (int i = 0;; i++) {
578
35.8k
        struct ofputil_table_features features;
579
35.8k
        struct ofputil_table_stats stats;
580
35.8k
        int retval;
581
582
35.8k
        retval = ofputil_decode_table_stats_reply(&b, &stats, &features);
583
35.8k
        if (retval) {
584
3.95k
            ofputil_table_features_format_finish(string,
585
3.95k
                                                 first_ditto, last_ditto);
586
3.95k
            return retval != EOF ? retval : 0;
587
3.95k
        }
588
589
31.8k
        ofputil_table_features_format(string,
590
31.8k
                                      &features, i ? &prev_features : NULL,
591
31.8k
                                      &stats, i ? &prev_stats : NULL,
592
31.8k
                                      &first_ditto, &last_ditto);
593
31.8k
        prev_features = features;
594
31.8k
        prev_stats = stats;
595
31.8k
    }
596
3.95k
}
597
598
static enum ofperr
599
ofp_print_ofpst_port_desc_request(struct ds *string,
600
                                  const struct ofp_header *oh,
601
                                  const struct ofputil_port_map *port_map)
602
345
{
603
345
    enum ofperr error;
604
345
    ofp_port_t port;
605
606
345
    error = ofputil_decode_port_desc_stats_request(oh, &port);
607
345
    if (error) {
608
41
        return error;
609
41
    }
610
611
304
    ds_put_cstr(string, " port=");
612
304
    ofputil_format_port(port, port_map, string);
613
614
304
    return 0;
615
345
}
616
617
static enum ofperr
618
ofp_print_ofpst_port_desc_reply(struct ds *string,
619
                                const struct ofp_header *oh)
620
1.26k
{
621
1.26k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
622
1.26k
    ofpraw_pull_assert(&b);
623
1.26k
    ds_put_char(string, '\n');
624
1.26k
    return ofputil_phy_ports_format(string, oh->version, &b);
625
1.26k
}
626
627
static void
628
ofp_print_stats(struct ds *string, const struct ofp_header *oh)
629
103k
{
630
103k
    uint16_t flags = ofpmp_flags(oh);
631
632
103k
    if (flags) {
633
93.0k
        ds_put_cstr(string, " flags=");
634
93.0k
        if ((!ofpmsg_is_stat_request(oh) || oh->version >= OFP13_VERSION)
635
86.2k
            && (flags & OFPSF_REPLY_MORE)) {
636
30.2k
            ds_put_cstr(string, "[more]");
637
30.2k
            flags &= ~OFPSF_REPLY_MORE;
638
30.2k
        }
639
93.0k
        if (flags) {
640
91.9k
            ds_put_format(string, "[***unknown flags 0x%04"PRIx16"***]",
641
91.9k
                          flags);
642
91.9k
        }
643
93.0k
    }
644
103k
}
645
646
static enum ofperr
647
ofp_print_echo(struct ds *string, const struct ofp_header *oh, int verbosity)
648
10.5k
{
649
10.5k
    size_t len = ntohs(oh->length);
650
651
10.5k
    ds_put_format(string, " %"PRIuSIZE" bytes of payload\n", len - sizeof *oh);
652
10.5k
    if (verbosity > 1) {
653
10.4k
        ds_put_hex_dump(string, oh + 1, len - sizeof *oh, 0, true);
654
10.4k
    }
655
656
10.5k
    return 0;
657
10.5k
}
658
659
static enum ofperr
660
ofp_print_role_message(struct ds *string, const struct ofp_header *oh)
661
11.5k
{
662
11.5k
    struct ofputil_role_request rr;
663
11.5k
    enum ofperr error = ofputil_decode_role_message(oh, &rr);
664
11.5k
    if (!error) {
665
8.69k
        ofputil_format_role_message(string, &rr);
666
8.69k
    }
667
11.5k
    return error;
668
11.5k
}
669
670
static enum ofperr
671
ofp_print_role_status_message(struct ds *string, const struct ofp_header *oh)
672
430
{
673
430
    struct ofputil_role_status rs;
674
430
    enum ofperr error = ofputil_decode_role_status(oh, &rs);
675
430
    if (!error) {
676
216
        ofputil_format_role_status(string, &rs);
677
216
    }
678
430
    return error;
679
430
}
680
681
static enum ofperr
682
ofp_print_nxt_flow_mod_table_id(struct ds *string, const struct ofp_header *oh)
683
69
{
684
69
    bool enable = ofputil_decode_nx_flow_mod_table_id(oh);
685
69
    ds_put_format(string, " %s", enable ? "enable" : "disable");
686
69
    return 0;
687
69
}
688
689
static enum ofperr
690
ofp_print_nxt_set_flow_format(struct ds *string, const struct ofp_header *oh)
691
1.81k
{
692
1.81k
    enum ofputil_protocol p = ofputil_decode_nx_set_flow_format(oh);
693
1.81k
    ds_put_format(string, " format=%s",
694
1.81k
                  p == OFPUTIL_P_OF10_STD ? "openflow10"
695
1.81k
                  : p == OFPUTIL_P_OF10_NXM ? "nxm"
696
1.68k
                  : "(unknown)");
697
1.81k
    return 0;
698
1.81k
}
699
700
static enum ofperr
701
ofp_print_nxt_set_packet_in_format(struct ds *string,
702
                                   const struct ofp_header *oh)
703
274
{
704
274
    enum ofputil_packet_in_format format;
705
274
    enum ofperr error = ofputil_decode_set_packet_in_format(oh, &format);
706
274
    if (!error) {
707
239
        ds_put_format(string, " format=%s",
708
239
                      ofputil_packet_in_format_to_string(format));
709
239
    }
710
274
    return error;
711
274
}
712
713
static enum ofperr
714
ofp_print_set_async_config(struct ds *string, const struct ofp_header *oh,
715
                           enum ofptype ofptype)
716
14.4k
{
717
14.4k
    struct ofputil_async_cfg basis = OFPUTIL_ASYNC_CFG_INIT;
718
14.4k
    struct ofputil_async_cfg ac;
719
720
14.4k
    bool is_reply = ofptype == OFPTYPE_GET_ASYNC_REPLY;
721
14.4k
    enum ofperr error = ofputil_decode_set_async_config(oh, is_reply,
722
14.4k
                                                        &basis, &ac);
723
14.4k
    if (error) {
724
5.58k
        return error;
725
5.58k
    }
726
8.88k
    ofputil_format_set_async_config(string, &ac);
727
8.88k
    return 0;
728
14.4k
}
729
730
static enum ofperr
731
ofp_print_nxt_set_controller_id(struct ds *string,
732
                                const struct nx_controller_id *nci)
733
36
{
734
36
    ds_put_format(string, " id=%"PRIu16, ntohs(nci->controller_id));
735
36
    return 0;
736
36
}
737
738
static enum ofperr
739
ofp_print_nxt_flow_monitor_cancel(struct ds *string,
740
                                  const struct ofp_header *oh)
741
78
{
742
78
    ds_put_format(string, " id=%"PRIu32,
743
78
                  ofputil_decode_flow_monitor_cancel(oh));
744
78
    return 0;
745
78
}
746
747
static enum ofperr
748
ofp_print_flow_monitor_request(struct ds *string,
749
                               const struct ofp_header *oh,
750
                               const struct ofputil_port_map *port_map,
751
                               const struct ofputil_table_map *table_map)
752
6.03k
{
753
6.03k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
754
9.49k
    for (;;) {
755
9.49k
        struct ofputil_flow_monitor_request request;
756
9.49k
        int retval;
757
758
9.49k
        retval = ofputil_decode_flow_monitor_request(&request, &b);
759
9.49k
        if (retval) {
760
6.03k
            return retval != EOF ? retval : 0;
761
6.03k
        }
762
763
3.46k
        ofputil_flow_monitor_request_format(string, &request,
764
3.46k
                                            port_map, table_map);
765
3.46k
    }
766
6.03k
}
767
768
static enum ofperr
769
ofp_print_flow_monitor_reply(struct ds *string,
770
                             const struct ofp_header *oh,
771
                             const struct ofputil_port_map *port_map,
772
                             const struct ofputil_table_map *table_map)
773
8.75k
{
774
8.75k
    uint64_t ofpacts_stub[1024 / 8];
775
8.75k
    struct ofpbuf ofpacts = OFPBUF_STUB_INITIALIZER(ofpacts_stub);
776
8.75k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
777
778
14.5k
    for (;;) {
779
14.5k
        struct ofputil_flow_update update;
780
14.5k
        int retval = ofputil_decode_flow_update(&update, &b, &ofpacts);
781
14.5k
        if (retval) {
782
8.75k
            ofpbuf_uninit(&ofpacts);
783
8.75k
            return retval != EOF ? retval : 0;
784
8.75k
        }
785
5.84k
        ofputil_flow_update_format(string, &update, port_map, table_map);
786
5.84k
    }
787
8.75k
}
788
789
void
790
ofp_print_version(const struct ofp_header *oh,
791
                  struct ds *string)
792
1.09M
{
793
1.09M
    switch (oh->version) {
794
211k
    case OFP10_VERSION:
795
211k
        break;
796
90.4k
    case OFP11_VERSION:
797
90.4k
        ds_put_cstr(string, " (OF1.1)");
798
90.4k
        break;
799
179k
    case OFP12_VERSION:
800
179k
        ds_put_cstr(string, " (OF1.2)");
801
179k
        break;
802
113k
    case OFP13_VERSION:
803
113k
        ds_put_cstr(string, " (OF1.3)");
804
113k
        break;
805
192k
    case OFP14_VERSION:
806
192k
        ds_put_cstr(string, " (OF1.4)");
807
192k
        break;
808
242k
    case OFP15_VERSION:
809
242k
        ds_put_cstr(string, " (OF1.5)");
810
242k
        break;
811
62.8k
    default:
812
62.8k
        ds_put_format(string, " (OF 0x%02"PRIx8")", oh->version);
813
62.8k
        break;
814
1.09M
    }
815
1.09M
    ds_put_format(string, " (xid=0x%"PRIx32"):", ntohl(oh->xid));
816
1.09M
}
817
818
static void
819
ofp_header_to_string__(const struct ofp_header *oh, enum ofpraw raw,
820
                       struct ds *string)
821
1.09M
{
822
1.09M
    ds_put_cstr(string, ofpraw_get_name(raw));
823
1.09M
    ofp_print_version(oh, string);
824
1.09M
}
825
826
static enum ofperr
827
ofp_print_table_desc_reply(struct ds *s, const struct ofp_header *oh,
828
                           const struct ofputil_table_map *table_map)
829
2.58k
{
830
2.58k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
831
3.89k
    for (;;) {
832
3.89k
        struct ofputil_table_desc td;
833
3.89k
        int retval;
834
835
3.89k
        retval = ofputil_decode_table_desc(&b, &td, oh->version);
836
3.89k
        if (retval) {
837
2.58k
            return retval != EOF ? retval : 0;
838
2.58k
        }
839
1.31k
        ofputil_table_desc_format(s, &td, table_map);
840
1.31k
    }
841
2.58k
}
842
843
static enum ofperr
844
ofp_print_bundle_ctrl(struct ds *s, const struct ofp_header *oh)
845
3.03k
{
846
3.03k
    int error;
847
3.03k
    struct ofputil_bundle_ctrl_msg bctrl;
848
849
3.03k
    error = ofputil_decode_bundle_ctrl(oh, &bctrl);
850
3.03k
    if (error) {
851
0
        return error;
852
0
    }
853
3.03k
    ofputil_format_bundle_ctrl_request(s, &bctrl);
854
3.03k
    return 0;
855
3.03k
}
856
857
static enum ofperr
858
ofp_print_bundle_add(struct ds *s, const struct ofp_header *oh,
859
                     const struct ofputil_port_map *port_map,
860
                     const struct ofputil_table_map *table_map,
861
                     int verbosity)
862
4.50k
{
863
4.50k
    struct ofputil_bundle_add_msg badd;
864
4.50k
    int error = ofputil_decode_bundle_add(oh, &badd, NULL);
865
4.50k
    if (!error) {
866
2.30k
        ofputil_format_bundle_add(s, &badd, port_map, table_map, verbosity);
867
2.30k
    }
868
4.50k
    return error;
869
4.50k
}
870
871
static enum ofperr
872
ofp_print_tlv_table_mod(struct ds *s, const struct ofp_header *oh)
873
1.87k
{
874
1.87k
    struct ofputil_tlv_table_mod ttm;
875
1.87k
    int error = ofputil_decode_tlv_table_mod(oh, &ttm);
876
1.87k
    if (!error) {
877
745
        ofputil_format_tlv_table_mod(s, &ttm);
878
745
        ofputil_uninit_tlv_table(&ttm.mappings);
879
745
    }
880
1.87k
    return error;
881
1.87k
}
882
883
static enum ofperr
884
ofp_print_tlv_table_reply(struct ds *s, const struct ofp_header *oh)
885
2.06k
{
886
2.06k
    struct ofputil_tlv_table_reply ttr;
887
2.06k
    int error = ofputil_decode_tlv_table_reply(oh, &ttr);
888
2.06k
    if (!error) {
889
946
        ofputil_format_tlv_table_reply(s, &ttr);
890
946
        ofputil_uninit_tlv_table(&ttr.mappings);
891
946
    }
892
2.06k
    return error;
893
2.06k
}
894
895
/* This function will print the request forward message. The reason for
896
 * request forward is taken from rf.request.type */
897
static enum ofperr
898
ofp_print_requestforward(struct ds *string, const struct ofp_header *oh,
899
                         const struct ofputil_port_map *port_map,
900
                         const struct ofputil_table_map *table_map)
901
1.37k
{
902
1.37k
    struct ofputil_requestforward rf;
903
1.37k
    enum ofperr error = ofputil_decode_requestforward(oh, &rf);
904
1.37k
    if (!error) {
905
444
        ofputil_format_requestforward(string, oh->version,
906
444
                                      &rf, port_map, table_map);
907
444
        ofputil_destroy_requestforward(&rf);
908
444
    }
909
1.37k
    return error;
910
1.37k
}
911
912
static enum ofperr
913
ofp_print_nxst_ipfix_bridge_reply(struct ds *string, const struct ofp_header *oh)
914
545
{
915
545
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
916
1.09k
    for (;;) {
917
1.09k
        struct ofputil_ipfix_stats is;
918
1.09k
        int retval;
919
920
1.09k
        retval = ofputil_pull_ipfix_stats(&is, &b);
921
1.09k
        if (retval) {
922
545
            return retval != EOF ? retval : 0;
923
545
        }
924
545
        ofputil_format_ipfix_stats_bridge(string, &is);
925
545
    }
926
545
}
927
928
static enum ofperr
929
ofp_print_nxst_ipfix_flow_reply(struct ds *string, const struct ofp_header *oh)
930
1.57k
{
931
1.57k
    ds_put_format(string, " %"PRIuSIZE" ids\n", ofputil_count_ipfix_stats(oh));
932
933
1.57k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
934
8.58k
    for (;;) {
935
8.58k
        struct ofputil_ipfix_stats is;
936
8.58k
        int retval;
937
938
8.58k
        retval = ofputil_pull_ipfix_stats(&is, &b);
939
8.58k
        if (retval) {
940
1.57k
            return retval != EOF ? retval : 0;
941
1.57k
        }
942
7.01k
        ofputil_format_ipfix_stats_flow(string, &is);
943
7.01k
    }
944
1.57k
}
945
946
static enum ofperr
947
ofp_print_nxt_ct_flush_zone(struct ds *string, const struct nx_zone_id *nzi)
948
35
{
949
35
    ds_put_format(string, " zone_id=%"PRIu16, ntohs(nzi->zone_id));
950
35
    return 0;
951
35
}
952
953
static enum ofperr
954
ofp_print_nxt_ct_flush(struct ds *string, const struct ofp_header *oh)
955
13.7k
{
956
13.7k
    uint16_t zone_id = 0;
957
13.7k
    struct ofp_ct_match match = {0};
958
959
13.7k
    enum ofperr error = ofp_ct_match_decode(&match, NULL, &zone_id, oh);
960
13.7k
    if (error) {
961
9.37k
        return error;
962
9.37k
    }
963
964
4.40k
    ds_put_format(string, " zone=%"PRIu16" ", zone_id);
965
4.40k
    ofp_ct_match_format(string, &match);
966
967
4.40k
    return 0;
968
13.7k
}
969
970
static enum ofperr
971
ofp_to_string__(const struct ofp_header *oh,
972
                const struct ofputil_port_map *port_map,
973
                const struct ofputil_table_map *table_map, enum ofpraw raw,
974
                struct ds *string, int verbosity)
975
1.08M
{
976
1.08M
    if (ofpmsg_is_stat(oh)) {
977
103k
        ofp_print_stats(string, oh);
978
103k
    }
979
980
1.08M
    const void *msg = oh;
981
1.08M
    enum ofptype type = ofptype_from_ofpraw(raw);
982
1.08M
    switch (type) {
983
186
    case OFPTYPE_GROUP_STATS_REQUEST:
984
186
        return ofputil_group_stats_request_format(string, oh);
985
986
1.90k
    case OFPTYPE_GROUP_STATS_REPLY:
987
1.90k
        return ofputil_group_stats_format(string, oh);
988
989
306
    case OFPTYPE_GROUP_DESC_STATS_REQUEST:
990
306
        return ofputil_group_desc_request_format(string, oh);
991
992
3.72k
    case OFPTYPE_GROUP_DESC_STATS_REPLY:
993
3.72k
        return ofputil_group_desc_format(string, oh, port_map, table_map);
994
995
32
    case OFPTYPE_GROUP_FEATURES_STATS_REQUEST:
996
32
        break;
997
998
798
    case OFPTYPE_GROUP_FEATURES_STATS_REPLY:
999
798
        return ofputil_group_features_format(string, oh);
1000
1001
36.0k
    case OFPTYPE_GROUP_MOD:
1002
36.0k
        return ofputil_group_mod_format(string, oh, port_map, table_map);
1003
1004
1.91k
    case OFPTYPE_TABLE_FEATURES_STATS_REQUEST:
1005
6.70k
    case OFPTYPE_TABLE_FEATURES_STATS_REPLY:
1006
6.70k
        return ofp_print_table_features_reply(string, oh);
1007
1008
130
    case OFPTYPE_TABLE_DESC_REQUEST:
1009
2.58k
    case OFPTYPE_TABLE_DESC_REPLY:
1010
2.58k
        return ofp_print_table_desc_reply(string, oh, table_map);
1011
1012
37.8k
    case OFPTYPE_HELLO:
1013
37.8k
        return ofp_print_hello(string, oh);
1014
1015
520k
    case OFPTYPE_ERROR:
1016
520k
        return ofp_print_error_msg(string, oh, port_map, table_map);
1017
1018
5.96k
    case OFPTYPE_ECHO_REQUEST:
1019
10.5k
    case OFPTYPE_ECHO_REPLY:
1020
10.5k
        return ofp_print_echo(string, oh, verbosity);
1021
1022
25
    case OFPTYPE_FEATURES_REQUEST:
1023
25
        break;
1024
1025
3.63k
    case OFPTYPE_FEATURES_REPLY:
1026
3.63k
        return ofp_print_switch_features(string, oh);
1027
1028
824
    case OFPTYPE_GET_CONFIG_REQUEST:
1029
824
        break;
1030
1031
2.60k
    case OFPTYPE_GET_CONFIG_REPLY:
1032
2.60k
        return ofp_print_get_config_reply(string, oh);
1033
1034
2.86k
    case OFPTYPE_SET_CONFIG:
1035
2.86k
        return ofp_print_set_config(string, oh);
1036
1037
88.0k
    case OFPTYPE_PACKET_IN:
1038
88.0k
        return ofp_print_packet_in(string, oh, port_map, table_map, verbosity);
1039
1040
9.10k
    case OFPTYPE_FLOW_REMOVED:
1041
9.10k
        return ofp_print_flow_removed(string, oh, port_map, table_map);
1042
1043
1.87k
    case OFPTYPE_PORT_STATUS:
1044
1.87k
        return ofp_print_port_status(string, oh);
1045
1046
125k
    case OFPTYPE_PACKET_OUT:
1047
125k
        return ofp_print_packet_out(string, oh, port_map, table_map,
1048
125k
                                    verbosity);
1049
1050
64.2k
    case OFPTYPE_FLOW_MOD:
1051
64.2k
        return ofputil_flow_mod_format(string, oh, port_map, table_map,
1052
64.2k
                                       verbosity);
1053
1054
3.31k
    case OFPTYPE_PORT_MOD:
1055
3.31k
        return ofp_print_port_mod(string, oh, port_map);
1056
1057
6.46k
    case OFPTYPE_TABLE_MOD:
1058
6.46k
        return ofp_print_table_mod(string, oh, table_map);
1059
1060
5.12k
    case OFPTYPE_METER_MOD:
1061
5.12k
        return ofp_print_meter_mod(string, oh);
1062
1063
1.59k
    case OFPTYPE_BARRIER_REQUEST:
1064
1.61k
    case OFPTYPE_BARRIER_REPLY:
1065
1.61k
        break;
1066
1067
1.82k
    case OFPTYPE_QUEUE_GET_CONFIG_REQUEST:
1068
1.82k
        return ofputil_queue_get_config_request_format(string, oh, port_map);
1069
1070
10.2k
    case OFPTYPE_QUEUE_GET_CONFIG_REPLY:
1071
10.2k
        return ofputil_queue_get_config_reply_format(string, oh, port_map);
1072
1073
2.07k
    case OFPTYPE_ROLE_REQUEST:
1074
11.5k
    case OFPTYPE_ROLE_REPLY:
1075
11.5k
        return ofp_print_role_message(string, oh);
1076
430
    case OFPTYPE_ROLE_STATUS:
1077
430
        return ofp_print_role_status_message(string, oh);
1078
1079
1.37k
    case OFPTYPE_REQUESTFORWARD:
1080
1.37k
        return ofp_print_requestforward(string, oh, port_map, table_map);
1081
1082
866
    case OFPTYPE_TABLE_STATUS:
1083
866
        return ofp_print_table_status_message(string, oh, table_map);
1084
1085
322
    case OFPTYPE_METER_STATS_REQUEST:
1086
1.28k
    case OFPTYPE_METER_CONFIG_STATS_REQUEST:
1087
1.28k
        return ofp_print_meter_stats_request(string, oh);
1088
1089
2.84k
    case OFPTYPE_METER_STATS_REPLY:
1090
2.84k
        return ofp_print_meter_stats_reply(string, oh);
1091
1092
1.09k
    case OFPTYPE_METER_CONFIG_STATS_REPLY:
1093
1.09k
        return ofp_print_meter_config_reply(string, oh);
1094
1095
197
    case OFPTYPE_METER_FEATURES_STATS_REPLY:
1096
197
        return ofp_print_meter_features_reply(string, oh);
1097
1098
47
    case OFPTYPE_DESC_STATS_REQUEST:
1099
51
    case OFPTYPE_METER_FEATURES_STATS_REQUEST:
1100
51
        break;
1101
1102
1.78k
    case OFPTYPE_FLOW_STATS_REQUEST:
1103
5.72k
    case OFPTYPE_AGGREGATE_STATS_REQUEST:
1104
5.72k
        return ofp_print_flow_stats_request(string, oh, port_map, table_map);
1105
1106
18
    case OFPTYPE_TABLE_STATS_REQUEST:
1107
18
        break;
1108
1109
1.11k
    case OFPTYPE_PORT_STATS_REQUEST:
1110
1.11k
        return ofp_print_ofpst_port_request(string, oh, port_map);
1111
1112
1.57k
    case OFPTYPE_QUEUE_STATS_REQUEST:
1113
1.57k
        return ofputil_queue_stats_request_format(string, oh, port_map);
1114
1115
24
    case OFPTYPE_DESC_STATS_REPLY:
1116
24
        return ofp_print_ofpst_desc_reply(string, oh);
1117
1118
19.9k
    case OFPTYPE_FLOW_STATS_REPLY:
1119
19.9k
        return ofp_print_flow_stats_reply(string, oh, port_map, table_map);
1120
1121
11.4k
    case OFPTYPE_QUEUE_STATS_REPLY:
1122
11.4k
        return ofputil_queue_stats_reply_format(string, oh, port_map,
1123
11.4k
                                                verbosity);
1124
1125
11.2k
    case OFPTYPE_PORT_STATS_REPLY:
1126
11.2k
        return ofp_print_ofpst_port_reply(string, oh, port_map, verbosity);
1127
1128
3.95k
    case OFPTYPE_TABLE_STATS_REPLY:
1129
3.95k
        return ofp_print_table_stats_reply(string, oh);
1130
1131
1.47k
    case OFPTYPE_AGGREGATE_STATS_REPLY:
1132
1.47k
        return ofp_print_aggregate_stats_reply(string, oh);
1133
1134
345
    case OFPTYPE_PORT_DESC_STATS_REQUEST:
1135
345
        return ofp_print_ofpst_port_desc_request(string, oh, port_map);
1136
1137
1.26k
    case OFPTYPE_PORT_DESC_STATS_REPLY:
1138
1.26k
        return ofp_print_ofpst_port_desc_reply(string, oh);
1139
1140
69
    case OFPTYPE_FLOW_MOD_TABLE_ID:
1141
69
        return ofp_print_nxt_flow_mod_table_id(string, oh);
1142
1143
1.81k
    case OFPTYPE_SET_FLOW_FORMAT:
1144
1.81k
        return ofp_print_nxt_set_flow_format(string, oh);
1145
1146
274
    case OFPTYPE_SET_PACKET_IN_FORMAT:
1147
274
        return ofp_print_nxt_set_packet_in_format(string, oh);
1148
1149
0
    case OFPTYPE_FLOW_AGE:
1150
0
        break;
1151
1152
36
    case OFPTYPE_SET_CONTROLLER_ID:
1153
36
        return ofp_print_nxt_set_controller_id(string, ofpmsg_body(oh));
1154
1155
10.2k
    case OFPTYPE_GET_ASYNC_REPLY:
1156
14.4k
    case OFPTYPE_SET_ASYNC_CONFIG:
1157
14.4k
        return ofp_print_set_async_config(string, oh, type);
1158
0
    case OFPTYPE_GET_ASYNC_REQUEST:
1159
0
        break;
1160
78
    case OFPTYPE_FLOW_MONITOR_CANCEL:
1161
78
        return ofp_print_nxt_flow_monitor_cancel(string, msg);
1162
1163
0
    case OFPTYPE_FLOW_MONITOR_PAUSED:
1164
0
    case OFPTYPE_FLOW_MONITOR_RESUMED:
1165
0
        break;
1166
1167
6.03k
    case OFPTYPE_FLOW_MONITOR_STATS_REQUEST:
1168
6.03k
        return ofp_print_flow_monitor_request(string, msg, port_map,
1169
6.03k
                                              table_map);
1170
1171
8.75k
    case OFPTYPE_FLOW_MONITOR_STATS_REPLY:
1172
8.75k
        return ofp_print_flow_monitor_reply(string, msg, port_map,
1173
8.75k
                                            table_map);
1174
1175
3.03k
    case OFPTYPE_BUNDLE_CONTROL:
1176
3.03k
        return ofp_print_bundle_ctrl(string, msg);
1177
1178
4.50k
    case OFPTYPE_BUNDLE_ADD_MESSAGE:
1179
4.50k
        return ofp_print_bundle_add(string, msg, port_map, table_map,
1180
4.50k
                                    verbosity);
1181
1182
1.87k
    case OFPTYPE_NXT_TLV_TABLE_MOD:
1183
1.87k
        return ofp_print_tlv_table_mod(string, msg);
1184
1185
0
    case OFPTYPE_NXT_TLV_TABLE_REQUEST:
1186
0
        break;
1187
1188
2.06k
    case OFPTYPE_NXT_TLV_TABLE_REPLY:
1189
2.06k
        return ofp_print_tlv_table_reply(string, msg);
1190
1191
3.84k
    case OFPTYPE_NXT_RESUME:
1192
3.84k
        return ofp_print_packet_in(string, msg, port_map, table_map,
1193
3.84k
                                   verbosity);
1194
0
    case OFPTYPE_IPFIX_BRIDGE_STATS_REQUEST:
1195
0
        break;
1196
545
    case OFPTYPE_IPFIX_BRIDGE_STATS_REPLY:
1197
545
        return ofp_print_nxst_ipfix_bridge_reply(string, oh);
1198
0
    case OFPTYPE_IPFIX_FLOW_STATS_REQUEST:
1199
0
        break;
1200
1.57k
    case OFPTYPE_IPFIX_FLOW_STATS_REPLY:
1201
1.57k
        return ofp_print_nxst_ipfix_flow_reply(string, oh);
1202
1203
35
    case OFPTYPE_CT_FLUSH_ZONE:
1204
35
        return ofp_print_nxt_ct_flush_zone(string, ofpmsg_body(oh));
1205
13.7k
    case OFPTYPE_CT_FLUSH:
1206
13.7k
        return ofp_print_nxt_ct_flush(string, oh);
1207
1.08M
    }
1208
1209
2.56k
    return 0;
1210
1.08M
}
1211
1212
static void
1213
add_newline(struct ds *s)
1214
2.72M
{
1215
2.72M
    if (s->length && s->string[s->length - 1] != '\n') {
1216
156k
        ds_put_char(s, '\n');
1217
156k
    }
1218
2.72M
}
1219
1220
/* Composes and returns a string representing the OpenFlow packet of 'len'
1221
 * bytes at 'oh' at the given 'verbosity' level.  0 is a minimal amount of
1222
 * verbosity and higher numbers increase verbosity.  The caller is responsible
1223
 * for freeing the string. */
1224
char *
1225
ofp_to_string(const void *oh_, size_t len,
1226
              const struct ofputil_port_map *port_map,
1227
              const struct ofputil_table_map *table_map,
1228
              int verbosity)
1229
1.93M
{
1230
1.93M
    struct ds string = DS_EMPTY_INITIALIZER;
1231
1.93M
    const struct ofp_header *oh = oh_;
1232
1233
1.93M
    if (!len) {
1234
262k
        ds_put_cstr(&string, "OpenFlow message is empty\n");
1235
1.67M
    } else if (len < sizeof(struct ofp_header)) {
1236
55.0k
        ds_put_format(&string, "OpenFlow packet too short (only %"PRIuSIZE" bytes):\n",
1237
55.0k
                      len);
1238
1.62M
    } else if (ntohs(oh->length) > len) {
1239
8.66k
        enum ofperr error;
1240
8.66k
        enum ofpraw raw;
1241
1242
8.66k
        error = ofpraw_decode_partial(&raw, oh, len);
1243
8.66k
        if (!error) {
1244
3.86k
            ofp_header_to_string__(oh, raw, &string);
1245
3.86k
            ds_put_char(&string, '\n');
1246
3.86k
        }
1247
1248
8.66k
        ds_put_format(&string,
1249
8.66k
                      "(***truncated to %"PRIuSIZE" bytes from %"PRIu16"***)\n",
1250
8.66k
                      len, ntohs(oh->length));
1251
1.61M
    } else if (ntohs(oh->length) < len) {
1252
285
        ds_put_format(&string,
1253
285
                      "(***only uses %"PRIu16" bytes out of %"PRIuSIZE"***)\n",
1254
285
                      ntohs(oh->length), len);
1255
1.61M
    } else {
1256
1.61M
        enum ofperr error;
1257
1.61M
        enum ofpraw raw;
1258
1259
1.61M
        error = ofpraw_decode(&raw, oh);
1260
1.61M
        if (!error) {
1261
1.08M
            ofp_header_to_string__(oh, raw, &string);
1262
1.08M
            size_t header_len = string.length;
1263
1264
1.08M
            error = ofp_to_string__(oh, port_map, table_map,
1265
1.08M
                                    raw, &string, verbosity);
1266
1.08M
            if (error) {
1267
516k
                if (string.length > header_len) {
1268
70.5k
                    ds_chomp(&string, ' ');
1269
70.5k
                    add_newline(&string);
1270
445k
                } else {
1271
445k
                    ds_put_char(&string, ' ');
1272
445k
                }
1273
516k
                ofp_print_error(&string, error);
1274
572k
            } else {
1275
572k
                ds_chomp(&string, ' ');
1276
572k
            }
1277
1.08M
        } else {
1278
524k
            ofp_print_error(&string, error);
1279
524k
        }
1280
1281
1.61M
        if (verbosity >= 5 || error) {
1282
1.04M
            add_newline(&string);
1283
1.04M
            ds_put_hex_dump(&string, oh, len, 0, true);
1284
1.04M
        }
1285
1286
1.61M
        add_newline(&string);
1287
1.61M
        return ds_steal_cstr(&string);
1288
1.61M
    }
1289
326k
    ds_put_hex_dump(&string, oh, len, 0, true);
1290
326k
    return ds_steal_cstr(&string);
1291
1.93M
}
1292

1293
static void
1294
print_and_free(FILE *stream, char *string)
1295
1.61M
{
1296
1.61M
    fputs(string, stream);
1297
1.61M
    free(string);
1298
1.61M
}
1299
1300
/* Pretty-print the OpenFlow packet of 'len' bytes at 'oh' to 'stream' at the
1301
 * given 'verbosity' level.  0 is a minimal amount of verbosity and higher
1302
 * numbers increase verbosity. */
1303
void
1304
ofp_print(FILE *stream, const void *oh, size_t len,
1305
          const struct ofputil_port_map *port_map,
1306
          const struct ofputil_table_map *table_map, int verbosity)
1307
1.61M
{
1308
1.61M
    print_and_free(stream, ofp_to_string(oh, len, port_map, table_map,
1309
1.61M
                                         verbosity));
1310
1.61M
}
1311
1312
/* Dumps the contents of the Ethernet frame in the 'len' bytes starting at
1313
 * 'data' to 'stream'. */
1314
void
1315
ofp_print_packet(FILE *stream, const void *data, size_t len,
1316
                 ovs_be32 packet_type)
1317
2.18k
{
1318
2.18k
    print_and_free(stream, ofp_packet_to_string(data, len, packet_type));
1319
2.18k
}
1320
1321
void
1322
ofp_print_dp_packet(FILE *stream, const struct dp_packet *packet)
1323
0
{
1324
0
    print_and_free(stream, ofp_dp_packet_to_string(packet));
1325
0
}