blob: 0a602b43ac740e965904d76e67b2cf6b9eecb257 [file] [log] [blame]
Florin Corascea194d2017-10-02 00:18:51 -07001/*
2 * Copyright (c) 2017 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/fib/fib_table.h>
18
Neale Ranns3d5f08a2021-01-22 16:12:38 +000019u32 ip_flow_hash_router_id;
20
Neale Rannsb28652e2021-10-25 09:47:09 +000021ethernet_type_t
22ip_address_family_to_ether_type (ip_address_family_t af)
23{
24 switch (af)
25 {
26 case AF_IP4:
27 return (ETHERNET_TYPE_IP4);
28 case AF_IP6:
29 return (ETHERNET_TYPE_IP6);
30 }
31 ASSERT (0);
32 return (ETHERNET_TYPE_IP4);
33}
34
Florin Corascea194d2017-10-02 00:18:51 -070035u8
36ip_is_zero (ip46_address_t * ip46_address, u8 is_ip4)
37{
38 if (is_ip4)
39 return (ip46_address->ip4.as_u32 == 0);
40 else
41 return (ip46_address->as_u64[0] == 0 && ip46_address->as_u64[1] == 0);
42}
43
44u8
45ip_is_local_host (ip46_address_t * ip46_address, u8 is_ip4)
46{
47 if (is_ip4)
48 return (ip46_address->ip4.as_u8[0] == 127);
49 else
Dave Wallacede910062018-03-20 09:22:13 -040050 return (ip46_address->as_u64[0] == 0 &&
51 clib_net_to_host_u64 (ip46_address->as_u64[1]) == 1);
Florin Corascea194d2017-10-02 00:18:51 -070052}
53
Florin Coras477e91a2018-02-27 10:05:57 -080054u8
55ip4_is_local_host (ip4_address_t * ip4_address)
56{
57 return (ip4_address->as_u8[0] == 127);
58}
59
60u8
61ip6_is_local_host (ip6_address_t * ip6_address)
62{
Dave Wallacede910062018-03-20 09:22:13 -040063 return (ip6_address->as_u64[0] == 0 &&
64 clib_net_to_host_u64 (ip6_address->as_u64[1]) == 1);
Florin Coras477e91a2018-02-27 10:05:57 -080065}
66
Florin Corascea194d2017-10-02 00:18:51 -070067/**
68 * Checks that an ip is local to the requested fib
69 */
70u8
71ip_is_local (u32 fib_index, ip46_address_t * ip46_address, u8 is_ip4)
72{
73 fib_node_index_t fei;
74 fib_entry_flag_t flags;
75 fib_prefix_t prefix;
76
77 /* Check if requester is local */
78 if (is_ip4)
79 {
80 prefix.fp_len = 32;
81 prefix.fp_proto = FIB_PROTOCOL_IP4;
82 }
83 else
84 {
85 prefix.fp_len = 128;
86 prefix.fp_proto = FIB_PROTOCOL_IP6;
87 }
88
Dave Barach178cf492018-11-13 16:34:13 -050089 clib_memcpy_fast (&prefix.fp_addr, ip46_address, sizeof (ip46_address_t));
Florin Corasa46b4c92018-04-03 02:10:05 -070090 fei = fib_table_lookup (fib_index, &prefix);
Florin Corascea194d2017-10-02 00:18:51 -070091 flags = fib_entry_get_flags (fei);
92
93 return (flags & FIB_ENTRY_FLAG_LOCAL);
94}
95
Florin Coras3cbc04b2017-10-02 00:18:51 -070096void
97ip_copy (ip46_address_t * dst, ip46_address_t * src, u8 is_ip4)
98{
99 if (is_ip4)
jiangxiaoming9268b582020-06-22 09:06:16 +0800100 {
101 ip46_address_mask_ip4 (dst);
102 dst->ip4.as_u32 = src->ip4.as_u32;
103 }
Florin Coras3cbc04b2017-10-02 00:18:51 -0700104 else
Dave Barach178cf492018-11-13 16:34:13 -0500105 clib_memcpy_fast (&dst->ip6, &src->ip6, sizeof (ip6_address_t));
Florin Coras3cbc04b2017-10-02 00:18:51 -0700106}
107
108void
109ip_set (ip46_address_t * dst, void *src, u8 is_ip4)
110{
111 if (is_ip4)
jiangxiaoming9268b582020-06-22 09:06:16 +0800112 {
113 ip46_address_mask_ip4 (dst);
114 dst->ip4.as_u32 = ((ip4_address_t *) src)->as_u32;
115 }
Florin Coras3cbc04b2017-10-02 00:18:51 -0700116 else
Dave Barach178cf492018-11-13 16:34:13 -0500117 clib_memcpy_fast (&dst->ip6, (ip6_address_t *) src,
118 sizeof (ip6_address_t));
Florin Coras3cbc04b2017-10-02 00:18:51 -0700119}
120
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000121/* *INDENT-OFF* */
122static const char *ip_arc_names[N_IP_FEATURE_LOCATIONS][N_AF][N_SAFI] = {
123 [IP_FEATURE_INPUT] = {
124 [AF_IP4] = {
125 [SAFI_UNICAST] = "ip4-unicast",
126 [SAFI_MULTICAST] = "ip4-multicast",
127 },
128 [AF_IP6] = {
129 [SAFI_UNICAST] = "ip6-unicast",
130 [SAFI_MULTICAST] = "ip6-multicast",
131 },
132 },
133 [IP_FEATURE_OUTPUT] = {
134 [AF_IP4] = {
135 [SAFI_UNICAST] = "ip4-output",
136 [SAFI_MULTICAST] = "ip4-output",
137 },
138 [AF_IP6] = {
139 [SAFI_UNICAST] = "ip6-output",
140 [SAFI_MULTICAST] = "ip6-output",
141 },
142 },
143 [IP_FEATURE_LOCAL] = {
144 [AF_IP4] = {
145 [SAFI_UNICAST] = "ip4-local",
146 [SAFI_MULTICAST] = "ip4-local",
147 },
148 [AF_IP6] = {
149 [SAFI_UNICAST] = "ip6-local",
150 [SAFI_MULTICAST] = "ip6-local",
151 },
152 },
153 [IP_FEATURE_PUNT] = {
154 [AF_IP4] = {
155 [SAFI_UNICAST] = "ip4-punt",
156 [SAFI_MULTICAST] = "ip4-punt",
157 },
158 [AF_IP6] = {
159 [SAFI_UNICAST] = "ip6-punt",
160 [SAFI_MULTICAST] = "ip6-punt",
161 },
162 },
163 [IP_FEATURE_DROP] = {
164 [AF_IP4] = {
165 [SAFI_UNICAST] = "ip4-drop",
166 [SAFI_MULTICAST] = "ip4-drop",
167 },
168 [AF_IP6] = {
169 [SAFI_UNICAST] = "ip6-drop",
170 [SAFI_MULTICAST] = "ip6-drop",
171 },
172 },
173};
174/* *INDENT-ON* */
175
176void
177ip_feature_enable_disable (ip_address_family_t af,
178 ip_sub_address_family_t safi,
179 ip_feature_location_t loc,
180 const char *feature_name,
181 u32 sw_if_index, int enable,
182 void *feature_config, u32 n_feature_config_bytes)
183{
184 if (IP_FEATURE_INPUT == loc)
185 {
186 if (N_SAFI == safi)
187 FOR_EACH_IP_ADDRESS_SUB_FAMILY (safi)
188 vnet_feature_enable_disable (ip_arc_names[loc][af][safi],
189 feature_name, sw_if_index,
190 enable, feature_config,
191 n_feature_config_bytes);
192 else
193 vnet_feature_enable_disable (ip_arc_names[loc][af][safi],
194 feature_name, sw_if_index,
195 enable, feature_config,
196 n_feature_config_bytes);
197 }
198 else
199 vnet_feature_enable_disable (ip_arc_names[loc][af][SAFI_UNICAST],
200 feature_name, sw_if_index,
201 enable, feature_config,
202 n_feature_config_bytes);
203}
204
Ahmed Abdelsalamf2984bb2020-11-20 18:56:09 +0000205int
206ip_flow_hash_set (ip_address_family_t af, u32 table_id, u32 flow_hash_config)
207{
208 fib_protocol_t fproto;
209 u32 fib_index;
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000210
Ahmed Abdelsalamf2984bb2020-11-20 18:56:09 +0000211 fproto = ip_address_family_to_fib_proto (af);
212 fib_index = fib_table_find (fproto, table_id);
213
214 if (~0 == fib_index)
215 return VNET_API_ERROR_NO_SUCH_FIB;
216
217 fib_table_set_flow_hash_config (fib_index, fproto, flow_hash_config);
218
219 return 0;
220}
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000221
Neale Ranns3d5f08a2021-01-22 16:12:38 +0000222void
223ip_flow_hash_router_id_set (u32 router_id)
224{
225 ip_flow_hash_router_id = router_id;
226}
227
Neale Ranns50f0ac02019-05-15 02:13:37 -0700228u8 *
229format_ip_address_family (u8 * s, va_list * args)
230{
Neale Ranns3ec09e92020-02-24 13:32:30 +0000231 ip_address_family_t af = va_arg (*args, int); // int promo ip_address_family_t);
Neale Ranns50f0ac02019-05-15 02:13:37 -0700232
233 switch (af)
234 {
235 case AF_IP4:
236 return (format (s, "ip4"));
237 case AF_IP6:
238 return (format (s, "ip6"));
239 }
240
241 return (format (s, "unknown"));
242}
243
Neale Ranns3ec09e92020-02-24 13:32:30 +0000244uword
245unformat_ip_address_family (unformat_input_t * input, va_list * args)
246{
247 ip_address_family_t *af = va_arg (*args, ip_address_family_t *);
248
249 if (unformat (input, "ip4") || unformat (input, "ipv4") ||
250 unformat (input, "IP4") || unformat (input, "IPv4"))
251 {
252 *af = AF_IP4;
253 return (1);
254 }
255 else if (unformat (input, "ip6") || unformat (input, "ipv6") ||
256 unformat (input, "IP6") || unformat (input, "IPv6"))
257 {
258 *af = AF_IP6;
259 return (1);
260 }
261 return (0);
262}
263
Neale Ranns038e1df2019-07-19 14:01:02 +0000264u8 *
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000265format_ip_sub_address_family (u8 * s, va_list * args)
266{
267 ip_sub_address_family_t safi = va_arg (*args, int); // int promo ip_sub_address_family_t);
268
269 switch (safi)
270 {
271 case SAFI_UNICAST:
272 return (format (s, "unicast"));
273 case SAFI_MULTICAST:
274 return (format (s, "multicast"));
275 }
276
277 return (format (s, "unknown"));
278}
279
280uword
281unformat_ip_sub_address_family (unformat_input_t * input, va_list * args)
282{
283 ip_sub_address_family_t *safi = va_arg (*args, ip_sub_address_family_t *);
284
285 if (unformat (input, "unicast") || unformat (input, "uni"))
286 {
287 *safi = SAFI_UNICAST;
288 return (1);
289 }
290 else if (unformat (input, "multicast") || unformat (input, "multi"))
291 {
292 *safi = SAFI_MULTICAST;
293 return (1);
294 }
295 return (0);
296}
297
298u8 *
Neale Ranns038e1df2019-07-19 14:01:02 +0000299format_ip_dscp (u8 * s, va_list * va)
300{
301 ip_dscp_t dscp = va_arg (*va, u32); // int promotion of u8
302
303 switch (dscp)
304 {
305#define _(n,v) \
306 case IP_DSCP_##v: \
307 return (format (s, "%s", #v));
308 foreach_ip_dscp
309#undef _
310 }
311
Paul Vinciguerra5b755e22019-10-26 19:34:40 -0400312 return (format (s, "unknown"));
Neale Ranns038e1df2019-07-19 14:01:02 +0000313}
314
Neale Ranns041add72020-01-02 04:06:10 +0000315uword
316unformat_ip_dscp (unformat_input_t * input, va_list * args)
317{
318 ip_dscp_t *dscp = va_arg (*args, ip_dscp_t *);
319
320 if (0)
321 ;
322#define _(n,v) \
323 else if (unformat (input, #v)) \
324 *dscp = IP_DSCP_##v;
325 foreach_ip_dscp
326#undef _
327 else
328 return 0;
329
330 return 1;
331}
332
Neale Ranns95346962019-11-25 13:04:44 +0000333u8 *
334format_ip_ecn (u8 * s, va_list * va)
335{
336 ip_ecn_t ecn = va_arg (*va, u32); // int promotion of u8
337
338 switch (ecn)
339 {
340#define _(n,v) \
341 case IP_ECN_##v: \
342 return (format (s, "%s", #v));
343 foreach_ip_ecn
344#undef _
345 }
346
347 return (format (s, "unknown"));
348}
349
Florin Corascea194d2017-10-02 00:18:51 -0700350/*
351 * fd.io coding-style-patch-verification: ON
352 *
353 * Local Variables:
354 * eval: (c-set-style "gnu")
355 * End:
356 */