blob: 261dd6e1ea09752b7806b799182fb5e01202d184 [file] [log] [blame]
Neale Ranns2dd68522017-02-16 03:38:59 -08001/*
2 * dhcp_proxy.h: DHCP v4 & v6 proxy common functions/types
3 *
4 * Copyright (c) 2013 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#ifndef included_dhcp_proxy_h
19#define included_dhcp_proxy_h
20
21#include <vnet/vnet.h>
22#include <vnet/dhcp/dhcp4_packet.h>
23#include <vnet/ethernet/ethernet.h>
24#include <vnet/ip/ip.h>
25#include <vnet/ip/ip4.h>
26#include <vnet/ip/ip4_packet.h>
27#include <vnet/pg/pg.h>
28#include <vnet/ip/format.h>
Dave Barach68b0fb02017-02-28 15:15:56 -050029#include <vnet/udp/udp.h>
Neale Ranns2dd68522017-02-16 03:38:59 -080030
khemendra kumar34719e32017-12-08 18:06:52 +053031typedef enum
32{
Neale Ranns2dd68522017-02-16 03:38:59 -080033#define dhcp_proxy_error(n,s) DHCP_PROXY_ERROR_##n,
34#include <vnet/dhcp/dhcp4_proxy_error.def>
35#undef dhcp_proxy_error
36 DHCP_PROXY_N_ERROR,
37} dhcp_proxy_error_t;
38
khemendra kumar34719e32017-12-08 18:06:52 +053039typedef enum
40{
Neale Ranns2dd68522017-02-16 03:38:59 -080041#define dhcpv6_proxy_error(n,s) DHCPV6_PROXY_ERROR_##n,
42#include <vnet/dhcp/dhcp6_proxy_error.def>
43#undef dhcpv6_proxy_error
44 DHCPV6_PROXY_N_ERROR,
45} dhcpv6_proxy_error_t;
46
Matthew Smithd4bdd932019-03-27 12:30:29 -050047/* flags to indicate which DHCP ports should be or have been registered */
48typedef enum
49{
50 DHCP_PORT_REG_CLIENT = 0x1,
51 DHCP_PORT_REG_SERVER = 0x2,
52} dhcp_port_reg_flags_t;
Neale Ranns2dd68522017-02-16 03:38:59 -080053
54/**
55 * @brief The Virtual Sub-net Selection information for a given RX FIB
56 */
khemendra kumar34719e32017-12-08 18:06:52 +053057typedef struct dhcp_vss_t_
58{
Neale Ranns2dd68522017-02-16 03:38:59 -080059 /**
John Lo70bfcaf2017-11-14 13:19:26 -050060 * @brief VSS type as defined in RFC 6607:
61 * 0 for NVT ASCII VPN Identifier
62 * 1 for RFC 2685 VPN-ID of 7 octects - 3 bytes OUI & 4 bytes VPN index
63 * 255 for global default VPN
Neale Ranns2dd68522017-02-16 03:38:59 -080064 */
khemendra kumar34719e32017-12-08 18:06:52 +053065 u8 vss_type;
John Lo70bfcaf2017-11-14 13:19:26 -050066#define VSS_TYPE_ASCII 0
67#define VSS_TYPE_VPN_ID 1
68#define VSS_TYPE_INVALID 123
69#define VSS_TYPE_DEFAULT 255
Neale Ranns2dd68522017-02-16 03:38:59 -080070 /**
John Lo70bfcaf2017-11-14 13:19:26 -050071 * @brief Type 1 VPN-ID
Neale Ranns2dd68522017-02-16 03:38:59 -080072 */
khemendra kumar34719e32017-12-08 18:06:52 +053073 u8 vpn_id[7];
John Lo70bfcaf2017-11-14 13:19:26 -050074 /**
75 * @brief Type 0 ASCII VPN Identifier
76 */
khemendra kumar34719e32017-12-08 18:06:52 +053077 u8 *vpn_ascii_id;
Neale Ranns2dd68522017-02-16 03:38:59 -080078} dhcp_vss_t;
79
80/**
Neale Ranns3466c302017-02-16 07:45:03 -080081 * @brief A representation of a single DHCP Server within a given VRF config
Neale Ranns2dd68522017-02-16 03:38:59 -080082 */
Neale Ranns3466c302017-02-16 07:45:03 -080083typedef struct dhcp_server_t_
84{
Neale Ranns2dd68522017-02-16 03:38:59 -080085 /**
86 * @brief The address of the DHCP server to which to relay the client's
87 * messages
88 */
khemendra kumar34719e32017-12-08 18:06:52 +053089 ip46_address_t dhcp_server;
Neale Ranns2dd68522017-02-16 03:38:59 -080090
91 /**
Neale Ranns2dd68522017-02-16 03:38:59 -080092 * @brief The FIB index (not the external Table-ID) in which the server
93 * is reachable.
94 */
khemendra kumar34719e32017-12-08 18:06:52 +053095 u32 server_fib_index;
Neale Ranns3466c302017-02-16 07:45:03 -080096} dhcp_server_t;
97
98/**
99 * @brief A DHCP proxy represenation fpr per-client VRF config
100 */
khemendra kumar34719e32017-12-08 18:06:52 +0530101typedef struct dhcp_proxy_t_
102{
Neale Ranns3466c302017-02-16 07:45:03 -0800103 /**
104 * @brief The set of DHCP servers to which messages are relayed.
105 * If multiple servers are configured then discover/solict messages
106 * are relayed to each. A cookie is maintained for the relay, and only
107 * one message is replayed to the client, based on the presence of the
108 * cookie.
109 * The expectation is there are only 1 or 2 servers, hence no fancy DB.
110 */
khemendra kumar34719e32017-12-08 18:06:52 +0530111 dhcp_server_t *dhcp_servers;
Neale Ranns3466c302017-02-16 07:45:03 -0800112
113 /**
114 * @brief Hash table of pending requets key'd on the clients MAC address
115 */
khemendra kumar34719e32017-12-08 18:06:52 +0530116 uword *dhcp_pending;
Neale Ranns3466c302017-02-16 07:45:03 -0800117
118 /**
119 * @brief A lock for the pending request DB.
120 */
khemendra kumar34719e32017-12-08 18:06:52 +0530121 int lock;
Neale Ranns3466c302017-02-16 07:45:03 -0800122
123 /**
124 * @brief The source address to use in relayed messaes
125 */
khemendra kumar34719e32017-12-08 18:06:52 +0530126 ip46_address_t dhcp_src_address;
Neale Ranns2dd68522017-02-16 03:38:59 -0800127
128 /**
129 * @brief The FIB index (not the external Table-ID) in which the client
130 * is resides.
131 */
khemendra kumar34719e32017-12-08 18:06:52 +0530132 u32 rx_fib_index;
Neale Ranns3466c302017-02-16 07:45:03 -0800133} dhcp_proxy_t;
Neale Ranns2dd68522017-02-16 03:38:59 -0800134
135#define DHCP_N_PROTOS (FIB_PROTOCOL_IP6 + 1)
136
137/**
138 * @brief Collection of global DHCP proxy data
139 */
khemendra kumar34719e32017-12-08 18:06:52 +0530140typedef struct
141{
Neale Ranns2dd68522017-02-16 03:38:59 -0800142 /* Pool of DHCP servers */
Neale Ranns3466c302017-02-16 07:45:03 -0800143 dhcp_proxy_t *dhcp_servers[DHCP_N_PROTOS];
Neale Ranns2dd68522017-02-16 03:38:59 -0800144
145 /* Pool of selected DHCP server. Zero is the default server */
khemendra kumar34719e32017-12-08 18:06:52 +0530146 u32 *dhcp_server_index_by_rx_fib_index[DHCP_N_PROTOS];
Neale Ranns2dd68522017-02-16 03:38:59 -0800147
148 /* to drop pkts in server-to-client direction */
149 u32 error_drop_node_index;
150
151 dhcp_vss_t *vss[DHCP_N_PROTOS];
152
153 /* hash lookup specific vrf_id -> option 82 vss suboption */
154 u32 *vss_index_by_rx_fib_index[DHCP_N_PROTOS];
Dave Barach8a9566e2018-10-23 10:47:36 -0400155
Matthew Smithd4bdd932019-03-27 12:30:29 -0500156 /* flags to indicate which udp ports have been registered */
Dave Barach8a9566e2018-10-23 10:47:36 -0400157 int udp_ports_registered;
158
159 /* convenience */
160 vlib_main_t *vlib_main;
161
Neale Ranns2dd68522017-02-16 03:38:59 -0800162} dhcp_proxy_main_t;
163
164extern dhcp_proxy_main_t dhcp_proxy_main;
165
166/**
Matthew Smithd4bdd932019-03-27 12:30:29 -0500167 * @brief Register the dhcp client and/or server ports, if not already done
Dave Barach8a9566e2018-10-23 10:47:36 -0400168 */
Matthew Smithd4bdd932019-03-27 12:30:29 -0500169void dhcp_maybe_register_udp_ports (dhcp_port_reg_flags_t ports);
Dave Barach8a9566e2018-10-23 10:47:36 -0400170
171/**
Neale Ranns2dd68522017-02-16 03:38:59 -0800172 * @brief Send the details of a proxy session to the API client during a dump
173 */
174void dhcp_send_details (fib_protocol_t proto,
khemendra kumar34719e32017-12-08 18:06:52 +0530175 void *opaque, u32 context, dhcp_proxy_t * proxy);
Neale Ranns2dd68522017-02-16 03:38:59 -0800176
177/**
178 * @brief Show (on CLI) a VSS config during a show walk
179 */
khemendra kumar34719e32017-12-08 18:06:52 +0530180int dhcp_vss_show_walk (dhcp_vss_t * vss, u32 rx_table_id, void *ctx);
Neale Ranns2dd68522017-02-16 03:38:59 -0800181
182/**
183 * @brief Configure/set a new VSS info
184 */
John Lo70bfcaf2017-11-14 13:19:26 -0500185int dhcp_proxy_set_vss (fib_protocol_t proto,
khemendra kumar34719e32017-12-08 18:06:52 +0530186 u32 tbl_id,
John Lo70bfcaf2017-11-14 13:19:26 -0500187 u8 vss_type,
khemendra kumar34719e32017-12-08 18:06:52 +0530188 u8 * vpn_ascii_id, u32 oui, u32 vpn_index, u8 is_del);
Neale Ranns2dd68522017-02-16 03:38:59 -0800189
190/**
191 * @brief Dump the proxy configs to the API
192 */
khemendra kumar34719e32017-12-08 18:06:52 +0530193void dhcp_proxy_dump (fib_protocol_t proto, void *opaque, u32 context);
Neale Ranns2dd68522017-02-16 03:38:59 -0800194
195/**
196 * @brief Add a new DHCP proxy server configuration.
197 * @return 1 is the config is new,
198 * 0 otherwise (implying a modify of an existing)
199 */
khemendra kumar34719e32017-12-08 18:06:52 +0530200int dhcp_proxy_server_add (fib_protocol_t proto,
201 ip46_address_t * addr,
202 ip46_address_t * src_address,
203 u32 rx_fib_iindex, u32 server_table_id);
Neale Ranns2dd68522017-02-16 03:38:59 -0800204
205/**
206 * @brief Delete a DHCP proxy config
Neale Ranns3466c302017-02-16 07:45:03 -0800207 * @return 1 if the proxy is deleted, 0 otherwise
Neale Ranns2dd68522017-02-16 03:38:59 -0800208 */
khemendra kumar34719e32017-12-08 18:06:52 +0530209int dhcp_proxy_server_del (fib_protocol_t proto,
210 u32 rx_fib_index,
211 ip46_address_t * addr, u32 server_table_id);
Neale Ranns3466c302017-02-16 07:45:03 -0800212
khemendra kumar34719e32017-12-08 18:06:52 +0530213u32 dhcp_proxy_rx_table_get_table_id (fib_protocol_t proto, u32 fib_index);
Neale Ranns2dd68522017-02-16 03:38:59 -0800214
215/**
216 * @brief Callback function invoked for each DHCP proxy entry
217 * return 0 to break the walk, non-zero otherwise.
218 */
khemendra kumar34719e32017-12-08 18:06:52 +0530219typedef int (*dhcp_proxy_walk_fn_t) (dhcp_proxy_t * server, void *ctx);
Neale Ranns2dd68522017-02-16 03:38:59 -0800220
221/**
222 * @brief Walk/Visit each DHCP proxy server
223 */
khemendra kumar34719e32017-12-08 18:06:52 +0530224void dhcp_proxy_walk (fib_protocol_t proto,
225 dhcp_proxy_walk_fn_t fn, void *ctx);
Neale Ranns2dd68522017-02-16 03:38:59 -0800226
227/**
228 * @brief Callback function invoked for each DHCP VSS entry
229 * return 0 to break the walk, non-zero otherwise.
230 */
khemendra kumar34719e32017-12-08 18:06:52 +0530231typedef int (*dhcp_vss_walk_fn_t) (dhcp_vss_t * server,
232 u32 rx_table_id, void *ctx);
Neale Ranns2dd68522017-02-16 03:38:59 -0800233
234/**
235 * @brief Walk/Visit each DHCP proxy VSS
236 */
khemendra kumar34719e32017-12-08 18:06:52 +0530237void dhcp_vss_walk (fib_protocol_t proto, dhcp_vss_walk_fn_t fn, void *ctx);
Neale Ranns2dd68522017-02-16 03:38:59 -0800238
239/**
Neale Ranns3466c302017-02-16 07:45:03 -0800240 * @brief Lock a proxy object to prevent simultaneous access of its
241 * pending store
242 */
khemendra kumar34719e32017-12-08 18:06:52 +0530243void dhcp_proxy_lock (dhcp_proxy_t * server);
Neale Ranns3466c302017-02-16 07:45:03 -0800244
245/**
246 * @brief Lock a proxy object to prevent simultaneous access of its
247 * pending store
248 */
khemendra kumar34719e32017-12-08 18:06:52 +0530249void dhcp_proxy_unlock (dhcp_proxy_t * server);
Neale Ranns3466c302017-02-16 07:45:03 -0800250
251/**
Neale Ranns2dd68522017-02-16 03:38:59 -0800252 * @brief Get the VSS data for the FIB index
253 */
254static inline dhcp_vss_t *
khemendra kumar34719e32017-12-08 18:06:52 +0530255dhcp_get_vss_info (dhcp_proxy_main_t * dm,
256 u32 rx_fib_index, fib_protocol_t proto)
Neale Ranns2dd68522017-02-16 03:38:59 -0800257{
258 dhcp_vss_t *v = NULL;
259
khemendra kumar34719e32017-12-08 18:06:52 +0530260 if (vec_len (dm->vss_index_by_rx_fib_index[proto]) > rx_fib_index &&
Neale Ranns2dd68522017-02-16 03:38:59 -0800261 dm->vss_index_by_rx_fib_index[proto][rx_fib_index] != ~0)
khemendra kumar34719e32017-12-08 18:06:52 +0530262 {
263 v = pool_elt_at_index (dm->vss[proto],
264 dm->vss_index_by_rx_fib_index[proto]
265 [rx_fib_index]);
266 }
Neale Ranns2dd68522017-02-16 03:38:59 -0800267
268 return (v);
269}
270
271/**
272 * @brief Get the DHCP proxy server data for the FIB index
273 */
Neale Ranns3466c302017-02-16 07:45:03 -0800274static inline dhcp_proxy_t *
khemendra kumar34719e32017-12-08 18:06:52 +0530275dhcp_get_proxy (dhcp_proxy_main_t * dm,
276 u32 rx_fib_index, fib_protocol_t proto)
Neale Ranns2dd68522017-02-16 03:38:59 -0800277{
Neale Ranns3466c302017-02-16 07:45:03 -0800278 dhcp_proxy_t *s = NULL;
Neale Ranns2dd68522017-02-16 03:38:59 -0800279
khemendra kumar34719e32017-12-08 18:06:52 +0530280 if (vec_len (dm->dhcp_server_index_by_rx_fib_index[proto]) > rx_fib_index &&
Neale Ranns2dd68522017-02-16 03:38:59 -0800281 dm->dhcp_server_index_by_rx_fib_index[proto][rx_fib_index] != ~0)
khemendra kumar34719e32017-12-08 18:06:52 +0530282 {
283 s = pool_elt_at_index (dm->dhcp_servers[proto],
284 dm->dhcp_server_index_by_rx_fib_index[proto]
285 [rx_fib_index]);
286 }
Neale Ranns2dd68522017-02-16 03:38:59 -0800287
288 return (s);
289}
290
khemendra kumar34719e32017-12-08 18:06:52 +0530291int dhcp6_proxy_set_server (ip46_address_t * addr,
292 ip46_address_t * src_addr,
293 u32 rx_table_id, u32 server_table_id, int is_del);
294int dhcp4_proxy_set_server (ip46_address_t * addr,
295 ip46_address_t * src_addr,
296 u32 rx_table_id, u32 server_table_id, int is_del);
Neale Ranns2dd68522017-02-16 03:38:59 -0800297
298#endif /* included_dhcp_proxy_h */
khemendra kumar34719e32017-12-08 18:06:52 +0530299
300/*
301 * fd.io coding-style-patch-verification: ON
302 *
303 * Local Variables:
304 * eval: (c-set-style "gnu")
305 * End:
306 */