aboutsummaryrefslogtreecommitdiffstats
path: root/src/vnet/ipip/ipip.h
blob: fef5aab049032f3cc0b532c0acd49d1f442156f9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
/*
 * ipip.h: types/functions for ipip.
 *
 * Copyright (c) 2018 Cisco and/or its affiliates.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at:
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or aipiped to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef included_ipip_h
#define included_ipip_h

#include <vnet/adj/adj_types.h>
#include <vnet/ip/ip6_packet.h>
#include <vnet/ip/format.h>
#include <vnet/ip/ip.h>
#include <vnet/tunnel/tunnel.h>

extern vnet_hw_interface_class_t ipip_hw_interface_class;

#define foreach_ipip_error				\
  /* Must be first. */					\
  _(DECAP_PKTS, "packets decapsulated")			\
  _(BAD_PROTOCOL, "bad protocol")			\
  _(NO_TUNNEL, "no tunnel")				\
  _(FRAGMENTED_PACKET, "fragmented outer packet")

typedef enum
{
#define _(sym, str) IPIP_ERROR_##sym,
  foreach_ipip_error
#undef _
    IPIP_N_ERROR,
} ipip_error_t;

/**
 * @brief IPIP Tunnel key
 */
typedef enum
{
  IPIP_TRANSPORT_IP4,
  IPIP_TRANSPORT_IP6,
} __clib_packed ipip_transport_t;

typedef enum
{
  IPIP_MODE_P2P = 0,
  IPIP_MODE_P2MP,
  IPIP_MODE_6RD,
} __clib_packed ipip_mode_t;

typedef struct
{
  ip46_address_t src;
  ip46_address_t dst;
  u32 fib_index;
  ipip_transport_t transport;
  ipip_mode_t mode;
  u16 __pad;
} __clib_packed ipip_tunnel_key_t;

STATIC_ASSERT_SIZEOF (ipip_tunnel_key_t, 5 * sizeof (u64));

/**
 * @brief A representation of a IPIP tunnel
 */
typedef struct
{
  /* Required for pool_get_aligned */
  CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);

  ipip_mode_t mode;
  ipip_transport_t transport;
  ip46_address_t tunnel_src;
  ip46_address_t tunnel_dst;
  u32 fib_index;
  u32 hw_if_index;
  u32 sw_if_index;
  u32 dev_instance;		/* Real device instance in tunnel vector */
  u32 user_instance;		/* Instance name being shown to user */
  tunnel_encap_decap_flags_t flags;
  ip_dscp_t dscp;

  struct
  {
    ip6_address_t ip6_prefix;
    ip4_address_t ip4_prefix;
    u8 ip6_prefix_len;
    u8 ip4_prefix_len;
    u8 shift;
    bool security_check;
    u32 ip6_fib_index;
  } sixrd;
} ipip_tunnel_t;

typedef struct
{
  ipip_tunnel_t *tunnels;
  uword *tunnel_by_key;
  u32 *tunnel_index_by_sw_if_index;

  /* convenience */
  vlib_main_t *vlib_main;
  vnet_main_t *vnet_main;

  /* Record used instances */
  uword *instance_used;

  bool ip4_protocol_registered;
  bool ip6_protocol_registered;

  u16 msg_id_base;
} ipip_main_t;

extern ipip_main_t ipip_main;
extern vlib_node_registration_t ipip4_input_node;
extern vlib_node_registration_t ipip6_input_node;

/*
 * sixrd_get_addr_net
 */
static_always_inline u32
sixrd_get_addr_net (const ipip_tunnel_t * t, u64 dal)
{
  /* 1:1 mode */
  if (t->sixrd.ip4_prefix_len == 32)
    return (t->sixrd.ip4_prefix.as_u32);

  dal = clib_net_to_host_u64 (dal);

  /* Grab 32 - ip4_prefix_len bits out of IPv6 address from offset
   * ip6_prefix_len */
  u32 mask = ~(~0ULL << (32 - t->sixrd.ip4_prefix_len));
  u32 ip4 =
    clib_net_to_host_u32 (t->sixrd.
			  ip4_prefix.as_u32) | ((u32) (dal >> t->sixrd.
						       shift) & mask);
  return clib_host_to_net_u32 (ip4);
}

