de46ad38c5f11f6a726bf7c6f5a7c2a3d72112e9
[vpp.git] / vnet / vnet / ip / ip.h
1 /*
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>
44 #include <vppinfra/heap.h>              /* adjacency heap */
45
46 #include <vnet/vnet.h>
47
48 #include <vnet/ip/format.h>
49 #include <vnet/ip/ip_packet.h>
50 #include <vnet/ip/lookup.h>
51
52 #include <vnet/ip/tcp_packet.h>
53 #include <vnet/ip/udp_packet.h>
54 #include <vnet/ip/icmp46_packet.h>
55
56 #include <vnet/ip/ip4.h>
57 #include <vnet/ip/ip4_error.h>
58 #include <vnet/ip/ip4_packet.h>
59 #include <vnet/ip/icmp4.h>
60
61 #include <vnet/ip/ip6.h>
62 #include <vnet/ip/ip6_packet.h>
63 #include <vnet/ip/ip6_error.h>
64 #include <vnet/ip/icmp6.h>
65
66 #if DPDK > 0
67 #include <vnet/devices/dpdk/dpdk.h>
68 #endif
69
70 #include <vnet/classify/vnet_classify.h>
71
72 typedef union {
73   ip4_address_t ip4;
74   ip6_address_t ip6;
75 } ip46_address_t;
76
77 /* Per protocol info. */
78 typedef struct {
79   /* Protocol name (also used as hash key). */
80   u8 * name;
81
82   /* Protocol number. */
83   ip_protocol_t protocol;
84
85   /* Format function for this IP protocol. */
86   format_function_t * format_header;
87
88   /* Parser for header. */
89   unformat_function_t * unformat_header;
90
91   /* Parser for per-protocol matches. */
92   unformat_function_t * unformat_match;
93
94   /* Parser for packet generator edits for this protocol. */
95   unformat_function_t * unformat_pg_edit;
96 } ip_protocol_info_t;
97
98 /* Per TCP/UDP port info. */
99 typedef struct {
100   /* Port name (used as hash key). */
101   u8 * name;
102
103   /* UDP/TCP port number in network byte order. */
104   u16 port;
105
106   /* Port specific format function. */
107   format_function_t * format_header;
108
109   /* Parser for packet generator edits for this protocol. */
110   unformat_function_t * unformat_pg_edit;
111 } tcp_udp_port_info_t;
112
113 typedef struct {
114   /* Per IP protocol info. */
115   ip_protocol_info_t * protocol_infos;
116
117   /* Protocol info index hashed by 8 bit IP protocol. */
118   uword * protocol_info_by_protocol;
119
120   /* Hash table mapping IP protocol name (see protocols.def)
121      to protocol number. */
122   uword * protocol_info_by_name;
123
124   /* Per TCP/UDP port info. */
125   tcp_udp_port_info_t * port_infos;
126
127   /* Hash table from network-byte-order port to port info index. */
128   uword * port_info_by_port;
129
130   /* Hash table mapping TCP/UDP name to port info index. */
131   uword * port_info_by_name;
132 } ip_main_t;
133
134 extern ip_main_t ip_main;
135
136 clib_error_t *
137 ip_main_init (vlib_main_t * vm);
138
139 static inline ip_protocol_info_t *
140 ip_get_protocol_info (ip_main_t * im, u32 protocol)
141 {
142   uword * p;
143
144   p = hash_get (im->protocol_info_by_protocol, protocol);
145   return p ? vec_elt_at_index (im->protocol_infos, p[0]) : 0;
146 }
147
148 static inline tcp_udp_port_info_t *
149 ip_get_tcp_udp_port_info (ip_main_t * im, u32 port)
150 {
151   uword * p;
152
153   p = hash_get (im->port_info_by_port, port);
154   return p ? vec_elt_at_index (im->port_infos, p[0]) : 0;
155 }
156       
157 always_inline ip_csum_t
158 ip_incremental_checksum_buffer (vlib_main_t * vm, vlib_buffer_t * first_buffer,
159                                 u32 first_buffer_offset,
160                                 u32 n_bytes_to_checksum,
161                                 ip_csum_t sum)
162 #if DPDK > 0
163 {
164   u32 n_bytes_left = n_bytes_to_checksum;
165   struct rte_mbuf * mb = rte_mbuf_from_vlib_buffer(first_buffer);
166   u8 nb_segs = mb->nb_segs;
167   ASSERT(mb->data_len >= first_buffer_offset);
168   void * h;
169   u32 n;
170   
171   n = clib_min (n_bytes_left, mb->data_len);
172   h = vlib_buffer_get_current (first_buffer) + first_buffer_offset;
173   while (n_bytes_left)
174     {
175       sum = ip_incremental_checksum (sum, h, n);
176       n_bytes_left -= n;
177       nb_segs--;
178       mb = mb->next;
179       if ((nb_segs == 0) || (mb == 0))
180         break;
181
182       n = clib_min (n_bytes_left, mb->data_len);
183       h = rte_ctrlmbuf_data(mb);
184     }
185  
186   ASSERT(n_bytes_left == 0);
187   ASSERT(nb_segs == 0);
188   return sum;
189 }
190 #else
191 {
192   vlib_buffer_t * b = first_buffer;
193   u32 n_bytes_left = n_bytes_to_checksum;
194   ASSERT (b->current_length >= first_buffer_offset);
195   void * h;
196   u32 n;
197
198   n = clib_min (n_bytes_left, b->current_length);
199   h = vlib_buffer_get_current (b) + first_buffer_offset;
200   sum = ip_incremental_checksum (sum, h, n);
201   if (PREDICT_FALSE (b->flags & VLIB_BUFFER_NEXT_PRESENT))
202     {
203       while (1)
204         {
205           n_bytes_left -= n;
206           if (n_bytes_left == 0)
207             break;
208           b = vlib_get_buffer (vm, b->next_buffer);
209           n = clib_min (n_bytes_left, b->current_length);
210           h = vlib_buffer_get_current (b);
211           sum = ip_incremental_checksum (sum, h, n);
212         }
213     }
214
215   return sum;
216 }
217 #endif /* DPDK */
218
219 void ip_del_all_interface_addresses (vlib_main_t *vm, u32 sw_if_index);
220
221 extern vlib_node_registration_t ip4_inacl_node;
222 extern vlib_node_registration_t ip6_inacl_node;
223
224 #endif /* included_ip_main_h */