blob: 084243dccfa9ec482988f5b21da65fa15fb435aa [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 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 * ip/ip.h: ip generic (4 or 6) main
17 *
18 * Copyright (c) 2008 Eliot Dresselhaus
19 *
20 * Permission is hereby granted, free of charge, to any person obtaining
21 * a copy of this software and associated documentation files (the
22 * "Software"), to deal in the Software without restriction, including
23 * without limitation the rights to use, copy, modify, merge, publish,
24 * distribute, sublicense, and/or sell copies of the Software, and to
25 * permit persons to whom the Software is furnished to do so, subject to
26 * the following conditions:
27 *
28 * The above copyright notice and this permission notice shall be
29 * included in all copies or substantial portions of the Software.
30 *
31 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
32 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
33 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
34 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
35 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
36 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
37 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
38 */
39
40#ifndef included_ip_main_h
41#define included_ip_main_h
42
43#include <vppinfra/hash.h>
Dave Barachd7cb1b52016-12-09 09:52:16 -050044#include <vppinfra/heap.h> /* adjacency heap */
Dave Barachd6534602016-06-14 18:38:02 -040045#include <vppinfra/ptclosure.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070046
47#include <vnet/vnet.h>
48
Neale Rannsea93e482019-11-12 17:16:47 +000049#include <vnet/ip/ip_types.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070050#include <vnet/ip/format.h>
51#include <vnet/ip/ip_packet.h>
52#include <vnet/ip/lookup.h>
Neale Ranns59f71132020-04-08 12:19:38 +000053#include <vnet/ip/ip_interface.h>
Neale Rannse22a7042022-08-09 03:03:29 +000054#include <vnet/ip/ip.api_enum.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070055
Dave Barach68b0fb02017-02-28 15:15:56 -050056#include <vnet/tcp/tcp_packet.h>
57#include <vnet/udp/udp_packet.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070058#include <vnet/ip/icmp46_packet.h>
59
60#include <vnet/ip/ip4.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070061#include <vnet/ip/ip4_packet.h>
Ole Troan92eade12016-01-13 20:17:08 +010062#include <vnet/ip/icmp4.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070063
64#include <vnet/ip/ip6.h>
65#include <vnet/ip/ip6_packet.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070066#include <vnet/ip/icmp6.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070067
Ed Warnickecb9cada2015-12-08 15:45:58 -070068/* Per protocol info. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050069typedef struct
70{
Ed Warnickecb9cada2015-12-08 15:45:58 -070071 /* Protocol name (also used as hash key). */
Dave Barachd7cb1b52016-12-09 09:52:16 -050072 u8 *name;
Ed Warnickecb9cada2015-12-08 15:45:58 -070073
74 /* Protocol number. */
75 ip_protocol_t protocol;
76
77 /* Format function for this IP protocol. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050078 format_function_t *format_header;
Ed Warnickecb9cada2015-12-08 15:45:58 -070079
80 /* Parser for header. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050081 unformat_function_t *unformat_header;
Ed Warnickecb9cada2015-12-08 15:45:58 -070082
83 /* Parser for per-protocol matches. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050084 unformat_function_t *unformat_match;
Ed Warnickecb9cada2015-12-08 15:45:58 -070085
86 /* Parser for packet generator edits for this protocol. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050087 unformat_function_t *unformat_pg_edit;
Ed Warnickecb9cada2015-12-08 15:45:58 -070088} ip_protocol_info_t;
89
90/* Per TCP/UDP port info. */
Dave Barachd7cb1b52016-12-09 09:52:16 -050091typedef struct
92{
Ed Warnickecb9cada2015-12-08 15:45:58 -070093 /* Port name (used as hash key). */
Dave Barachd7cb1b52016-12-09 09:52:16 -050094 u8 *name;
Ed Warnickecb9cada2015-12-08 15:45:58 -070095
96 /* UDP/TCP port number in network byte order. */
97 u16 port;
98
99 /* Port specific format function. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500100 format_function_t *format_header;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700101
102 /* Parser for packet generator edits for this protocol. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500103 unformat_function_t *unformat_pg_edit;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700104} tcp_udp_port_info_t;
105
Dave Barachd7cb1b52016-12-09 09:52:16 -0500106typedef struct
107{
Ed Warnickecb9cada2015-12-08 15:45:58 -0700108 /* Per IP protocol info. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500109 ip_protocol_info_t *protocol_infos;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700110
111 /* Protocol info index hashed by 8 bit IP protocol. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500112 uword *protocol_info_by_protocol;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700113
114 /* Hash table mapping IP protocol name (see protocols.def)
115 to protocol number. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500116 uword *protocol_info_by_name;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700117
118 /* Per TCP/UDP port info. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500119 tcp_udp_port_info_t *port_infos;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700120
121 /* Hash table from network-byte-order port to port info index. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500122 uword *port_info_by_port;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700123
124 /* Hash table mapping TCP/UDP name to port info index. */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500125 uword *port_info_by_name;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700126} ip_main_t;
127
128extern ip_main_t ip_main;
129
Dave Barachd7cb1b52016-12-09 09:52:16 -0500130clib_error_t *ip_main_init (vlib_main_t * vm);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700131
132static inline ip_protocol_info_t *
133ip_get_protocol_info (ip_main_t * im, u32 protocol)
134{
Dave Barachd7cb1b52016-12-09 09:52:16 -0500135 uword *p;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700136
137 p = hash_get (im->protocol_info_by_protocol, protocol);
138 return p ? vec_elt_at_index (im->protocol_infos, p[0]) : 0;
139}
140
141static inline tcp_udp_port_info_t *
142ip_get_tcp_udp_port_info (ip_main_t * im, u32 port)
143{
Dave Barachd7cb1b52016-12-09 09:52:16 -0500144 uword *p;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700145
146 p = hash_get (im->port_info_by_port, port);
147 return p ? vec_elt_at_index (im->port_infos, p[0]) : 0;
148}
Dave Barachd7cb1b52016-12-09 09:52:16 -0500149
Ed Warnickecb9cada2015-12-08 15:45:58 -0700150always_inline ip_csum_t
Dave Barachd7cb1b52016-12-09 09:52:16 -0500151ip_incremental_checksum_buffer (vlib_main_t * vm,
152 vlib_buffer_t * first_buffer,
Ed Warnickecb9cada2015-12-08 15:45:58 -0700153 u32 first_buffer_offset,
Dave Barachd7cb1b52016-12-09 09:52:16 -0500154 u32 n_bytes_to_checksum, ip_csum_t sum)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700155{
Dave Barachd7cb1b52016-12-09 09:52:16 -0500156 vlib_buffer_t *b = first_buffer;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700157 u32 n_bytes_left = n_bytes_to_checksum;
158 ASSERT (b->current_length >= first_buffer_offset);
Dave Barachd7cb1b52016-12-09 09:52:16 -0500159 void *h;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700160 u32 n;
161
Juraj Sloboda8f39d552018-09-26 14:25:32 +0200162 n = clib_min (n_bytes_left, b->current_length - first_buffer_offset);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700163 h = vlib_buffer_get_current (b) + first_buffer_offset;
164 sum = ip_incremental_checksum (sum, h, n);
165 if (PREDICT_FALSE (b->flags & VLIB_BUFFER_NEXT_PRESENT))
166 {
167 while (1)
168 {
169 n_bytes_left -= n;
170 if (n_bytes_left == 0)
171 break;
172 b = vlib_get_buffer (vm, b->next_buffer);
173 n = clib_min (n_bytes_left, b->current_length);
174 h = vlib_buffer_get_current (b);
175 sum = ip_incremental_checksum (sum, h, n);
176 }
177 }
178
179 return sum;
180}
Ed Warnickecb9cada2015-12-08 15:45:58 -0700181
Srikanth A02833ff2019-10-02 17:48:58 -0700182always_inline u16
183ip_calculate_l4_checksum (vlib_main_t * vm, vlib_buffer_t * p0,
184 ip_csum_t sum0, u32 payload_length,
185 u8 * iph, u32 ip_header_size, u8 * l4h)
186{
187 u16 sum16;
188 u8 *data_this_buffer, length_odd;
189 u32 n_bytes_left, n_this_buffer, n_ip_bytes_this_buffer;
190
191 n_bytes_left = payload_length;
192
193 if (l4h) /* packet l4 header and no buffer chain involved */
194 {
195 ASSERT (p0 == NULL);
196 n_this_buffer = payload_length;
197 data_this_buffer = l4h;
198 }
199 else
200 {
201 ASSERT (p0);
202 if (iph) /* ip header pointer set to packet in buffer */
203 {
204 ASSERT (ip_header_size);
205 n_this_buffer = payload_length;
206 data_this_buffer = iph + ip_header_size; /* at l4 header */
207 n_ip_bytes_this_buffer =
208 p0->current_length - (((u8 *) iph - p0->data) - p0->current_data);
209 if (PREDICT_FALSE (payload_length + ip_header_size >
210 n_ip_bytes_this_buffer))
211 {
212 n_this_buffer = n_ip_bytes_this_buffer - ip_header_size;
213 if (PREDICT_FALSE (n_this_buffer >> 31))
214 { /* error - ip header don't fit this buffer */
Srikanth A02833ff2019-10-02 17:48:58 -0700215 return 0xfefe;
216 }
217 }
218 }
219 else /* packet in buffer with no ip header */
220 { /* buffer current pointer at l4 header */
221 n_this_buffer = p0->current_length;
222 data_this_buffer = vlib_buffer_get_current (p0);
223 }
224 n_this_buffer = clib_min (n_this_buffer, n_bytes_left);
225 }
226
227 while (1)
228 {
229 sum0 = ip_incremental_checksum (sum0, data_this_buffer, n_this_buffer);
230 n_bytes_left -= n_this_buffer;
231 if (n_bytes_left == 0)
232 break;
233
234 if (!(p0->flags & VLIB_BUFFER_NEXT_PRESENT))
235 {
Srikanth A02833ff2019-10-02 17:48:58 -0700236 return 0xfefe;
237 }
238
239 length_odd = (n_this_buffer & 1);
240
241 p0 = vlib_get_buffer (vm, p0->next_buffer);
242 data_this_buffer = vlib_buffer_get_current (p0);
243 n_this_buffer = clib_min (p0->current_length, n_bytes_left);
244
245 if (PREDICT_FALSE (length_odd))
246 {
247 /* Prepend a 0 byte to maintain 2-byte checksum alignment */
248 data_this_buffer--;
249 n_this_buffer++;
250 n_bytes_left++;
251 data_this_buffer[0] = 0;
252 }
253 }
254
255 sum16 = ~ip_csum_fold (sum0);
256 return sum16;
257}
258
Dave Barachd7cb1b52016-12-09 09:52:16 -0500259void ip_del_all_interface_addresses (vlib_main_t * vm, u32 sw_if_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700260
Dave Barachf39ff742016-03-20 10:14:45 -0400261extern vlib_node_registration_t ip4_inacl_node;
262extern vlib_node_registration_t ip6_inacl_node;
263
Neale Ranns2297af02017-09-12 09:45:04 -0700264void ip_table_create (fib_protocol_t fproto, u32 table_id, u8 is_api,
BenoƮt Ganneff570d32024-04-16 09:36:05 +0200265 u8 create_mfib, const u8 *name);
Neale Ranns15002542017-09-10 04:39:11 -0700266
Neale Ranns6b3a8ef2017-09-11 10:34:33 -0700267void ip_table_delete (fib_protocol_t fproto, u32 table_id, u8 is_api);
Neale Ranns15002542017-09-10 04:39:11 -0700268
Nathan Skrzypczaka424dd12021-08-20 15:53:43 +0200269void fib_table_bind (fib_protocol_t fproto, u32 sw_if_index, u32 fib_index);
270void mfib_table_bind (fib_protocol_t fproto, u32 sw_if_index, u32 mfib_index);
Nathan Skrzypczak275bd792021-09-17 17:29:14 +0200271int ip_table_bind (fib_protocol_t fproto, u32 sw_if_index, u32 table_id);
Neale Ranns15002542017-09-10 04:39:11 -0700272
Aloys Augustin6e4cfb52021-09-16 20:53:14 +0200273u32 ip_table_get_unused_id (fib_protocol_t fproto);
274
Florin Corascea194d2017-10-02 00:18:51 -0700275u8 ip_is_zero (ip46_address_t * ip46_address, u8 is_ip4);
276u8 ip_is_local_host (ip46_address_t * ip46_address, u8 is_ip4);
Florin Coras477e91a2018-02-27 10:05:57 -0800277u8 ip4_is_local_host (ip4_address_t * ip4_address);
278u8 ip6_is_local_host (ip6_address_t * ip6_address);
Florin Corascea194d2017-10-02 00:18:51 -0700279u8 ip_is_local (u32 fib_index, ip46_address_t * ip46_address, u8 is_ip4);
Florin Corascea194d2017-10-02 00:18:51 -0700280void ip_copy (ip46_address_t * dst, ip46_address_t * src, u8 is_ip4);
281void ip_set (ip46_address_t * dst, void *src, u8 is_ip4);
282
Neale Ranns6dc0c8d2020-11-26 14:15:33 +0000283void ip_feature_enable_disable (ip_address_family_t af,
284 ip_sub_address_family_t safi,
285 ip_feature_location_t loc,
286 const char *feature,
287 u32 sw_if_index, int enable_disable,
288 void *feature_config,
289 u32 n_feature_config_bytes);
290
Neale Rannsb28652e2021-10-25 09:47:09 +0000291ethernet_type_t ip_address_family_to_ether_type (ip_address_family_t af);
292
Nick Zavaritsky27518c22020-02-27 15:54:58 +0000293always_inline u32 vlib_buffer_get_ip4_fib_index (vlib_buffer_t * b);
294always_inline u32 vlib_buffer_get_ip6_fib_index (vlib_buffer_t * b);
295always_inline u32
296vlib_buffer_get_ip_fib_index (vlib_buffer_t * b, u8 is_ip4)
297{
298 return (is_ip4 ? vlib_buffer_get_ip4_fib_index
299 : vlib_buffer_get_ip6_fib_index) (b);
300}
301
Ed Warnickecb9cada2015-12-08 15:45:58 -0700302#endif /* included_ip_main_h */
Dave Barachd7cb1b52016-12-09 09:52:16 -0500303
304/*
305 * fd.io coding-style-patch-verification: ON
306 *
307 * Local Variables:
308 * eval: (c-set-style "gnu")
309 * End:
310 */