2 * Copyright (c) 2011, 2012, 2013, 2014, 2019 Nicira, Inc.
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:
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
21 #include "openvswitch/dynamic-string.h"
23 #include "openvswitch/match.h"
24 #include "openvswitch/ofpbuf.h"
27 #include "openvswitch/ofp-flow.h"
28 #include "openvswitch/vlog.h"
31 parse_keys(bool wc_keys
)
37 vlog_set_levels_from_string_assert("odp_util:console:dbg");
38 while (!ds_get_test_line(&in
, stdin
)) {
39 enum odp_key_fitness fitness
;
40 struct ofpbuf odp_key
;
41 struct ofpbuf odp_mask
;
46 /* Convert string to OVS DP key. */
47 ofpbuf_init(&odp_key
, 0);
48 ofpbuf_init(&odp_mask
, 0);
50 error
= odp_flow_from_string(ds_cstr(&in
), NULL
,
51 &odp_key
, &odp_mask
, &error_s
);
53 printf("odp_flow_from_string: error (%s)\n", error_s
);
59 struct odp_flow_key_parms odp_parms
= {
67 .max_vlan_headers
= SIZE_MAX
,
72 /* Convert odp_key to flow. */
73 fitness
= odp_flow_key_to_flow(odp_key
.data
, odp_key
.size
, &flow
,
79 case ODP_FIT_TOO_LITTLE
:
80 printf("ODP_FIT_TOO_LITTLE: ");
83 case ODP_FIT_TOO_MUCH
:
84 printf("ODP_FIT_TOO_MUCH: ");
88 printf("odp_flow_key_to_flow: error (%s)\n", error_s
);
92 /* Convert cls_rule back to odp_key. */
93 ofpbuf_uninit(&odp_key
);
94 ofpbuf_init(&odp_key
, 0);
95 odp_flow_key_from_flow(&odp_parms
, &odp_key
);
97 if (odp_key
.size
> ODPUTIL_FLOW_KEY_BYTES
) {
98 printf ("too long: %"PRIu32
" > %d\n",
99 odp_key
.size
, ODPUTIL_FLOW_KEY_BYTES
);
104 /* Convert odp_key to string. */
107 odp_flow_format(odp_key
.data
, odp_key
.size
,
108 odp_mask
.data
, odp_mask
.size
, NULL
, &out
, false);
110 odp_flow_key_format(odp_key
.data
, odp_key
.size
, &out
);
116 ofpbuf_uninit(&odp_key
);
117 ofpbuf_uninit(&odp_mask
);
130 vlog_set_levels_from_string_assert("odp_util:console:dbg");
131 while (!ds_get_test_line(&in
, stdin
)) {
132 struct ofpbuf odp_actions
;
136 /* Convert string to OVS DP actions. */
137 ofpbuf_init(&odp_actions
, 0);
138 error
= odp_actions_from_string(ds_cstr(&in
), NULL
, &odp_actions
);
140 printf("odp_actions_from_string: error\n");
144 /* Convert odp_actions back to string. */
146 format_odp_actions(&out
, odp_actions
.data
, odp_actions
.size
, NULL
);
151 ofpbuf_uninit(&odp_actions
);
159 parse_filter(char *filter_parse
)
162 struct flow flow_filter
;
163 struct flow_wildcards wc_filter
;
164 char *error
, *filter
= NULL
;
166 vlog_set_levels_from_string_assert("odp_util:console:dbg");
167 if (filter_parse
&& !strncmp(filter_parse
, "filter=", 7)) {
168 filter
= xstrdup(filter_parse
+ 7);
169 memset(&flow_filter
, 0, sizeof(flow_filter
));
170 memset(&wc_filter
, 0, sizeof(wc_filter
));
172 error
= parse_ofp_exact_flow(&flow_filter
, &wc_filter
, NULL
, filter
,
175 ovs_fatal(0, "Failed to parse filter (%s)", error
);
178 ovs_fatal(0, "No filter to parse.");
182 while (!ds_get_test_line(&in
, stdin
)) {
183 struct ofpbuf odp_key
;
184 struct ofpbuf odp_mask
;
187 /* Convert string to OVS DP key. */
188 ofpbuf_init(&odp_key
, 0);
189 ofpbuf_init(&odp_mask
, 0);
191 if (odp_flow_from_string(ds_cstr(&in
), NULL
, &odp_key
, &odp_mask
,
193 printf("odp_flow_from_string: error (%s)\n", error_s
);
200 struct flow_wildcards wc
;
201 struct match match
, match_filter
;
202 struct minimatch minimatch
;
204 odp_flow_key_to_flow(odp_key
.data
, odp_key
.size
, &flow
, NULL
);
205 odp_flow_key_to_mask(odp_mask
.data
, odp_mask
.size
, &wc
, &flow
,
207 match_init(&match
, &flow
, &wc
);
209 match_init(&match_filter
, &flow_filter
, &wc
);
210 match_init(&match_filter
, &match_filter
.flow
, &wc_filter
);
211 minimatch_init(&minimatch
, &match_filter
);
213 if (!minimatch_matches_flow(&minimatch
, &match
.flow
)) {
214 minimatch_destroy(&minimatch
);
217 minimatch_destroy(&minimatch
);
219 /* Convert odp_key to string. */
221 odp_flow_format(odp_key
.data
, odp_key
.size
,
222 odp_mask
.data
, odp_mask
.size
, NULL
, &out
, false);
227 ofpbuf_uninit(&odp_key
);
228 ofpbuf_uninit(&odp_mask
);
237 test_odp_main(int argc
, char *argv
[])
241 set_program_name(argv
[0]);
242 if (argc
== 2 &&!strcmp(argv
[1], "parse-keys")) {
243 exit_code
=parse_keys(false);
244 } else if (argc
== 2 &&!strcmp(argv
[1], "parse-wc-keys")) {
245 exit_code
=parse_keys(true);
246 } else if (argc
== 2 && !strcmp(argv
[1], "parse-actions")) {
247 exit_code
= parse_actions();
248 } else if (argc
== 3 && !strcmp(argv
[1], "parse-filter")) {
249 exit_code
=parse_filter(argv
[2]);
251 ovs_fatal(0, "usage: %s parse-keys | parse-wc-keys | parse-actions", argv
[0]);
257 OVSTEST_REGISTER("test-odp", test_odp_main
);