int ipip_add_tunnel (ipip_transport_t transport, u32 instance,
		     ip46_address_t * src, ip46_address_t * dst,
		     u32 fib_index, tunnel_encap_decap_flags_t flags,
		     ip_dscp_t dscp, tunnel_mode_t mode, u32 * sw_if_indexp);
int ipip_del_tunnel (u32 sw_if_index);
int sixrd_add_tunnel (ip6_address_t * ip6_prefix, u8 ip6_prefix_len,
		      ip4_address_t * ip4_prefix, u8 ip4_prefix_len,
		      ip4_address_t * ip4_src, bool security_check,
		      u32 ip4_fib_index, u32 ip6_fib_index,
		      u32 * sw_if_index);
int sixrd_del_tunnel (u32 sw_if_index);
void ipip_tunnel_db_add (ipip_tunnel_t * t, const ipip_tunnel_key_t * key);
void ipip_tunnel_db_remove (ipip_tunnel_t * t, const ipip_tunnel_key_t * key);
ipip_tunnel_t *ipip_tunnel_db_find (const ipip_tunnel_key_t * key);
ipip_tunnel_t *ipip_tunnel_db_find_by_sw_if_index (u32 sw_if_index);
void ipip_mk_key (const ipip_tunnel_t * t, ipip_tunnel_key_t * key);
void ipip_mk_key_i (ipip_transport_t transport,
		    ipip_mode_t mode,
		    const ip46_address_t * src,
		    const ip46_address_t * dst,
		    u32 fib_index, ipip_tunnel_key_t * key);

#endif

/*
 * fd.io coding-style-patch-verification: ON
 *
 * Local Variables:
 * eval: (c-set-style "gnu")
 * End:
 */
