blob: 3817c89a00904489d5c6bf378f3793fc22500ad6 [file] [log] [blame]
Neale Ranns039cbfe2018-02-27 03:45:38 -08001/*
2 * Copyright (c) 2018 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/ip/ip.h>
17#include <vnet/feature/feature.h>
18#include <vnet/qos/qos_egress_map.h>
19#include <vnet/qos/qos_mark.h>
20
21/**
22 * per-interface vector of which MAP is used by which interface
23 * for each output source
24 */
25index_t *qos_mark_configs[QOS_N_SOURCES];
26
27void
28qos_mark_ip_enable_disable (u32 sw_if_index, u8 enable)
29{
30 vnet_feature_enable_disable ("ip6-output", "ip6-qos-mark",
31 sw_if_index, enable, NULL, 0);
32 vnet_feature_enable_disable ("ip4-output", "ip4-qos-mark",
33 sw_if_index, enable, NULL, 0);
34}
35
36void
37qos_mark_vlan_enable_disable (u32 sw_if_index, u8 enable)
38{
Neale Ranns0809f6c2018-07-16 04:14:21 -070039 /*
40 * one cannot run a feature on a sub-interface, so we need
41 * to enable a feature on all the L3 output paths
42 */
43 vnet_feature_enable_disable ("ip6-output", "vlan-ip6-qos-mark",
44 sw_if_index, enable, NULL, 0);
45 vnet_feature_enable_disable ("ip4-output", "vlan-ip4-qos-mark",
46 sw_if_index, enable, NULL, 0);
47 vnet_feature_enable_disable ("mpls-output", "vlan-mpls-qos-mark",
Neale Ranns039cbfe2018-02-27 03:45:38 -080048 sw_if_index, enable, NULL, 0);
49}
50
51void
52qos_mark_mpls_enable_disable (u32 sw_if_index, u8 enable)
53{
54 vnet_feature_enable_disable ("mpls-output", "mpls-qos-mark",
55 sw_if_index, enable, NULL, 0);
56}
57
58static void
59qos_egress_map_feature_config (u32 sw_if_index, qos_source_t qs, u8 enable)
60{
61 switch (qs)
62 {
63 case QOS_SOURCE_EXT:
64 ASSERT (0);
65 break;
66 case QOS_SOURCE_VLAN:
67 qos_mark_vlan_enable_disable (sw_if_index, enable);
68 break;
69 case QOS_SOURCE_MPLS:
70 qos_mark_mpls_enable_disable (sw_if_index, enable);
71 break;
72 case QOS_SOURCE_IP:
73 qos_mark_ip_enable_disable (sw_if_index, enable);
74 break;
75 }
76}
77
Neale Ranns039cbfe2018-02-27 03:45:38 -080078int
79qos_mark_enable (u32 sw_if_index,
80 qos_source_t output_source, qos_egress_map_id_t mid)
81{
82 index_t qemi;
83
84 vec_validate_init_empty (qos_mark_configs[output_source],
85 sw_if_index, INDEX_INVALID);
86
87 qemi = qos_egress_map_find (mid);
88
89 if (INDEX_INVALID == qemi)
90 return VNET_API_ERROR_NO_SUCH_TABLE;
91
92 if (INDEX_INVALID == qos_mark_configs[output_source][sw_if_index])
93 {
94 qos_egress_map_feature_config (sw_if_index, output_source, 1);
95 }
96
97 qos_mark_configs[output_source][sw_if_index] = qemi;
98
99 return (0);
100}
101
102int
103qos_mark_disable (u32 sw_if_index, qos_source_t output_source)
104{
Eyal Bari07f35862018-07-18 17:15:26 +0300105 if (vec_len (qos_mark_configs[output_source]) <= sw_if_index)
Neale Ranns039cbfe2018-02-27 03:45:38 -0800106 return VNET_API_ERROR_NO_MATCHING_INTERFACE;
107 if (INDEX_INVALID == qos_mark_configs[output_source][sw_if_index])
108 return VNET_API_ERROR_VALUE_EXIST;
109
110 if (INDEX_INVALID != qos_mark_configs[output_source][sw_if_index])
111 {
112 qos_egress_map_feature_config (sw_if_index, output_source, 0);
113 }
114
115 qos_mark_configs[output_source][sw_if_index] = INDEX_INVALID;
116
117 return (0);
118}
119
Neale Ranns5281a902019-07-23 08:16:19 -0700120void
121qos_mark_walk (qos_mark_walk_cb_t fn, void *c)
122{
123 qos_source_t qs;
124
125 FOR_EACH_QOS_SOURCE (qs)
126 {
127 u32 sw_if_index;
128
129 vec_foreach_index (sw_if_index, qos_mark_configs[qs])
130 {
131 if (INDEX_INVALID != qos_mark_configs[qs][sw_if_index])
132 fn (sw_if_index,
133 qos_egress_map_get_id (qos_mark_configs[qs][sw_if_index]), qs, c);
134 }
135 }
136}
137
Neale Ranns039cbfe2018-02-27 03:45:38 -0800138static clib_error_t *
139qos_mark_cli (vlib_main_t * vm,
140 unformat_input_t * input, vlib_cli_command_t * cmd)
141{
142 qos_egress_map_id_t map_id;
143 u32 sw_if_index, qs;
144 vnet_main_t *vnm;
145 int rv, enable;
146
147 vnm = vnet_get_main ();
148 map_id = ~0;
149 qs = 0xff;
150 enable = 1;
151
152 while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
153 {
154 if (unformat (input, "id %d", &map_id))
155 ;
156 else if (unformat (input, "disable"))
157 enable = 0;
158 else if (unformat (input, "%U", unformat_qos_source, &qs))
159 ;
160 else if (unformat (input, "%U",
161 unformat_vnet_sw_interface, vnm, &sw_if_index))
162 ;
163 else
164 break;
165 }
166
167 if (~0 == sw_if_index)
168 return clib_error_return (0, "interface must be specified");
169 if (0xff == qs)
170 return clib_error_return (0, "output location must be specified");
171
172 if (enable)
173 rv = qos_mark_enable (sw_if_index, qs, map_id);
174 else
175 rv = qos_mark_disable (sw_if_index, qs);
176
177 if (0 == rv)
178 return (NULL);
179
180 return clib_error_return (0, "Failed to map interface");
181}
182
183/*?
184 * Apply a QoS egress mapping table to an interface for QoS marking packets
185 * at the given output protocol.
186 *
187 * @cliexpar
188 * @cliexcmd{qos egress interface GigEthernet0/9/0 id 0 output ip}
189 ?*/
Neale Ranns039cbfe2018-02-27 03:45:38 -0800190VLIB_CLI_COMMAND (qos_egress_map_interface_command, static) = {
191 .path = "qos mark",
192 .short_help = "qos mark <SOURCE> <INTERFACE> id <MAP>",
193 .function = qos_mark_cli,
194 .is_mp_safe = 1,
195};
Neale Ranns039cbfe2018-02-27 03:45:38 -0800196
Neale Ranns5281a902019-07-23 08:16:19 -0700197static void
198qos_mark_show_one_interface (vlib_main_t * vm, u32 sw_if_index)
199{
200 index_t qemis[QOS_N_SOURCES];
201 qos_source_t qs;
202 bool set;
203
204 set = false;
205 clib_memset_u32 (qemis, INDEX_INVALID, QOS_N_SOURCES);
206
207 FOR_EACH_QOS_SOURCE (qs)
208 {
209 if (vec_len (qos_mark_configs[qs]) <= sw_if_index)
210 continue;
211 if (INDEX_INVALID != (qemis[qs] = qos_mark_configs[qs][sw_if_index]))
212 set = true;
213 }
214
215 if (set)
216 {
217 vlib_cli_output (vm, " %U:", format_vnet_sw_if_index_name,
218 vnet_get_main (), sw_if_index);
219
220 FOR_EACH_QOS_SOURCE (qs)
221 {
222 if (qemis[qs] != INDEX_INVALID)
223 vlib_cli_output (vm, " %U: map:%d", format_qos_source, qs,
224 qemis[qs]);
225 }
226 }
227}
228
229static clib_error_t *
230qos_mark_show (vlib_main_t * vm,
231 unformat_input_t * input, vlib_cli_command_t * cmd)
232{
233 vnet_main_t *vnm = vnet_get_main ();
234 qos_source_t qs;
235 u32 sw_if_index;
236
237 sw_if_index = ~0;
238
239 while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
240 {
241 if (unformat (input, "%U", unformat_vnet_sw_interface,
242 vnm, &sw_if_index))
243 ;
244 }
245
246 if (~0 == sw_if_index)
247 {
248 u32 ii, n_ints = 0;
249
250 FOR_EACH_QOS_SOURCE (qs)
251 {
252 n_ints = clib_max (n_ints, vec_len (qos_mark_configs[qs]));
253 }
254
255 for (ii = 0; ii < n_ints; ii++)
256 {
257 qos_mark_show_one_interface (vm, ii);
258 }
259 }
260 else
261 qos_mark_show_one_interface (vm, sw_if_index);
262
263 return (NULL);
264}
265
266/*?
267 * Show Egress Qos Maps
268 *
269 * @cliexpar
270 * @cliexcmd{show qos egress map}
271 ?*/
Neale Ranns5281a902019-07-23 08:16:19 -0700272VLIB_CLI_COMMAND (qos_mark_show_command, static) = {
273 .path = "show qos mark",
274 .short_help = "show qos mark [interface]",
275 .function = qos_mark_show,
276 .is_mp_safe = 1,
277};
Neale Ranns5281a902019-07-23 08:16:19 -0700278
Neale Ranns039cbfe2018-02-27 03:45:38 -0800279/*
280 * fd.io coding-style-patch-verification: ON
281 *
282* Local Variables:
283* eval: (c-set-style "gnu")
284* End:
285*
286*/