blob: 0d27ad82538cf4b35474e16257b16498da396615 [file] [log] [blame]
Neale Ranns59ff9182019-12-29 23:55:18 +00001/*
2 * tunnel.h: shared definitions for tunnels.
3 *
4 * Copyright (c) 2019 Cisco and/or its affiliates.
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at:
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
17
18#include <vnet/tunnel/tunnel.h>
Neale Rannsa91cb452021-02-04 11:02:52 +000019#include <vnet/fib/fib_table.h>
20#include <vnet/fib/fib_entry_track.h>
21
22#include <vnet/ip/ip6_inlines.h>
23
24const u8 TUNNEL_ENCAP_DECAP_FLAG_MASK = (
25#define _(a, b, c) TUNNEL_ENCAP_DECAP_FLAG_##a |
26 foreach_tunnel_encap_decap_flag
27#undef _
28 0);
29const u8 TUNNEL_FLAG_MASK = (
30#define _(a, b, c) TUNNEL_FLAG_##a |
31 foreach_tunnel_flag
32#undef _
33 0);
Neale Ranns59ff9182019-12-29 23:55:18 +000034
35u8 *
36format_tunnel_mode (u8 * s, va_list * args)
37{
38 tunnel_mode_t mode = va_arg (*args, int);
39
40 switch (mode)
41 {
42#define _(n, v) case TUNNEL_MODE_##n: \
43 s = format (s, "%s", v); \
44 break;
45 foreach_tunnel_mode
46#undef _
47 }
48
49 return (s);
50}
51
Neale Ranns14053c92019-12-29 23:55:18 +000052uword
53unformat_tunnel_mode (unformat_input_t * input, va_list * args)
54{
55 tunnel_mode_t *m = va_arg (*args, tunnel_mode_t *);
56
57 if (unformat (input, "p2p"))
58 *m = TUNNEL_MODE_P2P;
59 else if (unformat (input, "p2mp") || unformat (input, "mp"))
60 *m = TUNNEL_MODE_MP;
61 else
62 return 0;
63 return 1;
64}
65
Neale Ranns59ff9182019-12-29 23:55:18 +000066u8 *
67format_tunnel_encap_decap_flags (u8 * s, va_list * args)
68{
Neale Ranns0a3160b2022-03-04 13:44:47 +000069 tunnel_encap_decap_flags_t f = va_arg (*args, u32);
Neale Ranns59ff9182019-12-29 23:55:18 +000070
71 if (f == TUNNEL_ENCAP_DECAP_FLAG_NONE)
Neale Rannsa91cb452021-02-04 11:02:52 +000072 s = format (s, "none");
Neale Ranns0a3160b2022-03-04 13:44:47 +000073 else
74 {
Neale Rannsa91cb452021-02-04 11:02:52 +000075#define _(a, b, c) \
Neale Ranns0a3160b2022-03-04 13:44:47 +000076 if (f & TUNNEL_ENCAP_DECAP_FLAG_##a) \
77 s = format (s, "%s ", b);
78 foreach_tunnel_encap_decap_flag
Neale Ranns59ff9182019-12-29 23:55:18 +000079#undef _
Neale Ranns0a3160b2022-03-04 13:44:47 +000080 }
81 return (s);
Neale Ranns59ff9182019-12-29 23:55:18 +000082}
83
Mohammed Hawari59b792f2020-12-01 11:30:57 +010084uword
85unformat_tunnel_encap_decap_flags (unformat_input_t * input, va_list * args)
86{
87 tunnel_encap_decap_flags_t *f =
88 va_arg (*args, tunnel_encap_decap_flags_t *);
89#define _(a,b,c) if (unformat(input, b)) {\
90 *f |= TUNNEL_ENCAP_DECAP_FLAG_##a;\
91 return 1;\
92 }
Neale Rannsa91cb452021-02-04 11:02:52 +000093 foreach_tunnel_encap_decap_flag;
Mohammed Hawari59b792f2020-12-01 11:30:57 +010094#undef _
95 return 0;
96}
97
Neale Rannsa91cb452021-02-04 11:02:52 +000098u8 *
99format_tunnel_flags (u8 *s, va_list *args)
100{
Neale Ranns0a3160b2022-03-04 13:44:47 +0000101 tunnel_flags_t f = va_arg (*args, u32);
Neale Rannsa91cb452021-02-04 11:02:52 +0000102
103 if (f == TUNNEL_FLAG_NONE)
104 s = format (s, "none");
Neale Ranns0a3160b2022-03-04 13:44:47 +0000105 else
106 {
107#define _(a, b, c) \
108 if (f & TUNNEL_FLAG_##a) \
109 s = format (s, "%s ", c);
110 foreach_tunnel_flag
Neale Rannsa91cb452021-02-04 11:02:52 +0000111#undef _
Neale Ranns0a3160b2022-03-04 13:44:47 +0000112 }
113 return (s);
Neale Rannsa91cb452021-02-04 11:02:52 +0000114}
115
116uword
117unformat_tunnel_flags (unformat_input_t *input, va_list *args)
118{
119 tunnel_flags_t *f = va_arg (*args, tunnel_flags_t *);
120#define _(a, b, c) \
121 if (unformat (input, c)) \
122 { \
123 *f |= TUNNEL_FLAG_##a; \
124 return 1; \
125 }
126 foreach_tunnel_flag;
127#undef _
128 return 0;
129}
130
131ip_address_family_t
132tunnel_get_af (const tunnel_t *t)
133{
134 return (ip_addr_version (&t->t_src));
135}
136
137void
138tunnel_copy (const tunnel_t *src, tunnel_t *dst)
139{
140 ip_address_copy (&dst->t_dst, &src->t_dst);
141 ip_address_copy (&dst->t_src, &src->t_src);
142
143 dst->t_encap_decap_flags = src->t_encap_decap_flags;
144 dst->t_flags = src->t_flags;
145 dst->t_mode = src->t_mode;
146 dst->t_table_id = src->t_table_id;
147 dst->t_dscp = src->t_dscp;
148 dst->t_hop_limit = src->t_hop_limit;
149 dst->t_fib_index = src->t_fib_index;
150
151 dst->t_flags &= ~TUNNEL_FLAG_RESOLVED;
152 dst->t_fib_entry_index = FIB_NODE_INDEX_INVALID;
153 dst->t_sibling = ~0;
154}
155
156u8 *
157format_tunnel (u8 *s, va_list *args)
158{
159 const tunnel_t *t = va_arg (*args, tunnel_t *);
160 u32 indent = va_arg (*args, u32);
161
162 s = format (s, "%Utable-ID:%d [%U->%U] hop-limit:%d %U %U [%U] [%U]",
163 format_white_space, indent, t->t_table_id, format_ip_address,
164 &t->t_src, format_ip_address, &t->t_dst, t->t_hop_limit,
165 format_tunnel_mode, t->t_mode, format_ip_dscp, t->t_dscp,
166 format_tunnel_flags, t->t_flags, format_tunnel_encap_decap_flags,
167 t->t_encap_decap_flags);
168 if (t->t_flags & TUNNEL_FLAG_RESOLVED)
169 s = format (s, " [resolved via fib-entry: %d]", t->t_fib_entry_index);
170
171 return (s);
172}
173
174uword
175unformat_tunnel (unformat_input_t *input, va_list *args)
176{
177 tunnel_t *t = va_arg (*args, tunnel_t *);
178
179 if (!unformat (input, "tunnel"))
180 return (0);
181
182 unformat (input, "src %U", unformat_ip_address, &t->t_src);
183 unformat (input, "dst %U", unformat_ip_address, &t->t_dst);
Neale Ranns9ec846c2021-02-09 14:04:02 +0000184 unformat (input, "table-id %d", &t->t_table_id);
185 unformat (input, "hop-limit %d", &t->t_hop_limit);
Neale Rannsa91cb452021-02-04 11:02:52 +0000186 unformat (input, "%U", unformat_ip_dscp, &t->t_dscp);
187 unformat (input, "%U", unformat_tunnel_encap_decap_flags,
188 &t->t_encap_decap_flags);
189 unformat (input, "%U", unformat_tunnel_flags, &t->t_flags);
190 unformat (input, "%U", unformat_tunnel_mode, &t->t_mode);
191
Neale Rannsa91cb452021-02-04 11:02:52 +0000192 return (1);
193}
194
195int
196tunnel_resolve (tunnel_t *t, fib_node_type_t child_type, index_t child_index)
197{
198 fib_prefix_t pfx;
199
200 ip_address_to_fib_prefix (&t->t_dst, &pfx);
201
202 t->t_fib_index = fib_table_find (pfx.fp_proto, t->t_table_id);
203
204 if (t->t_fib_index == ~((u32) 0))
205 return VNET_API_ERROR_NO_SUCH_FIB;
206
207 t->t_fib_entry_index = fib_entry_track (t->t_fib_index, &pfx, child_type,
208 child_index, &t->t_sibling);
209
210 t->t_flags |= TUNNEL_FLAG_RESOLVED;
211
212 return (0);
213}
214
215void
216tunnel_unresolve (tunnel_t *t)
217{
218 if (t->t_flags & TUNNEL_FLAG_RESOLVED)
219 fib_entry_untrack (t->t_fib_entry_index, t->t_sibling);
220
221 t->t_flags &= ~TUNNEL_FLAG_RESOLVED;
222}
223
224void
225tunnel_contribute_forwarding (const tunnel_t *t, dpo_id_t *dpo)
226{
227 fib_forward_chain_type_t fct;
228
229 fct = fib_forw_chain_type_from_fib_proto (
230 ip_address_family_to_fib_proto (ip_addr_version (&t->t_src)));
231
232 fib_entry_contribute_forwarding (t->t_fib_entry_index, fct, dpo);
233}
234
235void
236tunnel_build_v6_hdr (const tunnel_t *t, ip_protocol_t next_proto,
237 ip6_header_t *ip)
238{
239 ip->ip_version_traffic_class_and_flow_label =
240 clib_host_to_net_u32 (0x60000000);
241 ip6_set_dscp_network_order (ip, t->t_dscp);
242
243 ip->hop_limit = 254;
244 ip6_address_copy (&ip->src_address, &ip_addr_v6 (&t->t_src));
245 ip6_address_copy (&ip->dst_address, &ip_addr_v6 (&t->t_dst));
246
247 ip->protocol = next_proto;
248 ip->hop_limit = (t->t_hop_limit == 0 ? 254 : t->t_hop_limit);
249 ip6_set_flow_label_network_order (
250 ip, ip6_compute_flow_hash (ip, IP_FLOW_HASH_DEFAULT));
251}
252
253void
254tunnel_build_v4_hdr (const tunnel_t *t, ip_protocol_t next_proto,
255 ip4_header_t *ip)
256{
257 ip->ip_version_and_header_length = 0x45;
258 ip->ttl = (t->t_hop_limit == 0 ? 254 : t->t_hop_limit);
259 ip->src_address.as_u32 = t->t_src.ip.ip4.as_u32;
260 ip->dst_address.as_u32 = t->t_dst.ip.ip4.as_u32;
261 ip->tos = t->t_dscp << 2;
262 ip->protocol = next_proto;
263 ip->checksum = ip4_header_checksum (ip);
264}
Neale Ranns59ff9182019-12-29 23:55:18 +0000265
266/*
267 * fd.io coding-style-patch-verification: ON
268 *
269 * Local Variables:
270 * eval: (c-set-style "gnu")
271 * End:
272 */