Coverage Report

Created: 2026-05-01 06:12

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
53.3k
{
75
53.3k
    struct ds ds = DS_EMPTY_INITIALIZER;
76
53.3k
    struct dp_packet buf;
77
53.3k
    struct flow flow;
78
53.3k
    size_t l4_size;
79
80
53.3k
    dp_packet_use_const(&buf, data, len);
81
53.3k
    buf.packet_type = packet_type;
82
53.3k
    flow_extract(&buf, &flow);
83
53.3k
    flow_format(&ds, &flow, NULL);
84
85
53.3k
    l4_size = dp_packet_l4_size(&buf);
86
87
53.3k
    if (flow.nw_proto == IPPROTO_TCP && l4_size >= TCP_HEADER_LEN) {
88
8.36k
        struct tcp_header *th = dp_packet_l4(&buf);
89
8.36k
        ds_put_format(&ds, " tcp_csum:%"PRIx16, ntohs(th->tcp_csum));
90
44.9k
    } else if (flow.nw_proto == IPPROTO_UDP && l4_size >= UDP_HEADER_LEN) {
91
521
        struct udp_header *uh = dp_packet_l4(&buf);
92
521
        ds_put_format(&ds, " udp_csum:%"PRIx16, ntohs(uh->udp_csum));
93
44.4k
    } else if (flow.nw_proto == IPPROTO_SCTP && l4_size >= SCTP_HEADER_LEN) {
94
488
        struct sctp_header *sh = dp_packet_l4(&buf);
95
488
        ds_put_format(&ds, " sctp_csum:%"PRIx32,
96
488
                      ntohl(get_16aligned_be32(&sh->sctp_csum)));
97
43.9k
    } else if (flow.nw_proto == IPPROTO_ICMP && l4_size >= ICMP_HEADER_LEN) {
98
2.42k
        struct icmp_header *icmph = dp_packet_l4(&buf);
99
2.42k
        ds_put_format(&ds, " icmp_csum:%"PRIx16,
100
2.42k
                      ntohs(icmph->icmp_csum));
101
41.5k
    } else if (flow.nw_proto == IPPROTO_ICMPV6 && l4_size >= ICMP6_HEADER_LEN) {
102
1.16k
        struct icmp6_header *icmp6h = dp_packet_l4(&buf);
103
1.16k
        ds_put_format(&ds, " icmp6_csum:%"PRIx16,
104
1.16k
                      ntohs(icmp6h->icmp6_cksum));
105
1.16k
    }
106
107
53.3k
    ds_put_char(&ds, '\n');
108
109
53.3k
    return ds_cstr(&ds);
110
53.3k
}
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
83.1k
{
125
83.1k
    struct ofputil_packet_in_private pin;
126
83.1k
    uint32_t buffer_id;
127
83.1k
    size_t total_len;
128
83.1k
    enum ofperr error = ofputil_decode_packet_in_private(oh, true, NULL, NULL,
129
83.1k
                                                         &pin, &total_len,
130
83.1k
                                                         &buffer_id);
131
83.1k
    if (!error) {
132
50.4k
        ofputil_packet_in_private_format(string, &pin, total_len, buffer_id,
133
50.4k
                                         port_map, table_map, verbosity);
134
50.4k
        ofputil_packet_in_private_destroy(&pin);
135
50.4k
    }
136
83.1k
    return error;
137
83.1k
}
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
135k
{
144
135k
    struct ofputil_packet_out po;
145
135k
    struct ofpbuf ofpacts;
146
135k
    enum ofperr error;
147
148
135k
    ofpbuf_init(&ofpacts, 64);
149
135k
    error = ofputil_decode_packet_out(&po, oh, NULL, &ofpacts);
150
135k
    if (!error) {
151
17.5k
        ofputil_packet_out_format(string, &po, port_map, table_map, verbosity);
152
17.5k
    }
153
135k
    ofpbuf_uninit(&ofpacts);
154
135k
    return error;
155
135k
}
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
121k
{
162
121k
    int n = 0;
163
121k
    int i;
164
165
121k
    if (!bits) {
166
16.7k
        ds_put_cstr(string, "0");
167
16.7k
        return;
168
16.7k
    }
169
170
3.46M
    for (i = 0; i < 32; i++) {
171
3.36M
        uint32_t bit = UINT32_C(1) << i;
172
173
3.36M
        if (bits & bit) {
174
544k
            const char *name = bit_to_name(bit);
175
544k
            if (name) {
176
528k
                if (n++) {
177
424k
                    ds_put_char(string, separator);
178
424k
                }
179
528k
                ds_put_cstr(string, name);
180
528k
                bits &= ~bit;
181
528k
            }
182
544k
        }
183
3.36M
    }
184
185
105k
    if (bits) {
186
1.55k
        if (n) {
187
703
            ds_put_char(string, separator);
188
703
        }
189
1.55k
        ds_put_format(string, "0x%"PRIx32, bits);
190
1.55k
    }
191
105k
}
192
193
static enum ofperr
194
ofp_print_switch_features(struct ds *string, const struct ofp_header *oh)
195
3.49k
{
196
3.49k
    struct ofputil_switch_features features;
197
3.49k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
198
3.49k
    enum ofperr error = ofputil_pull_switch_features(&b, &features);
199
3.49k
    if (!error) {
200
3.49k
        ofputil_switch_features_format(string, &features);
201
3.49k
        error = ofputil_phy_ports_format(string, oh->version, &b);
202
3.49k
    }
203
3.49k
    return error;
204
3.49k
}
205
206
static enum ofperr
207
ofp_print_set_config(struct ds *string, const struct ofp_header *oh)
208
3.25k
{
209
3.25k
    struct ofputil_switch_config config;
210
3.25k
    enum ofperr error;
211
212
3.25k
    error = ofputil_decode_set_config(oh, &config);
213
3.25k
    if (error) {
214
2.70k
        return error;
215
2.70k
    }
216
547
    ofputil_switch_config_format(string, &config);
217
547
    return 0;
218
3.25k
}
219
220
static enum ofperr
221
ofp_print_get_config_reply(struct ds *string, const struct ofp_header *oh)
222
2.14k
{
223
2.14k
    struct ofputil_switch_config config;
224
2.14k
    ofputil_decode_get_config_reply(oh, &config);
225
2.14k
    ofputil_switch_config_format(string, &config);
226
2.14k
    return 0;
227
2.14k
}
228
229
static enum ofperr
230
ofp_print_table_features_reply(struct ds *s, const struct ofp_header *oh)
231
14.2k
{
232
14.2k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
233
234
14.2k
    struct ofputil_table_features prev;
235
14.2k
    int first_ditto = -1, last_ditto = -1;
236
15.0k
    for (int i = 0; ; i++) {
237
15.0k
        struct ofputil_table_features tf;
238
15.0k
        struct ofpbuf raw_properties;
239
15.0k
        int retval = ofputil_decode_table_features(&b, &tf, &raw_properties);
240
15.0k
        if (retval) {
241
14.2k
            ofputil_table_features_format_finish(s, first_ditto, last_ditto);
242
14.2k
            return retval != EOF ? retval : 0;
243
14.2k
        }
244
245
820
        ofputil_table_features_format(s, &tf, i ? &prev : NULL, NULL, NULL,
246
820
                                      &first_ditto, &last_ditto);
247
820
        prev = tf;
248
820
    }
249
14.2k
}
250
251
void
252
ofp_print_duration(struct ds *string, unsigned int sec, unsigned int nsec)
253
14.5k
{
254
14.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
14.5k
    if (nsec > 0) {
267
12.3k
        unsigned int msec = nsec / 1000000;
268
12.3k
        if (msec * 1000000 == nsec) {
269
259
            ds_put_format(string, ".%03u", msec);
270
12.1k
        } else {
271
12.1k
            ds_put_format(string, ".%09u", nsec);
272
13.7k
            while (string->string[string->length - 1] == '0') {
273
1.62k
                string->length--;
274
1.62k
            }
275
12.1k
        }
276
12.3k
    }
277
14.5k
    ds_put_char(string, 's');
278
14.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
6.16k
{
285
6.16k
    struct ofputil_flow_removed fr;
286
6.16k
    enum ofperr error = ofputil_decode_flow_removed(&fr, oh);
287
6.16k
    if (!error) {
288
2.04k
        ofputil_flow_removed_format(string, &fr, port_map, table_map);
289
2.04k
    }
290
6.16k
    return error;
291
6.16k
}
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
2.98k
{
297
2.98k
    struct ofputil_port_mod pm;
298
2.98k
    enum ofperr error = ofputil_decode_port_mod(oh, &pm, true);
299
2.98k
    if (!error) {
300
1.67k
        ofputil_port_mod_format(string, &pm, port_map);
301
1.67k
    }
302
2.98k
    return error;
303
2.98k
}
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.06k
{
309
6.06k
    struct ofputil_table_mod tm;
310
6.06k
    enum ofperr error = ofputil_decode_table_mod(oh, &tm);
311
6.06k
    if (!error) {
312
3.15k
        ofputil_table_mod_format(string, &tm, table_map);
313
3.15k
    }
314
6.06k
    return error;
315
6.06k
}
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
2.43k
{
321
2.43k
    struct ofputil_table_status ts;
322
2.43k
    enum ofperr error = ofputil_decode_table_status(oh, &ts);
323
2.43k
    if (!error) {
324
485
        ofputil_format_table_status(string, &ts, table_map);
325
485
    }
326
2.43k
    return error;
327
2.43k
}
328
329
static enum ofperr
330
ofp_print_meter_mod(struct ds *s, const struct ofp_header *oh)
331
5.54k
{
332
5.54k
    struct ofputil_meter_mod mm;
333
5.54k
    struct ofpbuf bands;
334
335
5.54k
    ofpbuf_init(&bands, 64);
336
5.54k
    enum ofperr error = ofputil_decode_meter_mod(oh, &mm, &bands);
337
5.54k
    if (!error) {
338
1.37k
        ofputil_format_meter_mod(s, &mm);
339
1.37k
    }
340
5.54k
    ofpbuf_uninit(&bands);
341
342
5.54k
    return error;
343
5.54k
}
344
345
static enum ofperr
346
ofp_print_meter_stats_request(struct ds *s, const struct ofp_header *oh)
347
945
{
348
945
    uint32_t meter_id;
349
350
945
    ofputil_decode_meter_request(oh, &meter_id);
351
945
    ds_put_char(s, ' ');
352
353
945
    ofputil_format_meter_id(s, meter_id, '=');
354
355
945
    return 0;
356
945
}
357
358
static enum ofperr
359
ofp_print_meter_features_reply(struct ds *s, const struct ofp_header *oh)
360
425
{
361
425
    struct ofputil_meter_features mf;
362
425
    ofputil_decode_meter_features(oh, &mf);
363
425
    ofputil_format_meter_features(s, &mf);
364
425
    return 0;
365
425
}
366
367
static enum ofperr
368
ofp_print_meter_config_reply(struct ds *s, const struct ofp_header *oh)
369
1.08k
{
370
1.08k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
371
1.08k
    struct ofpbuf bands;
372
1.08k
    int retval;
373
374
1.08k
    ofpbuf_init(&bands, 64);
375
2.33k
    for (;;) {
376
2.33k
        struct ofputil_meter_config mc;
377
378
2.33k
        retval = ofputil_decode_meter_config(&b, &mc, &bands);
379
2.33k
        if (retval) {
380
1.08k
            break;
381
1.08k
        }
382
1.24k
        ds_put_char(s, '\n');
383
1.24k
        ofputil_format_meter_config(s, &mc);
384
1.24k
    }
385
1.08k
    ofpbuf_uninit(&bands);
386
387
1.08k
    return retval != EOF ? retval : 0;
388
1.08k
}
389
390
static enum ofperr
391
ofp_print_meter_stats_reply(struct ds *s, const struct ofp_header *oh)
392
1.44k
{
393
1.44k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
394
1.44k
    struct ofpbuf bands;
395
1.44k
    int retval;
396
397
1.44k
    ofpbuf_init(&bands, 64);
398
2.62k
    for (;;) {
399
2.62k
        struct ofputil_meter_stats ms;
400
401
2.62k
        retval = ofputil_decode_meter_stats(&b, &ms, &bands);
402
2.62k
        if (retval) {
403
1.44k
            break;
404
1.44k
        }
405
1.17k
        ds_put_char(s, '\n');
406
1.17k
        ofputil_format_meter_stats(s, &ms);
407
1.17k
    }
408
1.44k
    ofpbuf_uninit(&bands);
409
410
1.44k
    return retval != EOF ? retval : 0;
411
1.44k
}
412
413
static void
414
ofp_print_error(struct ds *string, enum ofperr error)
415
1.10M
{
416
1.10M
    ds_put_format(string, "***decode error: %s***\n", ofperr_get_name(error));
417
1.10M
}
418
419
static enum ofperr
420
ofp_print_hello(struct ds *string, const struct ofp_header *oh)
421
36.0k
{
422
36.0k
    ofputil_hello_format(string, oh);
423
36.0k
    return 0;
424
36.0k
}
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
634k
{
431
634k
    struct ofpbuf payload;
432
634k
    enum ofperr error = ofperr_decode_msg(oh, &payload);
433
634k
    if (!error) {
434
225k
        return OFPERR_OFPBRC_BAD_LEN;
435
225k
    }
436
409k
    ofperr_msg_format(string, error, &payload, port_map, table_map);
437
409k
    ofpbuf_uninit(&payload);
438
439
409k
    return 0;
440
634k
}
441
442
static enum ofperr
443
ofp_print_port_status(struct ds *string, const struct ofp_header *oh)
444
1.75k
{
445
1.75k
    struct ofputil_port_status ps;
446
1.75k
    enum ofperr error = ofputil_decode_port_status(oh, &ps);
447
1.75k
    if (!error) {
448
1.28k
        ofputil_port_status_format(string, &ps);
449
1.28k
    }
450
1.75k
    return error;
451
1.75k
}
452
453
static enum ofperr
454
ofp_print_ofpst_desc_reply(struct ds *string, const struct ofp_header *oh)
455
19
{
456
19
    const struct ofp_desc_stats *ods = ofpmsg_body(oh);
457
458
19
    ds_put_char(string, '\n');
459
19
    ds_put_format(string, "Manufacturer: %.*s\n",
460
19
            (int) sizeof ods->mfr_desc, ods->mfr_desc);
461
19
    ds_put_format(string, "Hardware: %.*s\n",
462
19
            (int) sizeof ods->hw_desc, ods->hw_desc);
463
19
    ds_put_format(string, "Software: %.*s\n",
464
19
            (int) sizeof ods->sw_desc, ods->sw_desc);
465
19
    ds_put_format(string, "Serial Num: %.*s\n",
466
19
            (int) sizeof ods->serial_num, ods->serial_num);
467
19
    ds_put_format(string, "DP Description: %.*s\n",
468
19
            (int) sizeof ods->dp_desc, ods->dp_desc);
469
470
19
    return 0;
471
19
}
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.36k
{
478
5.36k
    struct ofputil_flow_stats_request fsr;
479
5.36k
    enum ofperr error = ofputil_decode_flow_stats_request(&fsr, oh, NULL,
480
5.36k
                                                          NULL);
481
5.36k
    if (!error) {
482
1.71k
        ofputil_flow_stats_request_format(string, &fsr, port_map, table_map);
483
1.71k
    }
484
5.36k
    return error;
485
5.36k
}
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
12.4k
{
492
12.4k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
493
12.4k
    struct ofpbuf ofpacts;
494
12.4k
    int retval;
495
496
12.4k
    ofpbuf_init(&ofpacts, 64);
497
16.6k
    for (;;) {
498
16.6k
        struct ofputil_flow_stats fs;
499
500
16.6k
        retval = ofputil_decode_flow_stats_reply(&fs, &b, true, &ofpacts);
501
16.6k
        if (retval) {
502
12.4k
            break;
503
12.4k
        }
504
4.11k
        ds_put_cstr(string, "\n ");
505
4.11k
        ofputil_flow_stats_format(string, &fs, port_map, table_map, true);
506
4.11k
     }
507
12.4k
    ofpbuf_uninit(&ofpacts);
508
509
12.4k
    return retval != EOF ? retval : 0;
510
12.4k
}
511
512
static enum ofperr
513
ofp_print_aggregate_stats_reply(struct ds *string, const struct ofp_header *oh)
514
2.55k
{
515
2.55k
    struct ofputil_aggregate_stats as;
516
2.55k
    enum ofperr error;
517
518
2.55k
    error = ofputil_decode_aggregate_stats_reply(&as, oh);
519
2.55k
    if (!error) {
520
136
        ofputil_aggregate_stats_format(string, &as);
521
136
    }
522
2.55k
    return error;
523
2.55k
}
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
2.90k
{
529
2.90k
    ofp_port_t ofp10_port;
530
2.90k
    enum ofperr error;
531
532
2.90k
    error = ofputil_decode_port_stats_request(oh, &ofp10_port);
533
2.90k
    if (error) {
534
2.31k
        return error;
535
2.31k
    }
536
537
594
    ds_put_cstr(string, " port_no=");
538
594
    ofputil_format_port(ofp10_port, port_map, string);
539
540
594
    return 0;
541
2.90k
}
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
9.24k
{
548
9.24k
    ds_put_format(string, " %"PRIuSIZE" ports\n",
549
9.24k
                  ofputil_count_port_stats(oh));
550
9.24k
    if (verbosity < 1) {
551
0
        return 0;
552
0
    }
553
554
9.24k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
555
18.1k
    for (;;) {
556
18.1k
        struct ofputil_port_stats ps;
557
18.1k
        int retval;
558
559
18.1k
        retval = ofputil_decode_port_stats(&ps, &b);
560
18.1k
        if (retval) {
561
9.24k
            return retval != EOF ? retval : 0;
562
9.24k
        }
563
8.92k
        ofputil_format_port_stats(string, &ps, port_map);
564
8.92k
        netdev_free_custom_stats_counters(&ps.custom_stats);
565
8.92k
    }
566
9.24k
}
567
568
static enum ofperr
569
ofp_print_table_stats_reply(struct ds *string, const struct ofp_header *oh)
570
4.03k
{
571
4.03k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
572
4.03k
    ofpraw_pull_assert(&b);
573
574
4.03k
    struct ofputil_table_features prev_features;
575
4.03k
    struct ofputil_table_stats prev_stats;
576
4.03k
    int first_ditto = -1, last_ditto = -1;
577
37.3k
    for (int i = 0;; i++) {
578
37.3k
        struct ofputil_table_features features;
579
37.3k
        struct ofputil_table_stats stats;
580
37.3k
        int retval;
581
582
37.3k
        retval = ofputil_decode_table_stats_reply(&b, &stats, &features);
583
37.3k
        if (retval) {
584
4.03k
            ofputil_table_features_format_finish(string,
585
4.03k
                                                 first_ditto, last_ditto);
586
4.03k
            return retval != EOF ? retval : 0;
587
4.03k
        }
588
589
33.2k
        ofputil_table_features_format(string,
590
33.2k
                                      &features, i ? &prev_features : NULL,
591
33.2k
                                      &stats, i ? &prev_stats : NULL,
592
33.2k
                                      &first_ditto, &last_ditto);
593
33.2k
        prev_features = features;
594
33.2k
        prev_stats = stats;
595
33.2k
    }
596
4.03k
}
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
451
{
603
451
    enum ofperr error;
604
451
    ofp_port_t port;
605
606
451
    error = ofputil_decode_port_desc_stats_request(oh, &port);
607
451
    if (error) {
608
89
        return error;
609
89
    }
610
611
362
    ds_put_cstr(string, " port=");
612
362
    ofputil_format_port(port, port_map, string);
613
614
362
    return 0;
615
451
}
616
617
static enum ofperr
618
ofp_print_ofpst_port_desc_reply(struct ds *string,
619
                                const struct ofp_header *oh)
620
1.71k
{
621
1.71k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
622
1.71k
    ofpraw_pull_assert(&b);
623
1.71k
    ds_put_char(string, '\n');
624
1.71k
    return ofputil_phy_ports_format(string, oh->version, &b);
625
1.71k
}
626
627
static void
628
ofp_print_stats(struct ds *string, const struct ofp_header *oh)
629
95.9k
{
630
95.9k
    uint16_t flags = ofpmp_flags(oh);
631
632
95.9k
    if (flags) {
633
85.0k
        ds_put_cstr(string, " flags=");
634
85.0k
        if ((!ofpmsg_is_stat_request(oh) || oh->version >= OFP13_VERSION)
635
77.3k
            && (flags & OFPSF_REPLY_MORE)) {
636
37.4k
            ds_put_cstr(string, "[more]");
637
37.4k
            flags &= ~OFPSF_REPLY_MORE;
638
37.4k
        }
639
85.0k
        if (flags) {
640
84.2k
            ds_put_format(string, "[***unknown flags 0x%04"PRIx16"***]",
641
84.2k
                          flags);
642
84.2k
        }
643
85.0k
    }
644
95.9k
}
645
646
static enum ofperr
647
ofp_print_echo(struct ds *string, const struct ofp_header *oh, int verbosity)
648
12.2k
{
649
12.2k
    size_t len = ntohs(oh->length);
650
651
12.2k
    ds_put_format(string, " %"PRIuSIZE" bytes of payload\n", len - sizeof *oh);
652
12.2k
    if (verbosity > 1) {
653
12.0k
        ds_put_hex_dump(string, oh + 1, len - sizeof *oh, 0, true);
654
12.0k
    }
655
656
12.2k
    return 0;
657
12.2k
}
658
659
static enum ofperr
660
ofp_print_role_message(struct ds *string, const struct ofp_header *oh)
661
6.10k
{
662
6.10k
    struct ofputil_role_request rr;
663
6.10k
    enum ofperr error = ofputil_decode_role_message(oh, &rr);
664
6.10k
    if (!error) {
665
4.40k
        ofputil_format_role_message(string, &rr);
666
4.40k
    }
667
6.10k
    return error;
668
6.10k
}
669
670
static enum ofperr
671
ofp_print_role_status_message(struct ds *string, const struct ofp_header *oh)
672
549
{
673
549
    struct ofputil_role_status rs;
674
549
    enum ofperr error = ofputil_decode_role_status(oh, &rs);
675
549
    if (!error) {
676
300
        ofputil_format_role_status(string, &rs);
677
300
    }
678
549
    return error;
679
549
}
680
681
static enum ofperr
682
ofp_print_nxt_flow_mod_table_id(struct ds *string, const struct ofp_header *oh)
683
68
{
684
68
    bool enable = ofputil_decode_nx_flow_mod_table_id(oh);
685
68
    ds_put_format(string, " %s", enable ? "enable" : "disable");
686
68
    return 0;
687
68
}
688
689
static enum ofperr
690
ofp_print_nxt_set_flow_format(struct ds *string, const struct ofp_header *oh)
691
612
{
692
612
    enum ofputil_protocol p = ofputil_decode_nx_set_flow_format(oh);
693
612
    ds_put_format(string, " format=%s",
694
612
                  p == OFPUTIL_P_OF10_STD ? "openflow10"
695
612
                  : p == OFPUTIL_P_OF10_NXM ? "nxm"
696
259
                  : "(unknown)");
697
612
    return 0;
698
612
}
699
700
static enum ofperr
701
ofp_print_nxt_set_packet_in_format(struct ds *string,
702
                                   const struct ofp_header *oh)
703
410
{
704
410
    enum ofputil_packet_in_format format;
705
410
    enum ofperr error = ofputil_decode_set_packet_in_format(oh, &format);
706
410
    if (!error) {
707
307
        ds_put_format(string, " format=%s",
708
307
                      ofputil_packet_in_format_to_string(format));
709
307
    }
710
410
    return error;
711
410
}
712
713
static enum ofperr
714
ofp_print_set_async_config(struct ds *string, const struct ofp_header *oh,
715
                           enum ofptype ofptype)
716
10.6k
{
717
10.6k
    struct ofputil_async_cfg basis = OFPUTIL_ASYNC_CFG_INIT;
718
10.6k
    struct ofputil_async_cfg ac;
719
720
10.6k
    bool is_reply = ofptype == OFPTYPE_GET_ASYNC_REPLY;
721
10.6k
    enum ofperr error = ofputil_decode_set_async_config(oh, is_reply,
722
10.6k
                                                        &basis, &ac);
723
10.6k
    if (error) {
724
3.46k
        return error;
725
3.46k
    }
726
7.23k
    ofputil_format_set_async_config(string, &ac);
727
7.23k
    return 0;
728
10.6k
}
729
730
static enum ofperr
731
ofp_print_nxt_set_controller_id(struct ds *string,
732
                                const struct nx_controller_id *nci)
733
52
{
734
52
    ds_put_format(string, " id=%"PRIu16, ntohs(nci->controller_id));
735
52
    return 0;
736
52
}
737
738
static enum ofperr
739
ofp_print_nxt_flow_monitor_cancel(struct ds *string,
740
                                  const struct ofp_header *oh)
741
64
{
742
64
    ds_put_format(string, " id=%"PRIu32,
743
64
                  ofputil_decode_flow_monitor_cancel(oh));
744
64
    return 0;
745
64
}
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
8.92k
{
753
8.92k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
754
11.2k
    for (;;) {
755
11.2k
        struct ofputil_flow_monitor_request request;
756
11.2k
        int retval;
757
758
11.2k
        retval = ofputil_decode_flow_monitor_request(&request, &b);
759
11.2k
        if (retval) {
760
8.92k
            return retval != EOF ? retval : 0;
761
8.92k
        }
762
763
2.37k
        ofputil_flow_monitor_request_format(string, &request,
764
2.37k
                                            port_map, table_map);
765
2.37k
    }
766
8.92k
}
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
9.33k
{
774
9.33k
    uint64_t ofpacts_stub[1024 / 8];
775
9.33k
    struct ofpbuf ofpacts = OFPBUF_STUB_INITIALIZER(ofpacts_stub);
776
9.33k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
777
778
15.6k
    for (;;) {
779
15.6k
        struct ofputil_flow_update update;
780
15.6k
        int retval = ofputil_decode_flow_update(&update, &b, &ofpacts);
781
15.6k
        if (retval) {
782
9.33k
            ofpbuf_uninit(&ofpacts);
783
9.33k
            return retval != EOF ? retval : 0;
784
9.33k
        }
785
6.32k
        ofputil_flow_update_format(string, &update, port_map, table_map);
786
6.32k
    }
787
9.33k
}
788
789
void
790
ofp_print_version(const struct ofp_header *oh,
791
                  struct ds *string)
792
1.20M
{
793
1.20M
    switch (oh->version) {
794
196k
    case OFP10_VERSION:
795
196k
        break;
796
100k
    case OFP11_VERSION:
797
100k
        ds_put_cstr(string, " (OF1.1)");
798
100k
        break;
799
214k
    case OFP12_VERSION:
800
214k
        ds_put_cstr(string, " (OF1.2)");
801
214k
        break;
802
125k
    case OFP13_VERSION:
803
125k
        ds_put_cstr(string, " (OF1.3)");
804
125k
        break;
805
213k
    case OFP14_VERSION:
806
213k
        ds_put_cstr(string, " (OF1.4)");
807
213k
        break;
808
288k
    case OFP15_VERSION:
809
288k
        ds_put_cstr(string, " (OF1.5)");
810
288k
        break;
811
65.5k
    default:
812
65.5k
        ds_put_format(string, " (OF 0x%02"PRIx8")", oh->version);
813
65.5k
        break;
814
1.20M
    }
815
1.20M
    ds_put_format(string, " (xid=0x%"PRIx32"):", ntohl(oh->xid));
816
1.20M
}
817
818
static void
819
ofp_header_to_string__(const struct ofp_header *oh, enum ofpraw raw,
820
                       struct ds *string)
821
1.20M
{
822
1.20M
    ds_put_cstr(string, ofpraw_get_name(raw));
823
1.20M
    ofp_print_version(oh, string);
824
1.20M
}
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
1.81k
{
830
1.81k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
831
3.46k
    for (;;) {
832
3.46k
        struct ofputil_table_desc td;
833
3.46k
        int retval;
834
835
3.46k
        retval = ofputil_decode_table_desc(&b, &td, oh->version);
836
3.46k
        if (retval) {
837
1.81k
            return retval != EOF ? retval : 0;
838
1.81k
        }
839
1.65k
        ofputil_table_desc_format(s, &td, table_map);
840
1.65k
    }
841
1.81k
}
842
843
static enum ofperr
844
ofp_print_bundle_ctrl(struct ds *s, const struct ofp_header *oh)
845
1.04k
{
846
1.04k
    int error;
847
1.04k
    struct ofputil_bundle_ctrl_msg bctrl;
848
849
1.04k
    error = ofputil_decode_bundle_ctrl(oh, &bctrl);
850
1.04k
    if (error) {
851
0
        return error;
852
0
    }
853
1.04k
    ofputil_format_bundle_ctrl_request(s, &bctrl);
854
1.04k
    return 0;
855
1.04k
}
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
3.49k
{
863
3.49k
    struct ofputil_bundle_add_msg badd;
864
3.49k
    int error = ofputil_decode_bundle_add(oh, &badd, NULL);
865
3.49k
    if (!error) {
866
1.62k
        ofputil_format_bundle_add(s, &badd, port_map, table_map, verbosity);
867
1.62k
    }
868
3.49k
    return error;
869
3.49k
}
870
871
static enum ofperr
872
ofp_print_tlv_table_mod(struct ds *s, const struct ofp_header *oh)
873
1.96k
{
874
1.96k
    struct ofputil_tlv_table_mod ttm;
875
1.96k
    int error = ofputil_decode_tlv_table_mod(oh, &ttm);
876
1.96k
    if (!error) {
877
812
        ofputil_format_tlv_table_mod(s, &ttm);
878
812
        ofputil_uninit_tlv_table(&ttm.mappings);
879
812
    }
880
1.96k
    return error;
881
1.96k
}
882
883
static enum ofperr
884
ofp_print_tlv_table_reply(struct ds *s, const struct ofp_header *oh)
885
2.16k
{
886
2.16k
    struct ofputil_tlv_table_reply ttr;
887
2.16k
    int error = ofputil_decode_tlv_table_reply(oh, &ttr);
888
2.16k
    if (!error) {
889
1.03k
        ofputil_format_tlv_table_reply(s, &ttr);
890
1.03k
        ofputil_uninit_tlv_table(&ttr.mappings);
891
1.03k
    }
892
2.16k
    return error;
893
2.16k
}
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.44k
{
902
1.44k
    struct ofputil_requestforward rf;
903
1.44k
    enum ofperr error = ofputil_decode_requestforward(oh, &rf);
904
1.44k
    if (!error) {
905
362
        ofputil_format_requestforward(string, oh->version,
906
362
                                      &rf, port_map, table_map);
907
362
        ofputil_destroy_requestforward(&rf);
908
362
    }
909
1.44k
    return error;
910
1.44k
}
911
912
static enum ofperr
913
ofp_print_nxst_ipfix_bridge_reply(struct ds *string, const struct ofp_header *oh)
914
413
{
915
413
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
916
826
    for (;;) {
917
826
        struct ofputil_ipfix_stats is;
918
826
        int retval;
919
920
826
        retval = ofputil_pull_ipfix_stats(&is, &b);
921
826
        if (retval) {
922
413
            return retval != EOF ? retval : 0;
923
413
        }
924
413
        ofputil_format_ipfix_stats_bridge(string, &is);
925
413
    }
926
413
}
927
928
static enum ofperr
929
ofp_print_nxst_ipfix_flow_reply(struct ds *string, const struct ofp_header *oh)
930
1.25k
{
931
1.25k
    ds_put_format(string, " %"PRIuSIZE" ids\n", ofputil_count_ipfix_stats(oh));
932
933
1.25k
    struct ofpbuf b = ofpbuf_const_initializer(oh, ntohs(oh->length));
934
9.01k
    for (;;) {
935
9.01k
        struct ofputil_ipfix_stats is;
936
9.01k
        int retval;
937
938
9.01k
        retval = ofputil_pull_ipfix_stats(&is, &b);
939
9.01k
        if (retval) {
940
1.25k
            return retval != EOF ? retval : 0;
941
1.25k
        }
942
7.75k
        ofputil_format_ipfix_stats_flow(string, &is);
943
7.75k
    }
944
1.25k
}
945
946
static enum ofperr
947
ofp_print_nxt_ct_flush_zone(struct ds *string, const struct nx_zone_id *nzi)
948
34
{
949
34
    ds_put_format(string, " zone_id=%"PRIu16, ntohs(nzi->zone_id));
950
34
    return 0;
951
34
}
952
953
static enum ofperr
954
ofp_print_nxt_ct_flush(struct ds *string, const struct ofp_header *oh)
955
15.7k
{
956
15.7k
    uint16_t zone_id = 0;
957
15.7k
    struct ofp_ct_match match = {0};
958
959
15.7k
    enum ofperr error = ofp_ct_match_decode(&match, NULL, &zone_id, oh);
960
15.7k
    if (error) {
961
11.0k
        return error;
962
11.0k
    }
963
964
4.64k
    ds_put_format(string, " zone=%"PRIu16" ", zone_id);
965
4.64k
    ofp_ct_match_format(string, &match);
966
967
4.64k
    return 0;
968
15.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.20M
{
976
1.20M
    if (ofpmsg_is_stat(oh)) {
977
95.9k
        ofp_print_stats(string, oh);
978
95.9k
    }
979
980
1.20M
    const void *msg = oh;
981
1.20M
    enum ofptype type = ofptype_from_ofpraw(raw);
982
1.20M
    switch (type) {
983
217
    case OFPTYPE_GROUP_STATS_REQUEST:
984
217
        return ofputil_group_stats_request_format(string, oh);
985
986
1.51k
    case OFPTYPE_GROUP_STATS_REPLY:
987
1.51k
        return ofputil_group_stats_format(string, oh);
988
989
345
    case OFPTYPE_GROUP_DESC_STATS_REQUEST:
990
345
        return ofputil_group_desc_request_format(string, oh);
991
992
2.63k
    case OFPTYPE_GROUP_DESC_STATS_REPLY:
993
2.63k
        return ofputil_group_desc_format(string, oh, port_map, table_map);
994
995
33
    case OFPTYPE_GROUP_FEATURES_STATS_REQUEST:
996
33
        break;
997
998
1.00k
    case OFPTYPE_GROUP_FEATURES_STATS_REPLY:
999
1.00k
        return ofputil_group_features_format(string, oh);
1000
1001
43.0k
    case OFPTYPE_GROUP_MOD:
1002
43.0k
        return ofputil_group_mod_format(string, oh, port_map, table_map);
1003
1004
671
    case OFPTYPE_TABLE_FEATURES_STATS_REQUEST:
1005
14.2k
    case OFPTYPE_TABLE_FEATURES_STATS_REPLY:
1006
14.2k
        return ofp_print_table_features_reply(string, oh);
1007
1008
131
    case OFPTYPE_TABLE_DESC_REQUEST:
1009
1.81k
    case OFPTYPE_TABLE_DESC_REPLY:
1010
1.81k
        return ofp_print_table_desc_reply(string, oh, table_map);
1011
1012
36.0k
    case OFPTYPE_HELLO:
1013
36.0k
        return ofp_print_hello(string, oh);
1014
1015
634k
    case OFPTYPE_ERROR:
1016
634k
        return ofp_print_error_msg(string, oh, port_map, table_map);
1017
1018
6.36k
    case OFPTYPE_ECHO_REQUEST:
1019
12.2k
    case OFPTYPE_ECHO_REPLY:
1020
12.2k
        return ofp_print_echo(string, oh, verbosity);
1021
1022
35
    case OFPTYPE_FEATURES_REQUEST:
1023
35
        break;
1024
1025
3.49k
    case OFPTYPE_FEATURES_REPLY:
1026
3.49k
        return ofp_print_switch_features(string, oh);
1027
1028
1.25k
    case OFPTYPE_GET_CONFIG_REQUEST:
1029
1.25k
        break;
1030
1031
2.14k
    case OFPTYPE_GET_CONFIG_REPLY:
1032
2.14k
        return ofp_print_get_config_reply(string, oh);
1033
1034
3.25k
    case OFPTYPE_SET_CONFIG:
1035
3.25k
        return ofp_print_set_config(string, oh);
1036
1037
79.6k
    case OFPTYPE_PACKET_IN:
1038
79.6k
        return ofp_print_packet_in(string, oh, port_map, table_map, verbosity);
1039
1040
6.16k
    case OFPTYPE_FLOW_REMOVED:
1041
6.16k
        return ofp_print_flow_removed(string, oh, port_map, table_map);
1042
1043
1.75k
    case OFPTYPE_PORT_STATUS:
1044
1.75k
        return ofp_print_port_status(string, oh);
1045
1046
135k
    case OFPTYPE_PACKET_OUT:
1047
135k
        return ofp_print_packet_out(string, oh, port_map, table_map,
1048
135k
                                    verbosity);
1049
1050
71.3k
    case OFPTYPE_FLOW_MOD:
1051
71.3k
        return ofputil_flow_mod_format(string, oh, port_map, table_map,
1052
71.3k
                                       verbosity);
1053
1054
2.98k
    case OFPTYPE_PORT_MOD:
1055
2.98k
        return ofp_print_port_mod(string, oh, port_map);
1056
1057
6.06k
    case OFPTYPE_TABLE_MOD:
1058
6.06k
        return ofp_print_table_mod(string, oh, table_map);
1059
1060
5.54k
    case OFPTYPE_METER_MOD:
1061
5.54k
        return ofp_print_meter_mod(string, oh);
1062
1063
1.85k
    case OFPTYPE_BARRIER_REQUEST:
1064
1.86k
    case OFPTYPE_BARRIER_REPLY:
1065
1.86k
        break;
1066
1067
2.51k
    case OFPTYPE_QUEUE_GET_CONFIG_REQUEST:
1068
2.51k
        return ofputil_queue_get_config_request_format(string, oh, port_map);
1069
1070
10.6k
    case OFPTYPE_QUEUE_GET_CONFIG_REPLY:
1071
10.6k
        return ofputil_queue_get_config_reply_format(string, oh, port_map);
1072
1073
1.10k
    case OFPTYPE_ROLE_REQUEST:
1074
6.10k
    case OFPTYPE_ROLE_REPLY:
1075
6.10k
        return ofp_print_role_message(string, oh);
1076
549
    case OFPTYPE_ROLE_STATUS:
1077
549
        return ofp_print_role_status_message(string, oh);
1078
1079
1.44k
    case OFPTYPE_REQUESTFORWARD:
1080
1.44k
        return ofp_print_requestforward(string, oh, port_map, table_map);
1081
1082
2.43k
    case OFPTYPE_TABLE_STATUS:
1083
2.43k
        return ofp_print_table_status_message(string, oh, table_map);
1084
1085
135
    case OFPTYPE_METER_STATS_REQUEST:
1086
945
    case OFPTYPE_METER_CONFIG_STATS_REQUEST:
1087
945
        return ofp_print_meter_stats_request(string, oh);
1088
1089
1.44k
    case OFPTYPE_METER_STATS_REPLY:
1090
1.44k
        return ofp_print_meter_stats_reply(string, oh);
1091
1092
1.08k
    case OFPTYPE_METER_CONFIG_STATS_REPLY:
1093
1.08k
        return ofp_print_meter_config_reply(string, oh);
1094
1095
425
    case OFPTYPE_METER_FEATURES_STATS_REPLY:
1096
425
        return ofp_print_meter_features_reply(string, oh);
1097
1098
83
    case OFPTYPE_DESC_STATS_REQUEST:
1099
83
    case OFPTYPE_METER_FEATURES_STATS_REQUEST:
1100
83
        break;
1101
1102
1.23k
    case OFPTYPE_FLOW_STATS_REQUEST:
1103
5.36k
    case OFPTYPE_AGGREGATE_STATS_REQUEST:
1104
5.36k
        return ofp_print_flow_stats_request(string, oh, port_map, table_map);
1105
1106
22
    case OFPTYPE_TABLE_STATS_REQUEST:
1107
22
        break;
1108
1109
2.90k
    case OFPTYPE_PORT_STATS_REQUEST:
1110
2.90k
        return ofp_print_ofpst_port_request(string, oh, port_map);
1111
1112
1.47k
    case OFPTYPE_QUEUE_STATS_REQUEST:
1113
1.47k
        return ofputil_queue_stats_request_format(string, oh, port_map);
1114
1115
19
    case OFPTYPE_DESC_STATS_REPLY:
1116
19
        return ofp_print_ofpst_desc_reply(string, oh);
1117
1118
12.4k
    case OFPTYPE_FLOW_STATS_REPLY:
1119
12.4k
        return ofp_print_flow_stats_reply(string, oh, port_map, table_map);
1120
1121
4.61k
    case OFPTYPE_QUEUE_STATS_REPLY:
1122
4.61k
        return ofputil_queue_stats_reply_format(string, oh, port_map,
1123
4.61k
                                                verbosity);
1124
1125
9.24k
    case OFPTYPE_PORT_STATS_REPLY:
1126
9.24k
        return ofp_print_ofpst_port_reply(string, oh, port_map, verbosity);
1127
1128
4.03k
    case OFPTYPE_TABLE_STATS_REPLY:
1129
4.03k
        return ofp_print_table_stats_reply(string, oh);
1130
1131
2.55k
    case OFPTYPE_AGGREGATE_STATS_REPLY:
1132
2.55k
        return ofp_print_aggregate_stats_reply(string, oh);
1133
1134
451
    case OFPTYPE_PORT_DESC_STATS_REQUEST:
1135
451
        return ofp_print_ofpst_port_desc_request(string, oh, port_map);
1136
1137
1.71k
    case OFPTYPE_PORT_DESC_STATS_REPLY:
1138
1.71k
        return ofp_print_ofpst_port_desc_reply(string, oh);
1139
1140
68
    case OFPTYPE_FLOW_MOD_TABLE_ID:
1141
68
        return ofp_print_nxt_flow_mod_table_id(string, oh);
1142
1143
612
    case OFPTYPE_SET_FLOW_FORMAT:
1144
612
        return ofp_print_nxt_set_flow_format(string, oh);
1145
1146
410
    case OFPTYPE_SET_PACKET_IN_FORMAT:
1147
410
        return ofp_print_nxt_set_packet_in_format(string, oh);
1148
1149
0
    case OFPTYPE_FLOW_AGE:
1150
0
        break;
1151
1152
52
    case OFPTYPE_SET_CONTROLLER_ID:
1153
52
        return ofp_print_nxt_set_controller_id(string, ofpmsg_body(oh));
1154
1155
8.31k
    case OFPTYPE_GET_ASYNC_REPLY:
1156
10.6k
    case OFPTYPE_SET_ASYNC_CONFIG:
1157
10.6k
        return ofp_print_set_async_config(string, oh, type);
1158
0
    case OFPTYPE_GET_ASYNC_REQUEST:
1159
0
        break;
1160
64
    case OFPTYPE_FLOW_MONITOR_CANCEL:
1161
64
        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
8.92k
    case OFPTYPE_FLOW_MONITOR_STATS_REQUEST:
1168
8.92k
        return ofp_print_flow_monitor_request(string, msg, port_map,
1169
8.92k
                                              table_map);
1170
1171
9.33k
    case OFPTYPE_FLOW_MONITOR_STATS_REPLY:
1172
9.33k
        return ofp_print_flow_monitor_reply(string, msg, port_map,
1173
9.33k
                                            table_map);
1174
1175
1.04k
    case OFPTYPE_BUNDLE_CONTROL:
1176
1.04k
        return ofp_print_bundle_ctrl(string, msg);
1177
1178
3.49k
    case OFPTYPE_BUNDLE_ADD_MESSAGE:
1179
3.49k
        return ofp_print_bundle_add(string, msg, port_map, table_map,
1180
3.49k
                                    verbosity);
1181
1182
1.96k
    case OFPTYPE_NXT_TLV_TABLE_MOD:
1183
1.96k
        return ofp_print_tlv_table_mod(string, msg);
1184
1185
0
    case OFPTYPE_NXT_TLV_TABLE_REQUEST:
1186
0
        break;
1187
1188
2.16k
    case OFPTYPE_NXT_TLV_TABLE_REPLY:
1189
2.16k
        return ofp_print_tlv_table_reply(string, msg);
1190
1191
3.46k
    case OFPTYPE_NXT_RESUME:
1192
3.46k
        return ofp_print_packet_in(string, msg, port_map, table_map,
1193
3.46k
                                   verbosity);
1194
0
    case OFPTYPE_IPFIX_BRIDGE_STATS_REQUEST:
1195
0
        break;
1196
413
    case OFPTYPE_IPFIX_BRIDGE_STATS_REPLY:
1197
413
        return ofp_print_nxst_ipfix_bridge_reply(string, oh);
1198
0
    case OFPTYPE_IPFIX_FLOW_STATS_REQUEST:
1199
0
        break;
1200
1.25k
    case OFPTYPE_IPFIX_FLOW_STATS_REPLY:
1201
1.25k
        return ofp_print_nxst_ipfix_flow_reply(string, oh);
1202
1203
34
    case OFPTYPE_CT_FLUSH_ZONE:
1204
34
        return ofp_print_nxt_ct_flush_zone(string, ofpmsg_body(oh));
1205
15.7k
    case OFPTYPE_CT_FLUSH:
1206
15.7k
        return ofp_print_nxt_ct_flush(string, oh);
1207
1.20M
    }
1208
1209
3.29k
    return 0;
1210
1.20M
}
1211
1212
static void
1213
add_newline(struct ds *s)
1214
2.90M
{
1215
2.90M
    if (s->length && s->string[s->length - 1] != '\n') {
1216
157k
        ds_put_char(s, '\n');
1217
157k
    }
1218
2.90M
}
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
2.13M
{
1230
2.13M
    struct ds string = DS_EMPTY_INITIALIZER;
1231
2.13M
    const struct ofp_header *oh = oh_;
1232
1233
2.13M
    if (!len) {
1234
331k
        ds_put_cstr(&string, "OpenFlow message is empty\n");
1235
1.80M
    } else if (len < sizeof(struct ofp_header)) {
1236
56.2k
        ds_put_format(&string, "OpenFlow packet too short (only %"PRIuSIZE" bytes):\n",
1237
56.2k
                      len);
1238
1.75M
    } else if (ntohs(oh->length) > len) {
1239
9.79k
        enum ofperr error;
1240
9.79k
        enum ofpraw raw;
1241
1242
9.79k
        error = ofpraw_decode_partial(&raw, oh, len);
1243
9.79k
        if (!error) {
1244
4.39k
            ofp_header_to_string__(oh, raw, &string);
1245
4.39k
            ds_put_char(&string, '\n');
1246
4.39k
        }
1247
1248
9.79k
        ds_put_format(&string,
1249
9.79k
                      "(***truncated to %"PRIuSIZE" bytes from %"PRIu16"***)\n",
1250
9.79k
                      len, ntohs(oh->length));
1251
1.74M
    } else if (ntohs(oh->length) < len) {
1252
521
        ds_put_format(&string,
1253
521
                      "(***only uses %"PRIu16" bytes out of %"PRIuSIZE"***)\n",
1254
521
                      ntohs(oh->length), len);
1255
1.73M
    } else {
1256
1.73M
        enum ofperr error;
1257
1.73M
        enum ofpraw raw;
1258
1259
1.73M
        error = ofpraw_decode(&raw, oh);
1260
1.73M
        if (!error) {
1261
1.20M
            ofp_header_to_string__(oh, raw, &string);
1262
1.20M
            size_t header_len = string.length;
1263
1264
1.20M
            error = ofp_to_string__(oh, port_map, table_map,
1265
1.20M
                                    raw, &string, verbosity);
1266
1.20M
            if (error) {
1267
564k
                if (string.length > header_len) {
1268
65.9k
                    ds_chomp(&string, ' ');
1269
65.9k
                    add_newline(&string);
1270
498k
                } else {
1271
498k
                    ds_put_char(&string, ' ');
1272
498k
                }
1273
564k
                ofp_print_error(&string, error);
1274
636k
            } else {
1275
636k
                ds_chomp(&string, ' ');
1276
636k
            }
1277
1.20M
        } else {
1278
538k
            ofp_print_error(&string, error);
1279
538k
        }
1280
1281
1.73M
        if (verbosity >= 5 || error) {
1282
1.10M
            add_newline(&string);
1283
1.10M
            ds_put_hex_dump(&string, oh, len, 0, true);
1284
1.10M
        }
1285
1286
1.73M
        add_newline(&string);
1287
1.73M
        return ds_steal_cstr(&string);
1288
1.73M
    }
1289
398k
    ds_put_hex_dump(&string, oh, len, 0, true);
1290
398k
    return ds_steal_cstr(&string);
1291
2.13M
}
1292

1293
static void
1294
print_and_free(FILE *stream, char *string)
1295
1.74M
{
1296
1.74M
    fputs(string, stream);
1297
1.74M
    free(string);
1298
1.74M
}
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.73M
{
1308
1.73M
    print_and_free(stream, ofp_to_string(oh, len, port_map, table_map,
1309
1.73M
                                         verbosity));
1310
1.73M
}
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.24k
{
1318
2.24k
    print_and_free(stream, ofp_packet_to_string(data, len, packet_type));
1319
2.24k
}
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
}