blob: 586f7dfbc85dfa7db81ddd53e6eef1cc8cc18446 [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 +0000121static const char *ip_arc_names[N_IP_FEATURE_LOCATIONS][N_AF][N_SAFI] = {
122 [IP_FEATURE_INPUT] = {
123 [AF_IP4] = {
124 [SAFI_UNICAST] = "ip4-unicast",
125 [SAFI_MULTICAST] = "ip4-multicast",
126 },
127 [AF_IP6] = {
128 [SAFI_UNICAST] = "ip6-unicast",
129 [SAFI_MULTICAST] = "ip6-multicast",
130 },
131 },
132 [IP_FEATURE_OUTPUT] = {
133 [AF_IP4] = {
134 [SAFI_UNICAST] = "ip4-output",
135 [SAFI_MULTICAST] = "ip4-output",
136 },
137 [AF_IP6] = {
138 [SAFI_UNICAST] = "ip6-output",
139 [SAFI_MULTICAST] = "ip6-output",
140 },
141 },
142 [IP_FEATURE_LOCAL] = {
143 [AF_IP4] = {
144 [SAFI_UNICAST] = "ip4-local",
145 [SAFI_MULTICAST] = "ip4-local",
146 },
147 [AF_IP6] = {
148 [SAFI_UNICAST] = "ip6-local",
149 [SAFI_MULTICAST] = "ip6-local",
150 },
151 },
152 [IP_FEATURE_PUNT] = {
153 [AF_IP4] = {
154 [SAFI_UNICAST] = "ip4-punt",
155 [SAFI_MULTICAST] = "ip4-punt",
156 },
157 [AF_IP6] = {
158 [SAFI_UNICAST] = "ip6-punt",
159 [SAFI_MULTICAST] = "ip6-punt",
160 },
161 },
162 [IP_FEATURE_DROP] = {
163 [AF_IP4] = {
164 [SAFI_UNICAST] = "ip4-drop",
165 [SAFI_MULTICAST] = "ip4-drop",
166 },
167 [AF_IP6] = {
168 [SAFI_UNICAST] = "ip6-drop",
169 [SAFI_MULTICAST] = "ip6-drop",
170 },
171 },
172};
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000173
174void
175ip_feature_enable_disable (ip_address_family_t af,
176 ip_sub_address_family_t safi,
177 ip_feature_location_t loc,
178 const char *feature_name,
179 u32 sw_if_index, int enable,
180 void *feature_config, u32 n_feature_config_bytes)
181{
182 if (IP_FEATURE_INPUT == loc)
183 {
184 if (N_SAFI == safi)
185 FOR_EACH_IP_ADDRESS_SUB_FAMILY (safi)
186 vnet_feature_enable_disable (ip_arc_names[loc][af][safi],
187 feature_name, sw_if_index,
188 enable, feature_config,
189 n_feature_config_bytes);
190 else
191 vnet_feature_enable_disable (ip_arc_names[loc][af][safi],
192 feature_name, sw_if_index,
193 enable, feature_config,
194 n_feature_config_bytes);
195 }
196 else
197 vnet_feature_enable_disable (ip_arc_names[loc][af][SAFI_UNICAST],
198 feature_name, sw_if_index,
199 enable, feature_config,
200 n_feature_config_bytes);
201}
202
Ahmed Abdelsalamf2984bb2020-11-20 18:56:09 +0000203int
Damjan Marion804255c2024-04-24 11:21:05 +0000204ip_flow_hash_set (ip_address_family_t af, u32 table_id,
205 flow_hash_config_t flow_hash_config)
Ahmed Abdelsalamf2984bb2020-11-20 18:56:09 +0000206{
207 fib_protocol_t fproto;
208 u32 fib_index;
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000209
Ahmed Abdelsalamf2984bb2020-11-20 18:56:09 +0000210 fproto = ip_address_family_to_fib_proto (af);
211 fib_index = fib_table_find (fproto, table_id);
212
213 if (~0 == fib_index)
214 return VNET_API_ERROR_NO_SUCH_FIB;
215
216 fib_table_set_flow_hash_config (fib_index, fproto, flow_hash_config);
217
218 return 0;
219}
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000220
Neale Ranns3d5f08a2021-01-22 16:12:38 +0000221void
222ip_flow_hash_router_id_set (u32 router_id)
223{
224 ip_flow_hash_router_id = router_id;
225}
226
Neale Ranns50f0ac02019-05-15 02:13:37 -0700227u8 *
228format_ip_address_family (u8 * s, va_list * args)
229{
Neale Ranns3ec09e92020-02-24 13:32:30 +0000230 ip_address_family_t af = va_arg (*args, int); // int promo ip_address_family_t);
Neale Ranns50f0ac02019-05-15 02:13:37 -0700231
232 switch (af)
233 {
234 case AF_IP4:
235 return (format (s, "ip4"));
236 case AF_IP6:
237 return (format (s, "ip6"));
238 }
239
240 return (format (s, "unknown"));
241}
242
Neale Ranns3ec09e92020-02-24 13:32:30 +0000243uword
244unformat_ip_address_family (unformat_input_t * input, va_list * args)
245{
246 ip_address_family_t *af = va_arg (*args, ip_address_family_t *);
247
248 if (unformat (input, "ip4") || unformat (input, "ipv4") ||
249 unformat (input, "IP4") || unformat (input, "IPv4"))
250 {
251 *af = AF_IP4;
252 return (1);
253 }
254 else if (unformat (input, "ip6") || unformat (input, "ipv6") ||
255 unformat (input, "IP6") || unformat (input, "IPv6"))
256 {
257 *af = AF_IP6;
258 return (1);
259 }
260 return (0);
261}
262
Neale Ranns038e1df2019-07-19 14:01:02 +0000263u8 *
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000264format_ip_sub_address_family (u8 * s, va_list * args)
265{
266 ip_sub_address_family_t safi = va_arg (*args, int); // int promo ip_sub_address_family_t);
267
268 switch (safi)
269 {
270 case SAFI_UNICAST:
271 return (format (s, "unicast"));
272 case SAFI_MULTICAST:
273 return (format (s, "multicast"));
274 }
275
276 return (format (s, "unknown"));
277}
278
279uword
280unformat_ip_sub_address_family (unformat_input_t * input, va_list * args)
281{
282 ip_sub_address_family_t *safi = va_arg (*args, ip_sub_address_family_t *);
283
284 if (unformat (input, "unicast") || unformat (input, "uni"))
285 {
286 *safi = SAFI_UNICAST;
287 return (1);
288 }
289 else if (unformat (input, "multicast") || unformat (input, "multi"))
290 {
291 *safi = SAFI_MULTICAST;
292 return (1);
293 }
294 return (0);
295}
296
297u8 *
Neale Ranns038e1df2019-07-19 14:01:02 +0000298format_ip_dscp (u8 * s, va_list * va)
299{
300 ip_dscp_t dscp = va_arg (*va, u32); // int promotion of u8
301
302 switch (dscp)
303 {
304#define _(n,v) \
305 case IP_DSCP_##v: \
306 return (format (s, "%s", #v));
307 foreach_ip_dscp
308#undef _
309 }
310
Paul Vinciguerra5b755e22019-10-26 19:34:40 -0400311 return (format (s, "unknown"));
Neale Ranns038e1df2019-07-19 14:01:02 +0000312}
313
Neale Ranns041add72020-01-02 04:06:10 +0000314uword
315unformat_ip_dscp (unformat_input_t * input, va_list * args)
316{
317 ip_dscp_t *dscp = va_arg (*args, ip_dscp_t *);
318
319 if (0)
320 ;
321#define _(n,v) \
322 else if (unformat (input, #v)) \
323 *dscp = IP_DSCP_##v;
324 foreach_ip_dscp
325#undef _
326 else
327 return 0;
328
329 return 1;
330}
331
Neale Ranns95346962019-11-25 13:04:44 +0000332u8 *
333format_ip_ecn (u8 * s, va_list * va)
334{
335 ip_ecn_t ecn = va_arg (*va, u32); // int promotion of u8
336
337 switch (ecn)
338 {
339#define _(n,v) \
340 case IP_ECN_##v: \
341 return (format (s, "%s", #v));
342 foreach_ip_ecn
343#undef _
344 }
345
346 return (format (s, "unknown"));
347}
348
Florin Corascea194d2017-10-02 00:18:51 -0700349/*
350 * fd.io coding-style-patch-verification: ON
351 *
352 * Local Variables:
353 * eval: (c-set-style "gnu")
354 * End:
355 */