Coverage Report

Created: 2025-07-18 06:07

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