an class="p">); } static void add_version_tlv (vnet_hw_interface_t * hw, u8 ** t0p) { cdp_tlv_t *t = (cdp_tlv_t *) * t0p; t->t = htons (CDP_TLV_version); t->l = htons (12 + sizeof (*t)); clib_memcpy (&t->v, "VPP Software", 12); *t0p += ntohs (t->l); } static void add_platform_tlv (vnet_hw_interface_t * hw, u8 ** t0p) { cdp_tlv_t *t = (cdp_tlv_t *) * t0p; t->t = htons (CDP_TLV_platform); t->l = htons (2 + sizeof (*t)); clib_memcpy (&t->v, "SW", 2); *t0p += ntohs (t->l); } static void add_capability_tlv (vnet_hw_interface_t * hw, u8 ** t0p) { cdp_tlv_t *t = (cdp_tlv_t *) * t0p; u32 capabilities; t->t = htons (CDP_TLV_capabilities); t->l = htons (4 + sizeof (*t)); capabilities = CDP_ROUTER_DEVICE; capabilities = htonl (capabilities); clib_memcpy (&t->v, &capabilities, sizeof (capabilities)); *t0p += ntohs (t->l); } static void add_tlvs (cdp_main_t * cm, vnet_hw_interface_t * hw, u8 ** t0p) { add_device_name_tlv (hw, t0p); add_port_id_tlv (hw, t0p); add_version_tlv (hw, t0p); add_platform_tlv (hw, t0p); add_capability_tlv (hw, t0p); } /* * send a cdp pkt on an ethernet interface */ static void send_ethernet_hello (cdp_main_t * cm, cdp_neighbor_t * n, int count) { u32 *to_next; ethernet_llc_snap_and_cdp_header_t *h0; vnet_hw_interface_t *hw; u32 bi0; vlib_buffer_t *b0; u8 *t0; u16 checksum; int nbytes_to_checksum; int i; vlib_frame_t *f; vlib_main_t *vm = cm->vlib_main; vnet_main_t *vnm = cm->vnet_main; for (i = 0; i < count; i++) { /* * see cdp_periodic_init() to understand what's already painted * into the buffer by the packet template mechanism */ h0 = vlib_packet_template_get_packet (vm, &cm->packet_templates[n->packet_template_index], &bi0); if (!h0) break; /* Add the interface's ethernet source address */ hw = vnet_get_sup_hw_interface (vnm, n->sw_if_index); clib_memcpy (h0->ethernet.src_address, hw->hw_address, vec_len (hw->hw_address)); t0 = (u8 *) & h0->cdp.data; /* add TLVs */ add_tlvs (cm, hw, &t0); /* add the cdp packet checksum */ nbytes_to_checksum = t0 - (u8 *) & h0->cdp; checksum = cdp_checksum (&h0->cdp, nbytes_to_checksum); h0->cdp.checksum = htons (checksum); /* Set the outbound packet length */ b0 = vlib_get_buffer (vm, bi0); b0->current_length = nbytes_to_checksum + sizeof (*h0) - sizeof (cdp_hdr_t); /* And the outbound interface */ vnet_buffer (b0)->sw_if_index[VLIB_TX] = hw->sw_if_index; /* Set the 802.3 ethernet length */ h0->ethernet.len = htons (b0->current_length - sizeof (ethernet_802_3_header_t)); /* And output the packet on the correct interface */ f = vlib_get_frame_to_node (vm, hw->output_node_index); to_next = vlib_frame_vector_args (f); to_next[0] = bi0; f->n_vectors = 1; vlib_put_frame_to_node (vm, hw->output_node_index, f); n->last_sent = vlib_time_now (vm); } } /* * send a cdp pkt on an hdlc interface */ static void send_hdlc_hello (cdp_main_t * cm, cdp_neighbor_t * n, int count) { u32 *to_next; hdlc_and_cdp_header_t *h0; vnet_hw_interface_t *hw; u32 bi0; vlib_buffer_t *b0; u8 *t0; u16 checksum; int nbytes_to_checksum; int i; vlib_frame_t *f; vlib_main_t *vm = cm->vlib_main; vnet_main_t *vnm = cm->vnet_main; for (i = 0; i < count; i++) { /* * see cdp_periodic_init() to understand what's already painted * into the buffer by the packet template mechanism */ h0 = vlib_packet_template_get_packet (vm, &cm->packet_templates[n->packet_template_index], &bi0); if (!h0) break; hw = vnet_get_sup_hw_interface (vnm, n->sw_if_index); t0 = (u8 *) & h0->cdp.data; /* add TLVs */ add_tlvs (cm, hw, &t0); /* add the cdp packet checksum */ nbytes_to_checksum = t0 - (u8 *) & h0->cdp; checksum = cdp_checksum (&h0->cdp, nbytes_to_checksum); h0->cdp.checksum = htons (checksum); /* Set the outbound packet length */ b0 = vlib_get_buffer (vm, bi0); b0->current_length = nbytes_to_checksum + sizeof (*h0) - sizeof (cdp_hdr_t); /* And output the packet on the correct interface */ f = vlib_get_frame_to_node (vm, hw->output_node_index); to_next = vlib_frame_vector_args (f); to_next[0] = bi0; f->n_vectors = 1; vlib_put_frame_to_node (vm, hw->output_node_index, f); n->last_sent = vlib_time_now (vm); } } /* * send a cdp pkt on an srp interface */ static void send_srp_hello (cdp_main_t * cm, cdp_neighbor_t * n, int count) { u32 *to_next; srp_and_cdp_header_t *h0; vnet_hw_interface_t *hw; u32 bi0; vlib_buffer_t *b0; u8 *t0; u16 checksum; int nbytes_to_checksum; int i; vlib_frame_t *f; vlib_main_t *vm = cm->vlib_main; vnet_main_t *vnm = cm->vnet_main; for (i = 0; i < count; i++) { /* * see cdp_periodic_init() to understand what's already painted * into the buffer by the packet template mechanism */ h0 = vlib_packet_template_get_packet (vm, &cm->packet_templates[n->packet_template_index], &bi0); if (!h0) break; hw = vnet_get_sup_hw_interface (vnm, n->sw_if_index); t0 = (u8 *) & h0->cdp.data; /* add TLVs */ add_tlvs (cm, hw, &t0); /* Add the interface's ethernet source address */ clib_memcpy (h0->ethernet.src_address, hw->hw_address, vec_len (hw->hw_address)); /* add the cdp packet checksum */ nbytes_to_checksum = t0 - (u8 *) & h0->cdp; checksum = cdp_checksum (&h0->cdp, nbytes_to_checksum); h0->cdp.checksum = htons (checksum); /* Set the outbound packet length */ b0 = vlib_get_buffer (vm, bi0); b0->current_length = nbytes_to_checksum + sizeof (*h0) - sizeof (cdp_hdr_t); /* And output the packet on the correct interface */ f = vlib_get_frame_to_node (vm, hw->output_node_index); to_next = vlib_frame_vector_args (f); to_next[0] = bi0; f->n_vectors = 1; vlib_put_frame_to_node (vm, hw->output_node_index, f); n->last_sent = vlib_time_now (vm); } } /* * Decide which cdp packet template to use */ static int pick_packet_template (cdp_main_t * cm, cdp_neighbor_t * n) { n->packet_template_index = CDP_PACKET_TEMPLATE_ETHERNET; return 0; } /* Send a cdp neighbor announcement */ static void send_hello (cdp_main_t * cm, cdp_neighbor_t * n, int count) { if (n->packet_template_index == (u8) ~ 0) { /* If we don't know how to talk to this peer, don't try again */ if (pick_packet_template (cm, n)) { n->last_sent = 1e70; return; } } switch (n->packet_template_index) { case CDP_PACKET_TEMPLATE_ETHERNET: send_ethernet_hello (cm, n, count); break; case CDP_PACKET_TEMPLATE_HDLC: send_hdlc_hello (cm, n, count); break; case CDP_PACKET_TEMPLATE_SRP: send_srp_hello (cm, n, count); break; default: ASSERT (0); } n->last_sent = vlib_time_now (cm->vlib_main); } static void delete_neighbor (cdp_main_t * cm, cdp_neighbor_t * n, int want_broadcast) { hash_unset (cm->neighbor_by_sw_if_index, n->sw_if_index); vec_free (n->device_name); vec_free (n->version); vec_free (n->port_id); vec_free (n->platform); vec_free (n->last_rx_pkt); pool_put (cm->neighbors, n); } void cdp_periodic (vlib_main_t * vm) { cdp_main_t *cm = &cdp_main; cdp_neighbor_t *n; f64 now = vlib_time_now (vm); vnet_sw_interface_t *sw; static u32 *delete_list = 0; int i; static cdp_neighbor_t **n_list = 0; /* *INDENT-OFF* */ pool_foreach (n, cm->neighbors, ({ vec_add1 (n_list, n); })); /* *INDENT-ON* */ /* Across all cdp neighbors known to the system */ for (i = 0; i < vec_len (n_list); i++) { n = n_list[i]; /* "no cdp run" provisioned on the interface? */ if (n->disabled == 1) continue; sw = vnet_get_sw_interface (cm->vnet_main, n->sw_if_index); /* Interface shutdown or rx timeout? */ if (!(sw->flags & VNET_SW_INTERFACE_FLAG_ADMIN_UP) || (now > (n->last_heard + (f64) n->ttl_in_seconds))) /* add to list of neighbors to delete */ vec_add1 (delete_list, n - cm->neighbors); else if (n->last_sent == 0.0) /* First time, send 3 hellos */ send_hello (cm, n, 3 /* three to begin with */ ); else if (now > (n->last_sent + (((f64) n->ttl_in_seconds) / 6.0))) /* Normal keepalive, send one */ send_hello (cm, n, 1 /* one as a keepalive */ ); } for (i = 0; i < vec_len (delete_list); i++) { n = vec_elt_at_index (cm->neighbors, delete_list[i]); delete_neighbor (cm, n, 1); } if (delete_list) _vec_len (delete_list) = 0; if (n_list) _vec_len (n_list) = 0; } static clib_error_t * cdp_periodic_init (vlib_main_t * vm) { cdp_main_t *cm = &cdp_main; /* Create the ethernet cdp hello packet template */ { ethernet_llc_snap_and_cdp_header_t h; clib_memset (&h, 0, sizeof (h)); /* Send to 01:00:0c:cc:cc */ h.ethernet.dst_address[0] = 0x01; /* h.ethernet.dst_address[1] = 0x00; (clib_memset) */ h.ethernet.dst_address[2] = 0x0C; h.ethernet.dst_address[3] = 0xCC; h.ethernet.dst_address[4] = 0xCC; h.ethernet.dst_address[5] = 0xCC; /* leave src address blank (fill in at send time) */ /* leave length blank (fill in at send time) */ /* LLC */ h.llc.dst_sap = h.llc.src_sap = 0xAA; /* SNAP */ h.llc.control = 0x03; /* UI (no extended control bytes) */ /* SNAP */ /* h.snap.oui[0] = 0x00; (clib_memset) */ /* h.snap.oui[1] = 0x00; (clib_memset) */ h.snap.oui[2] = 0x0C; /* Cisco = 0x00000C */ h.snap.protocol = htons (0x2000); /* CDP = 0x2000 */ /* CDP */ h.cdp.version = 2; h.cdp.ttl = 180; vlib_packet_template_init (vm, &cm->packet_templates[CDP_PACKET_TEMPLATE_ETHERNET], /* data */ &h, sizeof (h), /* alloc chunk size */ 8, "cdp-ethernet"); } #if 0 /* retain for reference */ /* Create the hdlc cdp hello packet template */ { hdlc_and_cdp_header_t h; clib_memset (&h, 0, sizeof (h)); h.hdlc.address = 0x0f; /* h.hdlc.control = 0; (clib_memset) */ h.hdlc.protocol = htons (0x2000); /* CDP = 0x2000 */ /* CDP */ h.cdp.version = 2; h.cdp.ttl = 180; vlib_packet_template_init (vm, &cm->packet_templates[CDP_PACKET_TEMPLATE_HDLC], /* data */ &h, sizeof (h), /* alloc chunk size */ 8, "cdp-hdlc"); } /* Create the srp cdp hello packet template */ { srp_and_cdp_header_t h; clib_memset (&h, 0, sizeof (h)); /* Send to 01:00:0c:cc:cc */ h.ethernet.dst_address[0] = 0x01; /* h.ethernet.dst_address[1] = 0x00; (clib_memset) */ h.ethernet.dst_address[2] = 0x0C; h.ethernet.dst_address[3] = 0xCC; h.ethernet.dst_address[4] = 0xCC; h.ethernet.dst_address[5] = 0xCC; /* leave src address blank (fill in at send time) */ /* The srp header is filled in at xmt */ h.srp.ttl = 1; h.srp.priority = 7; h.srp.mode = SRP_MODE_data; srp_header_compute_parity (&h.srp); /* Inner ring and parity will be set at send time */ h.ethernet.type = htons (0x2000); /* CDP = 0x2000 */ /* CDP */ h.cdp.version = 2; h.cdp.ttl = 180; vlib_packet_template_init (vm, &cm->packet_templates[CDP_PACKET_TEMPLATE_SRP], /* data */ &h, sizeof (h), /* alloc chunk size */ 8, "cdp-srp"); } #endif return 0; } VLIB_INIT_FUNCTION (cdp_periodic_init); /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */