blob: 80ef08d04a1cd529a9242083cd074ee55a28129d [file] [log] [blame]
Damjan Marion22311502016-10-28 20:30:15 +02001/*
2 * Copyright (c) 2016 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#include <vnet/feature/feature.h>
17
18vnet_feature_main_t feature_main;
19
20static clib_error_t *
21vnet_feature_init (vlib_main_t * vm)
22{
23 vnet_feature_main_t *fm = &feature_main;
24 vnet_feature_registration_t *freg;
25 vnet_feature_arc_registration_t *areg;
26 u32 arc_index = 0;
27
28 fm->arc_index_by_name = hash_create_string (0, sizeof (uword));
29 areg = fm->next_arc;
30
31 /* process feature arc registrations */
32 while (areg)
33 {
34 char *s;
35 int i = 0;
36 areg->feature_arc_index = arc_index;
Damjan Marion8b3191e2016-11-09 19:54:20 +010037 if (areg->arc_index_ptr)
38 *areg->arc_index_ptr = arc_index;
Damjan Marion22311502016-10-28 20:30:15 +020039 hash_set_mem (fm->arc_index_by_name, areg->arc_name,
40 pointer_to_uword (areg));
41
42 /* process start nodes */
43 while ((s = areg->start_nodes[i]))
44 {
Damjan Marion22311502016-10-28 20:30:15 +020045 i++;
46 }
47 areg->n_start_nodes = i;
48
49 /* next */
50 areg = areg->next;
51 arc_index++;
52 }
53
54 vec_validate (fm->next_feature_by_arc, arc_index - 1);
55 vec_validate (fm->feature_nodes, arc_index - 1);
56 vec_validate (fm->feature_config_mains, arc_index - 1);
57 vec_validate (fm->next_feature_by_name, arc_index - 1);
58 vec_validate (fm->sw_if_index_has_features, arc_index - 1);
59 vec_validate (fm->feature_count_by_sw_if_index, arc_index - 1);
60
61 freg = fm->next_feature;
62 while (freg)
63 {
Damjan Marion96b41f72016-11-10 18:01:42 +010064 vnet_feature_registration_t *next;
Damjan Marion22311502016-10-28 20:30:15 +020065 uword *p = hash_get_mem (fm->arc_index_by_name, freg->arc_name);
66 if (p == 0)
67 return clib_error_return (0, "Unknown feature arc '%s'",
68 freg->arc_name);
69
70 areg = uword_to_pointer (p[0], vnet_feature_arc_registration_t *);
71 arc_index = areg->feature_arc_index;
72
Damjan Marion96b41f72016-11-10 18:01:42 +010073 next = freg->next;
74 freg->next = fm->next_feature_by_arc[arc_index];
75 fm->next_feature_by_arc[arc_index] = freg;
Damjan Marion22311502016-10-28 20:30:15 +020076
77 /* next */
Damjan Marion96b41f72016-11-10 18:01:42 +010078 freg = next;
Damjan Marion22311502016-10-28 20:30:15 +020079 }
80
Damjan Marion96b41f72016-11-10 18:01:42 +010081 areg = fm->next_arc;
Damjan Marion22311502016-10-28 20:30:15 +020082 while (areg)
83 {
84 clib_error_t *error;
85 vnet_feature_config_main_t *cm;
86 vnet_config_main_t *vcm;
87
88 arc_index = areg->feature_arc_index;
89 cm = &fm->feature_config_mains[arc_index];
90 vcm = &cm->config_main;
91 if ((error = vnet_feature_arc_init (vm, vcm,
92 areg->start_nodes,
93 areg->n_start_nodes,
94 fm->next_feature_by_arc[arc_index],
95 &fm->feature_nodes[arc_index])))
96 {
97 return error;
98 }
99
100 fm->next_feature_by_name[arc_index] =
101 hash_create_string (0, sizeof (uword));
102 freg = fm->next_feature_by_arc[arc_index];
103
104 while (freg)
105 {
106 hash_set_mem (fm->next_feature_by_name[arc_index],
107 freg->node_name, pointer_to_uword (freg));
108 freg = freg->next;
109 }
110
Damjan Marion21da6ce2016-11-28 18:21:59 +0100111 cm->end_feature_index =
112 vnet_get_feature_index (arc_index, areg->end_node);
113
Damjan Marion22311502016-10-28 20:30:15 +0200114 /* next */
115 areg = areg->next;
116 arc_index++;
117 }
118
119 return 0;
120}
121
122VLIB_INIT_FUNCTION (vnet_feature_init);
123
Damjan Marion87cd1192016-11-04 11:00:27 +0100124u8
125vnet_get_feature_arc_index (const char *s)
Damjan Marion22311502016-10-28 20:30:15 +0200126{
127 vnet_feature_main_t *fm = &feature_main;
128 vnet_feature_arc_registration_t *reg;
129 uword *p;
130
131 p = hash_get_mem (fm->arc_index_by_name, s);
132 if (p == 0)
133 return ~0;
134
135 reg = uword_to_pointer (p[0], vnet_feature_arc_registration_t *);
136 return reg->feature_arc_index;
137}
138
Pavel Kotucek7490a752016-11-15 09:19:11 +0100139vnet_feature_registration_t *
140vnet_get_feature_reg (const char *arc_name, const char *node_name)
141{
142 u8 arc_index;
143
144 arc_index = vnet_get_feature_arc_index (arc_name);
145 if (arc_index == (u8) ~ 0)
146 return 0;
147
148 vnet_feature_main_t *fm = &feature_main;
149 vnet_feature_registration_t *reg;
150 uword *p;
151
152 p = hash_get_mem (fm->next_feature_by_name[arc_index], node_name);
153 if (p == 0)
154 return 0;
155
156 reg = uword_to_pointer (p[0], vnet_feature_registration_t *);
157 return reg;
158}
159
Damjan Marion22311502016-10-28 20:30:15 +0200160u32
Damjan Marion87cd1192016-11-04 11:00:27 +0100161vnet_get_feature_index (u8 arc, const char *s)
Damjan Marion22311502016-10-28 20:30:15 +0200162{
163 vnet_feature_main_t *fm = &feature_main;
164 vnet_feature_registration_t *reg;
165 uword *p;
166
Damjan Marion21da6ce2016-11-28 18:21:59 +0100167 if (s == 0)
168 return ~0;
169
Damjan Marion22311502016-10-28 20:30:15 +0200170 p = hash_get_mem (fm->next_feature_by_name[arc], s);
171 if (p == 0)
172 return ~0;
173
174 reg = uword_to_pointer (p[0], vnet_feature_registration_t *);
Damjan Marion8b3191e2016-11-09 19:54:20 +0100175 return reg->feature_index;
Damjan Marion22311502016-10-28 20:30:15 +0200176}
177
Damjan Marion96b41f72016-11-10 18:01:42 +0100178int
Damjan Marion8b3191e2016-11-09 19:54:20 +0100179vnet_feature_enable_disable_with_index (u8 arc_index, u32 feature_index,
180 u32 sw_if_index, int enable_disable,
181 void *feature_config,
182 u32 n_feature_config_bytes)
Damjan Marion22311502016-10-28 20:30:15 +0200183{
184 vnet_feature_main_t *fm = &feature_main;
185 vnet_feature_config_main_t *cm;
Damjan Marion21da6ce2016-11-28 18:21:59 +0100186 i16 feature_count;
187 int is_first_or_last;
Damjan Marion8b3191e2016-11-09 19:54:20 +0100188 u32 ci;
Damjan Marion22311502016-10-28 20:30:15 +0200189
Damjan Marion05bb1dd2016-11-08 21:28:22 +0100190 if (arc_index == (u8) ~ 0)
Damjan Marion96b41f72016-11-10 18:01:42 +0100191 return VNET_API_ERROR_INVALID_VALUE;
Damjan Marion22311502016-10-28 20:30:15 +0200192
Damjan Marion22311502016-10-28 20:30:15 +0200193 if (feature_index == ~0)
Damjan Marion96b41f72016-11-10 18:01:42 +0100194 return VNET_API_ERROR_INVALID_VALUE_2;
Damjan Marion8b3191e2016-11-09 19:54:20 +0100195
196 cm = &fm->feature_config_mains[arc_index];
197 vec_validate_init_empty (cm->config_index_by_sw_if_index, sw_if_index, ~0);
Damjan Marion22311502016-10-28 20:30:15 +0200198 ci = cm->config_index_by_sw_if_index[sw_if_index];
199
Pavel Kotucekf6e3dc42016-11-04 09:58:01 +0100200 vec_validate (fm->feature_count_by_sw_if_index[arc_index], sw_if_index);
Damjan Marion21da6ce2016-11-28 18:21:59 +0100201 feature_count = fm->feature_count_by_sw_if_index[arc_index][sw_if_index];
202
203 if (!enable_disable && feature_count < 1)
Pavel Kotucekf6e3dc42016-11-04 09:58:01 +0100204 return 0;
205
Damjan Marion22311502016-10-28 20:30:15 +0200206 ci = (enable_disable
207 ? vnet_config_add_feature
208 : vnet_config_del_feature)
209 (vlib_get_main (), &cm->config_main, ci, feature_index, feature_config,
210 n_feature_config_bytes);
211 cm->config_index_by_sw_if_index[sw_if_index] = ci;
212
Damjan Marion21da6ce2016-11-28 18:21:59 +0100213 /* update feature count */
214 enable_disable = (enable_disable > 0);
215 feature_count += enable_disable ? 1 : -1;
216 is_first_or_last = (feature_count == enable_disable);
217 ASSERT (feature_count >= 0);
Damjan Marion22311502016-10-28 20:30:15 +0200218
Damjan Marion21da6ce2016-11-28 18:21:59 +0100219 if (is_first_or_last && cm->end_feature_index != ~0)
220 {
221 /*register end node */
222 ci = (enable_disable
223 ? vnet_config_add_feature
224 : vnet_config_del_feature)
225 (vlib_get_main (), &cm->config_main, ci, cm->end_feature_index, 0, 0);
226 cm->config_index_by_sw_if_index[sw_if_index] = ci;
227 }
228
229 fm->sw_if_index_has_features[arc_index] =
230 clib_bitmap_set (fm->sw_if_index_has_features[arc_index], sw_if_index,
231 (feature_count > 0));
232
233 fm->feature_count_by_sw_if_index[arc_index][sw_if_index] = feature_count;
Damjan Marion96b41f72016-11-10 18:01:42 +0100234 return 0;
Damjan Marion22311502016-10-28 20:30:15 +0200235}
236
Damjan Marion8b3191e2016-11-09 19:54:20 +0100237int
238vnet_feature_enable_disable (const char *arc_name, const char *node_name,
239 u32 sw_if_index, int enable_disable,
240 void *feature_config, u32 n_feature_config_bytes)
241{
242 u32 feature_index;
243 u8 arc_index;
244
245 arc_index = vnet_get_feature_arc_index (arc_name);
246
247 if (arc_index == (u8) ~ 0)
248 return VNET_API_ERROR_INVALID_VALUE;
249
250 feature_index = vnet_get_feature_index (arc_index, node_name);
251
252 return vnet_feature_enable_disable_with_index (arc_index, feature_index,
253 sw_if_index, enable_disable,
254 feature_config,
255 n_feature_config_bytes);
256}
257
Damjan Marion22311502016-10-28 20:30:15 +0200258
259/** Display the set of available driver features.
260 Useful for verifying that expected features are present
261*/
262
263static clib_error_t *
264show_features_command_fn (vlib_main_t * vm,
265 unformat_input_t * input, vlib_cli_command_t * cmd)
266{
267 vnet_feature_main_t *fm = &feature_main;
268 vnet_feature_arc_registration_t *areg;
269 vnet_feature_registration_t *freg;
270
271 vlib_cli_output (vm, "Available feature paths");
272
273 areg = fm->next_arc;
274 while (areg)
275 {
276 vlib_cli_output (vm, "%s:", areg->arc_name);
Damjan Marion96b41f72016-11-10 18:01:42 +0100277 freg = fm->next_feature_by_arc[areg->feature_arc_index];
278 while (freg)
279 {
280 vlib_cli_output (vm, " %s\n", freg->node_name);
281 freg = freg->next;
282 }
Damjan Marion22311502016-10-28 20:30:15 +0200283
284
285 /* next */
286 areg = areg->next;
287 }
288
289 return 0;
290}
291
292/*?
293 * Display the set of available driver features
294 *
295 * @cliexpar
296 * Example:
297 * @cliexcmd{show ip features}
298 * @cliexend
299 * @endparblock
300?*/
301/* *INDENT-OFF* */
302VLIB_CLI_COMMAND (show_features_command, static) = {
303 .path = "show features",
304 .short_help = "show features",
305 .function = show_features_command_fn,
306};
307/* *INDENT-ON* */
308
309/** Display the set of driver features configured on a specific interface
310 * Called by "show interface" handler
311 */
312
313void
314vnet_interface_features_show (vlib_main_t * vm, u32 sw_if_index)
315{
316 vnet_feature_main_t *fm = &feature_main;
317 u32 node_index, current_config_index;
318 u16 feature_arc;
319 vnet_feature_config_main_t *cm = fm->feature_config_mains;
320 vnet_feature_arc_registration_t *areg;
321 vnet_config_main_t *vcm;
322 vnet_config_t *cfg;
323 u32 cfg_index;
324 vnet_config_feature_t *feat;
325 vlib_node_t *n;
326 int i;
327
328 vlib_cli_output (vm, "Driver feature paths configured on %U...",
329 format_vnet_sw_if_index_name,
330 vnet_get_main (), sw_if_index);
331
332 areg = fm->next_arc;
333 while (areg)
334 {
335 feature_arc = areg->feature_arc_index;
336 vcm = &(cm[feature_arc].config_main);
337
338 vlib_cli_output (vm, "\n%s:", areg->arc_name);
339 areg = areg->next;
340
341 if (NULL == cm[feature_arc].config_index_by_sw_if_index ||
Damjan Marion87cd1192016-11-04 11:00:27 +0100342 vec_len (cm[feature_arc].config_index_by_sw_if_index) <=
343 sw_if_index)
Damjan Marion22311502016-10-28 20:30:15 +0200344 {
345 vlib_cli_output (vm, " none configured");
346 continue;
347 }
348
349 current_config_index =
350 vec_elt (cm[feature_arc].config_index_by_sw_if_index, sw_if_index);
351
352 if (current_config_index == ~0)
353 {
354 vlib_cli_output (vm, " none configured");
355 continue;
356 }
357
358 ASSERT (current_config_index
359 < vec_len (vcm->config_pool_index_by_user_index));
360
361 cfg_index = vcm->config_pool_index_by_user_index[current_config_index];
362 cfg = pool_elt_at_index (vcm->config_pool, cfg_index);
363
364 for (i = 0; i < vec_len (cfg->features); i++)
365 {
366 feat = cfg->features + i;
367 node_index = feat->node_index;
368 n = vlib_get_node (vm, node_index);
369 vlib_cli_output (vm, " %v", n->name);
370 }
371 }
372}
373
Pavel Kotucek7490a752016-11-15 09:19:11 +0100374static clib_error_t *
375set_interface_features_command_fn (vlib_main_t * vm,
376 unformat_input_t * input,
377 vlib_cli_command_t * cmd)
378{
379 vnet_main_t *vnm = vnet_get_main ();
380 unformat_input_t _line_input, *line_input = &_line_input;
381 clib_error_t *error = 0;
382
383 u8 *arc_name = 0;
384 u8 *feature_name = 0;
385 u32 sw_if_index = ~0;
386 u8 enable = 1;
387
388 /* Get a line of input. */
389 if (!unformat_user (input, unformat_line_input, line_input))
390 goto done;
391
392 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
393 {
394 if (unformat
395 (line_input, "%U %v", unformat_vnet_sw_interface, vnm, &sw_if_index,
396 &feature_name))
397 ;
398 else if (unformat (line_input, "arc %v", &arc_name))
399 ;
400 else if (unformat (line_input, "disable"))
401 enable = 0;
402 else
403 {
404 error = unformat_parse_error (line_input);
405 goto done;
406 }
407 }
408
409 if (sw_if_index == ~0)
410 {
411 error = clib_error_return (0, "Interface not specified...");
412 goto done;
413 }
414
415 vec_add1 (arc_name, 0);
416 vec_add1 (feature_name, 0);
417
418 vnet_feature_registration_t *reg;
419 reg =
420 vnet_get_feature_reg ((const char *) arc_name,
421 (const char *) feature_name);
422 if (reg == 0)
423 {
424 error = clib_error_return (0, "Unknown feature...");
425 goto done;
426 }
427 if (reg->enable_disable_cb)
428 error = reg->enable_disable_cb (sw_if_index, enable);
429 if (!error)
430 vnet_feature_enable_disable ((const char *) arc_name,
431 (const char *) feature_name, sw_if_index,
432 enable, 0, 0);
433
434done:
435 vec_free (feature_name);
436 vec_free (arc_name);
Billy McFall614c1312017-03-01 17:01:06 -0500437 unformat_free (line_input);
Pavel Kotucek7490a752016-11-15 09:19:11 +0100438 return error;
439}
440
441/*?
442 * Set feature for given interface
443 *
444 * @cliexpar
445 * Example:
446 * @cliexcmd{set interface feature GigabitEthernet2/0/0 ip4_flow_classify arc ip4_unicast}
447 * @cliexend
448 * @endparblock
449?*/
450/* *INDENT-OFF* */
451VLIB_CLI_COMMAND (set_interface_feature_command, static) = {
452 .path = "set interface feature",
453 .short_help = "set interface feature <intfc> <feature_name> arc <arc_name>",
454 .function = set_interface_features_command_fn,
455};
456/* *INDENT-ON* */
457
Damjan Marion22311502016-10-28 20:30:15 +0200458/*
459 * fd.io coding-style-patch-verification: ON
460 *
461 * Local Variables:
462 * eval: (c-set-style "gnu")
463 * End:
464 */