blob: dcb0f9d1ab3199616c898d0eb41ad0b1db6b1c5d [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
120static clib_error_t *
121qos_mark_cli (vlib_main_t * vm,
122 unformat_input_t * input, vlib_cli_command_t * cmd)
123{
124 qos_egress_map_id_t map_id;
125 u32 sw_if_index, qs;
126 vnet_main_t *vnm;
127 int rv, enable;
128
129 vnm = vnet_get_main ();
130 map_id = ~0;
131 qs = 0xff;
132 enable = 1;
133
134 while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
135 {
136 if (unformat (input, "id %d", &map_id))
137 ;
138 else if (unformat (input, "disable"))
139 enable = 0;
140 else if (unformat (input, "%U", unformat_qos_source, &qs))
141 ;
142 else if (unformat (input, "%U",
143 unformat_vnet_sw_interface, vnm, &sw_if_index))
144 ;
145 else
146 break;
147 }
148
149 if (~0 == sw_if_index)
150 return clib_error_return (0, "interface must be specified");
151 if (0xff == qs)
152 return clib_error_return (0, "output location must be specified");
153
154 if (enable)
155 rv = qos_mark_enable (sw_if_index, qs, map_id);
156 else
157 rv = qos_mark_disable (sw_if_index, qs);
158
159 if (0 == rv)
160 return (NULL);
161
162 return clib_error_return (0, "Failed to map interface");
163}
164
165/*?
166 * Apply a QoS egress mapping table to an interface for QoS marking packets
167 * at the given output protocol.
168 *
169 * @cliexpar
170 * @cliexcmd{qos egress interface GigEthernet0/9/0 id 0 output ip}
171 ?*/
172/* *INDENT-OFF* */
173VLIB_CLI_COMMAND (qos_egress_map_interface_command, static) = {
174 .path = "qos mark",
175 .short_help = "qos mark <SOURCE> <INTERFACE> id <MAP>",
176 .function = qos_mark_cli,
177 .is_mp_safe = 1,
178};
179/* *INDENT-ON* */
180
181/*
182 * fd.io coding-style-patch-verification: ON
183 *
184* Local Variables:
185* eval: (c-set-style "gnu")
186* End:
187*
188*/