blob: 0a624204e342f1bef05608930d7f125b4ccf1b03 [file] [log] [blame]
Juraj Sloboda506b2452016-08-07 23:45:24 -07001/*
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#include <vnet/classify/flow_classify.h>
16
17static void
18vnet_flow_classify_feature_enable (vlib_main_t * vnm,
19 flow_classify_main_t * fcm,
20 u32 sw_if_index,
21 flow_classify_table_id_t tid,
22 int feature_enable)
23{
Damjan Marion8b3191e2016-11-09 19:54:20 +010024 vnet_feature_config_main_t *vfcm;
25 u8 arc;
Juraj Sloboda506b2452016-08-07 23:45:24 -070026
27 if (tid == FLOW_CLASSIFY_TABLE_IP4)
Damjan Marion8b3191e2016-11-09 19:54:20 +010028 {
29 vnet_feature_enable_disable ("ip4-unicast", "ip4-flow-classify",
30 sw_if_index, feature_enable, 0, 0);
31 arc = vnet_get_feature_arc_index ("ip4-unicast");
32 }
Juraj Sloboda506b2452016-08-07 23:45:24 -070033 else
Damjan Marion8b3191e2016-11-09 19:54:20 +010034 {
35 vnet_feature_enable_disable ("ip6-unicast", "ip6-flow-classify",
36 sw_if_index, feature_enable, 0, 0);
37 arc = vnet_get_feature_arc_index ("ip6-unicast");
38 }
Juraj Sloboda506b2452016-08-07 23:45:24 -070039
Damjan Marion8b3191e2016-11-09 19:54:20 +010040 vfcm = vnet_get_feature_arc_config_main (arc);
41 fcm->vnet_config_main[tid] = &vfcm->config_main;
Juraj Sloboda506b2452016-08-07 23:45:24 -070042}
43
44int vnet_set_flow_classify_intfc (vlib_main_t * vm, u32 sw_if_index,
45 u32 ip4_table_index, u32 ip6_table_index,
46 u32 is_add)
47{
48 flow_classify_main_t * fcm = &flow_classify_main;
49 vnet_classify_main_t * vcm = fcm->vnet_classify_main;
50 u32 pct[FLOW_CLASSIFY_N_TABLES] = {ip4_table_index, ip6_table_index};
51 u32 ti;
52
53 /* Assume that we've validated sw_if_index in the API layer */
54
55 for (ti = 0; ti < FLOW_CLASSIFY_N_TABLES; ti++)
56 {
57 if (pct[ti] == ~0)
58 continue;
59
60 if (pool_is_free_index (vcm->tables, pct[ti]))
61 return VNET_API_ERROR_NO_SUCH_TABLE;
62
63 vec_validate_init_empty
64 (fcm->classify_table_index_by_sw_if_index[ti], sw_if_index, ~0);
65
66 /* Reject any DEL operation with wrong sw_if_index */
67 if (!is_add &&
68 (pct[ti] != fcm->classify_table_index_by_sw_if_index[ti][sw_if_index]))
69 {
70 clib_warning ("Non-existent intf_idx=%d with table_index=%d for delete",
71 sw_if_index, pct[ti]);
72 return VNET_API_ERROR_NO_SUCH_TABLE;
73 }
74
75 /* Return ok on ADD operaton if feature is already enabled */
76 if (is_add &&
77 fcm->classify_table_index_by_sw_if_index[ti][sw_if_index] != ~0)
78 return 0;
79
80 vnet_flow_classify_feature_enable (vm, fcm, sw_if_index, ti, is_add);
81
82 if (is_add)
83 fcm->classify_table_index_by_sw_if_index[ti][sw_if_index] = pct[ti];
84 else
85 fcm->classify_table_index_by_sw_if_index[ti][sw_if_index] = ~0;
86 }
87
88
89 return 0;
90}
91
92static clib_error_t *
93set_flow_classify_command_fn (vlib_main_t * vm,
94 unformat_input_t * input,
95 vlib_cli_command_t * cmd)
96{
97 vnet_main_t * vnm = vnet_get_main();
98 u32 sw_if_index = ~0;
99 u32 ip4_table_index = ~0;
100 u32 ip6_table_index = ~0;
101 u32 is_add = 1;
102 u32 idx_cnt = 0;
103 int rv;
104
105 while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
106 {
107 if (unformat (input, "interface %U", unformat_vnet_sw_interface,
108 vnm, &sw_if_index))
109 ;
110 else if (unformat (input, "ip4-table %d", &ip4_table_index))
111 idx_cnt++;
112 else if (unformat (input, "ip6-table %d", &ip6_table_index))
113 idx_cnt++;
114 else if (unformat (input, "del"))
115 is_add = 0;
116 else
117 break;
118 }
119
120 if (sw_if_index == ~0)
121 return clib_error_return (0, "Interface must be specified.");
122
123 if (!idx_cnt)
124 return clib_error_return (0, "Table index should be specified.");
125
126 if (idx_cnt > 1)
127 return clib_error_return (0, "Only one table index per API is allowed.");
128
129 rv = vnet_set_flow_classify_intfc(vm, sw_if_index, ip4_table_index,
130 ip6_table_index, is_add);
131
132 switch (rv)
133 {
134 case 0:
135 break;
136
137 case VNET_API_ERROR_NO_MATCHING_INTERFACE:
138 return clib_error_return (0, "No such interface");
139
140 case VNET_API_ERROR_NO_SUCH_ENTRY:
141 return clib_error_return (0, "No such classifier table");
142 }
143 return 0;
144}
145
146VLIB_CLI_COMMAND (set_input_acl_command, static) = {
147 .path = "set flow classify",
148 .short_help =
149 "set flow classify interface <int> [ip4-table <index>]\n"
150 " [ip6-table <index>] [del]",
151 .function = set_flow_classify_command_fn,
152};
153
154static uword
155unformat_table_type (unformat_input_t * input, va_list * va)
156{
157 u32 * r = va_arg (*va, u32 *);
158 u32 tid;
159
160 if (unformat (input, "ip4"))
161 tid = FLOW_CLASSIFY_TABLE_IP4;
162 else if (unformat (input, "ip6"))
163 tid = FLOW_CLASSIFY_TABLE_IP6;
164 else
165 return 0;
166
167 *r = tid;
168 return 1;
169}
170static clib_error_t *
171show_flow_classify_command_fn (vlib_main_t * vm,
172 unformat_input_t * input,
173 vlib_cli_command_t * cmd)
174{
175 flow_classify_main_t * fcm = &flow_classify_main;
176 u32 type = FLOW_CLASSIFY_N_TABLES;
177 u32 * vec_tbl;
178 int i;
179
180 if (unformat (input, "type %U", unformat_table_type, &type))
181 ;
182 else
183 return clib_error_return (0, "Type must be specified.");;
184
185 if (type == FLOW_CLASSIFY_N_TABLES)
186 return clib_error_return (0, "Invalid table type.");
187
188 vec_tbl = fcm->classify_table_index_by_sw_if_index[type];
189
190 if (vec_len(vec_tbl))
191 vlib_cli_output (vm, "%10s%20s\t\t%s", "Intfc idx", "Classify table",
192 "Interface name");
193 else
194 vlib_cli_output (vm, "No tables configured.");
195
196 for (i = 0; i < vec_len (vec_tbl); i++)
197 {
198 if (vec_elt(vec_tbl, i) == ~0)
199 continue;
200
201 vlib_cli_output (vm, "%10d%20d\t\t%U", i, vec_elt(vec_tbl, i),
202 format_vnet_sw_if_index_name, fcm->vnet_main, i);
203 }
204
205 return 0;
206}
207
208VLIB_CLI_COMMAND (show_flow_classify_command, static) = {
209 .path = "show classify flow",
210 .short_help = "show classify flow type [ip4|ip6]",
211 .function = show_flow_classify_command_fn,
212};