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:
7 * http://www.apache.org/licenses/LICENSE-2.0
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.
16 #include <vnet/feature/feature.h>
18 vnet_feature_main_t feature_main;
21 vnet_feature_init (vlib_main_t * vm)
23 vnet_feature_main_t *fm = &feature_main;
24 vnet_feature_registration_t *freg;
25 vnet_feature_arc_registration_t *areg;
28 fm->arc_index_by_name = hash_create_string (0, sizeof (uword));
31 /* process feature arc registrations */
36 areg->feature_arc_index = arc_index;
37 hash_set_mem (fm->arc_index_by_name, areg->arc_name,
38 pointer_to_uword (areg));
40 /* process start nodes */
41 while ((s = areg->start_nodes[i]))
44 vlib_node_runtime_t *rt;
45 n = vlib_get_node_by_name (vm, (u8 *) s);
48 return clib_error_return (0,
49 "Unknown start node '%s' on feature arc '%s'",
52 rt = vlib_node_get_runtime (vm, n->index);
53 rt->feature_arc_index = arc_index;
56 areg->n_start_nodes = i;
63 vec_validate (fm->next_feature_by_arc, arc_index - 1);
64 vec_validate (fm->feature_nodes, arc_index - 1);
65 vec_validate (fm->feature_config_mains, arc_index - 1);
66 vec_validate (fm->next_feature_by_name, arc_index - 1);
67 vec_validate (fm->sw_if_index_has_features, arc_index - 1);
68 vec_validate (fm->feature_count_by_sw_if_index, arc_index - 1);
70 freg = fm->next_feature;
74 vlib_node_runtime_t *rt;
75 uword *p = hash_get_mem (fm->arc_index_by_name, freg->arc_name);
77 return clib_error_return (0, "Unknown feature arc '%s'",
80 areg = uword_to_pointer (p[0], vnet_feature_arc_registration_t *);
81 arc_index = areg->feature_arc_index;
83 /* set feature arc index in node runtime */
84 n = vlib_get_node_by_name (vm, (u8 *) freg->node_name);
86 return clib_error_return (0, "Unknown node '%s', freg->node_name");
87 rt = vlib_node_get_runtime (vm, n->index);
88 rt->feature_arc_index = arc_index;
90 vec_add1 (fm->next_feature_by_arc[arc_index], *freg);
99 vnet_feature_config_main_t *cm;
100 vnet_config_main_t *vcm;
102 arc_index = areg->feature_arc_index;
103 cm = &fm->feature_config_mains[arc_index];
104 vcm = &cm->config_main;
105 if ((error = vnet_feature_arc_init (vm, vcm,
108 fm->next_feature_by_arc[arc_index],
109 &fm->feature_nodes[arc_index])))
114 fm->next_feature_by_name[arc_index] =
115 hash_create_string (0, sizeof (uword));
116 freg = fm->next_feature_by_arc[arc_index];
120 hash_set_mem (fm->next_feature_by_name[arc_index],
121 freg->node_name, pointer_to_uword (freg));
133 VLIB_INIT_FUNCTION (vnet_feature_init);
136 vnet_config_update_feature_count (vnet_feature_main_t * fm, u16 arc,
137 u32 sw_if_index, int is_add)
141 vec_validate (fm->feature_count_by_sw_if_index[arc], sw_if_index);
143 fm->feature_count_by_sw_if_index[arc][sw_if_index] += is_add ? 1 : -1;
145 ASSERT (fm->feature_count_by_sw_if_index[arc][sw_if_index] >= 0);
147 bit_value = fm->feature_count_by_sw_if_index[arc][sw_if_index] > 0;
149 fm->sw_if_index_has_features[arc] =
150 clib_bitmap_set (fm->sw_if_index_has_features[arc], sw_if_index,
155 vnet_feature_arc_index_from_node_name (const char *s)
157 vnet_feature_main_t *fm = &feature_main;
158 vnet_feature_arc_registration_t *reg;
161 p = hash_get_mem (fm->arc_index_by_name, s);
165 reg = uword_to_pointer (p[0], vnet_feature_arc_registration_t *);
166 return reg->feature_arc_index;
170 vnet_feature_index_from_node_name (u16 arc, const char *s)
172 vnet_feature_main_t *fm = &feature_main;
173 vnet_feature_registration_t *reg;
176 p = hash_get_mem (fm->next_feature_by_name[arc], s);
180 reg = uword_to_pointer (p[0], vnet_feature_registration_t *);
181 return reg->feature_index_u32;
185 vnet_feature_enable_disable (const char *arc_name, const char *node_name,
186 u32 sw_if_index, int enable_disable,
187 void *feature_config, u32 n_feature_config_bytes)
189 vnet_feature_main_t *fm = &feature_main;
190 vnet_feature_config_main_t *cm;
191 u32 feature_index, ci;
194 arc_index = vnet_feature_arc_index_from_node_name (arc_name);
199 cm = &fm->feature_config_mains[arc_index];
200 vec_validate_init_empty (cm->config_index_by_sw_if_index, sw_if_index, ~0);
201 feature_index = vnet_feature_index_from_node_name (arc_index, node_name);
202 if (feature_index == ~0)
204 ci = cm->config_index_by_sw_if_index[sw_if_index];
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;
213 vnet_config_update_feature_count (fm, arc_index, sw_if_index,
219 /** Display the set of available driver features.
220 Useful for verifying that expected features are present
223 static clib_error_t *
224 show_features_command_fn (vlib_main_t * vm,
225 unformat_input_t * input, vlib_cli_command_t * cmd)
227 vnet_feature_main_t *fm = &feature_main;
228 vnet_feature_arc_registration_t *areg;
229 vnet_feature_registration_t *freg;
231 vlib_cli_output (vm, "Available feature paths");
236 vlib_cli_output (vm, "%s:", areg->arc_name);
237 vec_foreach (freg, fm->next_feature_by_arc[areg->feature_arc_index])
239 vlib_cli_output (vm, " %s\n", freg->node_name);
251 * Display the set of available driver features
255 * @cliexcmd{show ip features}
260 VLIB_CLI_COMMAND (show_features_command, static) = {
261 .path = "show features",
262 .short_help = "show features",
263 .function = show_features_command_fn,
267 /** Display the set of driver features configured on a specific interface
268 * Called by "show interface" handler
272 vnet_interface_features_show (vlib_main_t * vm, u32 sw_if_index)
274 vnet_feature_main_t *fm = &feature_main;
275 u32 node_index, current_config_index;
277 vnet_feature_config_main_t *cm = fm->feature_config_mains;
278 vnet_feature_arc_registration_t *areg;
279 vnet_config_main_t *vcm;
282 vnet_config_feature_t *feat;
286 vlib_cli_output (vm, "Driver feature paths configured on %U...",
287 format_vnet_sw_if_index_name,
288 vnet_get_main (), sw_if_index);
293 feature_arc = areg->feature_arc_index;
294 vcm = &(cm[feature_arc].config_main);
296 vlib_cli_output (vm, "\n%s:", areg->arc_name);
299 if (NULL == cm[feature_arc].config_index_by_sw_if_index ||
300 vec_len (cm[feature_arc].config_index_by_sw_if_index) < sw_if_index)
302 vlib_cli_output (vm, " none configured");
306 current_config_index =
307 vec_elt (cm[feature_arc].config_index_by_sw_if_index, sw_if_index);
309 if (current_config_index == ~0)
311 vlib_cli_output (vm, " none configured");
315 ASSERT (current_config_index
316 < vec_len (vcm->config_pool_index_by_user_index));
318 cfg_index = vcm->config_pool_index_by_user_index[current_config_index];
319 cfg = pool_elt_at_index (vcm->config_pool, cfg_index);
321 for (i = 0; i < vec_len (cfg->features); i++)
323 feat = cfg->features + i;
324 node_index = feat->node_index;
325 n = vlib_get_node (vm, node_index);
326 vlib_cli_output (vm, " %v", n->name);
332 * fd.io coding-style-patch-verification: ON
335 * eval: (c-set-style "gnu")