FD.io VPP  v19.08-27-gf4dcae4
Vector Packet Processing
dhcp_api.c
Go to the documentation of this file.
1 /*
2  *------------------------------------------------------------------
3  * dhcp_api.c - dhcp api
4  *
5  * Copyright (c) 2016 Cisco and/or its affiliates.
6  * Licensed under the Apache License, Version 2.0 (the "License");
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at:
9  *
10  * http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an "AS IS" BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  *------------------------------------------------------------------
18  */
19 
20 #include <vnet/vnet.h>
21 #include <vlibmemory/api.h>
22 
23 #include <vnet/interface.h>
24 #include <vnet/api_errno.h>
25 #include <vnet/dhcp/dhcp_proxy.h>
26 #include <vnet/dhcp/client.h>
30 #include <vnet/fib/fib_table.h>
31 #include <vnet/ip/ip_types_api.h>
32 
33 #include <vnet/vnet_msg_enum.h>
34 
35 #define vl_typedefs /* define message structures */
36 #include <vnet/vnet_all_api_h.h>
37 #undef vl_typedefs
38 
39 #define vl_endianfun /* define message structures */
40 #include <vnet/vnet_all_api_h.h>
41 #undef vl_endianfun
42 
43 /* instantiate all the print functions we know about */
44 #define vl_print(handle, ...) vlib_cli_output (handle, __VA_ARGS__)
45 #define vl_printfun
46 #include <vnet/vnet_all_api_h.h>
47 #undef vl_printfun
48 
50 
51 #define foreach_vpe_api_msg \
52 _(DHCP_PROXY_CONFIG,dhcp_proxy_config) \
53 _(DHCP_PROXY_DUMP,dhcp_proxy_dump) \
54 _(DHCP_PROXY_SET_VSS,dhcp_proxy_set_vss) \
55 _(DHCP_CLIENT_CONFIG, dhcp_client_config) \
56 _(DHCP_CLIENT_DUMP, dhcp_client_dump) \
57 _(WANT_DHCP6_PD_REPLY_EVENTS, want_dhcp6_pd_reply_events) \
58 _(DHCP6_PD_SEND_CLIENT_MESSAGE, dhcp6_pd_send_client_message) \
59 _(WANT_DHCP6_REPLY_EVENTS, want_dhcp6_reply_events) \
60 _(DHCP6_SEND_CLIENT_MESSAGE, dhcp6_send_client_message) \
61 _(DHCP6_CLIENTS_ENABLE_DISABLE, dhcp6_clients_enable_disable) \
62 _(DHCP6_DUID_LL_SET, dhcp6_duid_ll_set)
63 
64 
65 static void
67 {
68  vl_api_dhcp_proxy_set_vss_reply_t *rmp;
69  u8 *vpn_ascii_id;
70  int rv;
71 
72  mp->vpn_ascii_id[sizeof (mp->vpn_ascii_id) - 1] = 0;
73  vpn_ascii_id = format (0, "%s", mp->vpn_ascii_id);
74  rv =
76  ntohl (mp->tbl_id), mp->vss_type, vpn_ascii_id,
77  ntohl (mp->oui), ntohl (mp->vpn_index),
78  mp->is_add == 0);
79 
80  REPLY_MACRO (VL_API_DHCP_PROXY_SET_VSS_REPLY);
81 }
82 
83 
86 {
87  vl_api_dhcp_proxy_set_vss_reply_t *rmp;
88  ip46_address_t src, server;
89  int rv = -1;
90 
91  if (mp->is_ipv6)
92  {
93  clib_memcpy (&src.ip6, mp->dhcp_src_address, sizeof (src.ip6));
94  clib_memcpy (&server.ip6, mp->dhcp_server, sizeof (server.ip6));
95 
96  rv = dhcp6_proxy_set_server (&server,
97  &src,
98  (u32) ntohl (mp->rx_vrf_id),
99  (u32) ntohl (mp->server_vrf_id),
100  (int) (mp->is_add == 0));
101  }
102  else
103  {
104  ip46_address_reset (&src);
105  ip46_address_reset (&server);
106 
107  clib_memcpy (&src.ip4, mp->dhcp_src_address, sizeof (src.ip4));
108  clib_memcpy (&server.ip4, mp->dhcp_server, sizeof (server.ip4));
109 
110  rv = dhcp4_proxy_set_server (&server,
111  &src,
112  (u32) ntohl (mp->rx_vrf_id),
113  (u32) ntohl (mp->server_vrf_id),
114  (int) (mp->is_add == 0));
115  }
116 
117 
118  REPLY_MACRO (VL_API_DHCP_PROXY_CONFIG_REPLY);
119 }
120 
121 static void
123 {
125 
127  if (!reg)
128  return;;
129 
130  dhcp_proxy_dump ((mp->is_ip6 == 1 ?
132 }
133 
134 void
136  void *opaque, u32 context, dhcp_proxy_t * proxy)
137 {
139  vl_api_registration_t *reg = opaque;
140  vl_api_dhcp_server_t *v_server;
141  dhcp_server_t *server;
142  fib_table_t *s_fib;
143  dhcp_vss_t *vss;
144  u32 count;
145  size_t n;
146 
147  count = vec_len (proxy->dhcp_servers);
148  n = sizeof (*mp) + (count * sizeof (vl_api_dhcp_server_t));
149  mp = vl_msg_api_alloc (n);
150  if (!mp)
151  return;
152  clib_memset (mp, 0, n);
153  mp->_vl_msg_id = ntohs (VL_API_DHCP_PROXY_DETAILS);
154  mp->context = context;
155  mp->count = count;
156 
157  mp->is_ipv6 = (proto == FIB_PROTOCOL_IP6);
158  mp->rx_vrf_id =
159  htonl (dhcp_proxy_rx_table_get_table_id (proto, proxy->rx_fib_index));
160 
161  vss = dhcp_get_vss_info (&dhcp_proxy_main, proxy->rx_fib_index, proto);
162 
163  if (vss)
164  {
165  mp->vss_type = vss->vss_type;
166  if (vss->vss_type == VSS_TYPE_ASCII)
167  {
168  u32 id_len = vec_len (vss->vpn_ascii_id);
169  clib_memcpy (mp->vss_vpn_ascii_id, vss->vpn_ascii_id, id_len);
170  }
171  else if (vss->vss_type == VSS_TYPE_VPN_ID)
172  {
173  u32 oui = ((u32) vss->vpn_id[0] << 16) + ((u32) vss->vpn_id[1] << 8)
174  + ((u32) vss->vpn_id[2]);
175  u32 fib_id = ((u32) vss->vpn_id[3] << 24) +
176  ((u32) vss->vpn_id[4] << 16) + ((u32) vss->vpn_id[5] << 8) +
177  ((u32) vss->vpn_id[6]);
178  mp->vss_oui = htonl (oui);
179  mp->vss_fib_id = htonl (fib_id);
180  }
181  }
182  else
184 
185  vec_foreach_index (count, proxy->dhcp_servers)
186  {
187  server = &proxy->dhcp_servers[count];
188  v_server = &mp->servers[count];
189 
190  s_fib = fib_table_get (server->server_fib_index, proto);
191 
192  v_server->server_vrf_id = htonl (s_fib->ft_table_id);
193 
194  if (mp->is_ipv6)
195  {
196  memcpy (v_server->dhcp_server, &server->dhcp_server.ip6, 16);
197  }
198  else
199  {
200  /* put the address in the first bytes */
201  memcpy (v_server->dhcp_server, &server->dhcp_server.ip4, 4);
202  }
203  }
204 
205  if (mp->is_ipv6)
206  {
207  memcpy (mp->dhcp_src_address, &proxy->dhcp_src_address.ip6, 16);
208  }
209  else
210  {
211  /* put the address in the first bytes */
212  memcpy (mp->dhcp_src_address, &proxy->dhcp_src_address.ip4, 4);
213  }
214  vl_api_send_msg (reg, (u8 *) mp);
215 }
216 
217 static void
219  const dhcp_client_t * client)
220 {
221  size_t len;
222  u8 i;
223 
224  lease->is_ipv6 = 0; // only support IPv6 clients
225  lease->sw_if_index = ntohl (client->sw_if_index);
226  lease->state = client->state;
227  len = clib_min (sizeof (lease->hostname) - 1, vec_len (client->hostname));
228  clib_memcpy (&lease->hostname, client->hostname, len);
229  lease->hostname[len] = 0;
230 
231  lease->mask_width = client->subnet_mask_width;
232  clib_memcpy (&lease->host_address[0], (u8 *) & client->leased_address,
233  sizeof (ip4_address_t));
234  clib_memcpy (&lease->router_address[0], (u8 *) & client->router_address,
235  sizeof (ip4_address_t));
236 
237  lease->count = vec_len (client->domain_server_address);
238  for (i = 0; i < lease->count; i++)
239  clib_memcpy (&lease->domain_server[i].address,
240  (u8 *) & client->domain_server_address[i],
241  sizeof (ip4_address_t));
242 
243  clib_memcpy (&lease->host_mac[0], client->client_hardware_address, 6);
244 }
245 
246 static void
248  const dhcp_client_t * client)
249 {
250  size_t len;
251 
252  vclient->sw_if_index = ntohl (client->sw_if_index);
253  len = clib_min (sizeof (vclient->hostname) - 1, vec_len (client->hostname));
254  clib_memcpy (&vclient->hostname, client->hostname, len);
255  vclient->hostname[len] = 0;
256 
257  len = clib_min (sizeof (vclient->id) - 1,
258  vec_len (client->client_identifier));
259  clib_memcpy (&vclient->id, client->client_identifier, len);
260  vclient->id[len] = 0;
261 
262  if (NULL != client->event_callback)
263  vclient->want_dhcp_event = 1;
264  else
265  vclient->want_dhcp_event = 0;
266  vclient->set_broadcast_flag = client->set_broadcast_flag;
267  vclient->dscp = ip_dscp_encode (client->dscp);
268  vclient->pid = client->pid;
269 }
270 
271 static void
272 dhcp_compl_event_callback (u32 client_index, const dhcp_client_t * client)
273 {
276 
277  reg = vl_api_client_index_to_registration (client_index);
278  if (!reg)
279  return;
280 
281  mp = vl_msg_api_alloc (sizeof (*mp));
282  mp->client_index = client_index;
283  mp->pid = client->pid;
284  dhcp_client_lease_encode (&mp->lease, client);
285 
286  mp->_vl_msg_id = ntohs (VL_API_DHCP_COMPL_EVENT);
287 
288  vl_api_send_msg (reg, (u8 *) mp);
289 }
290 
293 {
295  vl_api_dhcp_client_config_reply_t *rmp;
297  ip_dscp_t dscp;
298  int rv = 0;
299 
300  VALIDATE_SW_IF_INDEX (&(mp->client));
301 
302  sw_if_index = ntohl (mp->client.sw_if_index);
303  dscp = ip_dscp_decode (mp->client.dscp);
304 
305  rv = dhcp_client_config (mp->is_add,
306  mp->client_index,
307  vm,
308  sw_if_index,
309  mp->client.hostname,
310  mp->client.id,
311  (mp->client.want_dhcp_event ?
313  NULL),
315  dscp, mp->client.pid);
316 
318  REPLY_MACRO (VL_API_DHCP_CLIENT_CONFIG_REPLY);
319 }
320 
322 {
326 
327 static int
328 send_dhcp_client_entry (const dhcp_client_t * client, void *arg)
329 {
332 
333  ctx = arg;
334 
335  mp = vl_msg_api_alloc (sizeof (*mp));
336  clib_memset (mp, 0, sizeof (*mp));
337 
338  mp->_vl_msg_id = ntohs (VL_API_DHCP_CLIENT_DETAILS);
339  mp->context = ctx->context;
340 
341  dhcp_client_data_encode (&mp->client, client);
342  dhcp_client_lease_encode (&mp->lease, client);
343 
344  vl_api_send_msg (ctx->reg, (u8 *) mp);
345 
346  return (1);
347 }
348 
349 static void
351 {
353 
355  if (!reg)
356  return;
357 
359  .reg = reg,
360  .context = mp->context,
361  };
363 }
364 
365 /*
366  * dhcp_api_hookup
367  * Add vpe's API message handlers to the table.
368  * vlib has already mapped shared memory and
369  * added the client registration handlers.
370  * See .../vlib-api/vlibmemory/memclnt_vlib.c:memclnt_process()
371  */
372 #define vl_msg_name_crc_list
373 #include <vnet/vnet_all_api_h.h>
374 #undef vl_msg_name_crc_list
375 
376 static void
378 {
379 #define _(id,n,crc) vl_msg_api_add_msg_name_crc (am, #n "_" #crc, id);
380  foreach_vl_msg_name_crc_dhcp;
381 #undef _
382 }
383 
384 static clib_error_t *
386 {
387  api_main_t *am = &api_main;
388 
389 #define _(N,n) \
390  vl_msg_api_set_handlers(VL_API_##N, #n, \
391  vl_api_##n##_t_handler, \
392  vl_noop_handler, \
393  vl_api_##n##_t_endian, \
394  vl_api_##n##_t_print, \
395  sizeof(vl_api_##n##_t), 1);
397 #undef _
398 
399  /*
400  * Set up the (msg_name, crc, message-id) table
401  */
403 
408 
409  return 0;
410 }
411 
413 
414 /*
415  * fd.io coding-style-patch-verification: ON
416  *
417  * Local Variables:
418  * eval: (c-set-style "gnu")
419  * End:
420  */
DHCP Proxy set / unset vss request.
Definition: dhcp.api:54
vl_api_dhcp_lease_t lease
Definition: dhcp.api:166
#define vec_foreach_index(var, v)
Iterate over vector indices.
static dhcp_vss_t * dhcp_get_vss_info(dhcp_proxy_main_t *dm, u32 rx_fib_index, fib_protocol_t proto)
Get the VSS data for the FIB index.
Definition: dhcp_proxy.h:255
vl_api_domain_server_t domain_server[count]
Definition: dhcp.api:131
#define clib_min(x, y)
Definition: clib.h:295
static clib_error_t * dhcp_api_hookup(vlib_main_t *vm)
Definition: dhcp_api.c:385
#define VSS_TYPE_INVALID
Definition: dhcp_proxy.h:68
DHCP Client details returned from dump.
Definition: dhcp.api:162
dhcp_server_t * dhcp_servers
The set of DHCP servers to which messages are relayed.
Definition: dhcp_proxy.h:111
DHCP Client config data.
Definition: dhcp.api:78
ip_dscp_t dscp
Definition: client.h:93
#define NULL
Definition: clib.h:58
dhcp_proxy_main_t dhcp_proxy_main
Shard 4/6 instance of DHCP main.
Definition: dhcp_proxy.c:25
int dhcp6_proxy_set_server(ip46_address_t *addr, ip46_address_t *src_addr, u32 rx_table_id, u32 server_table_id, int is_del)
static void vl_api_send_msg(vl_api_registration_t *rp, u8 *elem)
Definition: api.h:34
ip46_address_t dhcp_src_address
The source address to use in relayed messaes.
Definition: dhcp_proxy.h:126
The Virtual Sub-net Selection information for a given RX FIB.
Definition: dhcp_proxy.h:57
vlib_node_registration_t dhcp6_reply_process_node
(constructor) VLIB_REGISTER_NODE (dhcp6_reply_process_node)
vl_api_address_t src
Definition: gre.api:51
int i
u32 sw_if_index
Definition: client.h:50
void dhcp_send_details(fib_protocol_t proto, void *opaque, u32 context, dhcp_proxy_t *proxy)
Send the details of a proxy session to the API client during a dump.
Definition: dhcp_api.c:135
u32 rx_fib_index
The FIB index (not the external Table-ID) in which the client is resides.
Definition: dhcp_proxy.h:132
clib_memset(h->entries, 0, sizeof(h->entries[0])*entries)
void dhcp_client_walk(dhcp_client_walk_cb_t cb, void *ctx)
Walk (visit each) DHCP client configuration.
Definition: client.c:1112
u8 * format(u8 *s, const char *fmt,...)
Definition: format.c:424
u8 vss_type
VSS type as defined in RFC 6607: 0 for NVT ASCII VPN Identifier 1 for RFC 2685 VPN-ID of 7 octects - ...
Definition: dhcp_proxy.h:65
static void vl_api_dhcp_client_dump_t_handler(vl_api_dhcp_client_dump_t *mp)
Definition: dhcp_api.c:350
dhcp_client_state_t state
Definition: client.h:47
void * vl_msg_api_alloc(int nbytes)
unsigned char u8
Definition: types.h:56
void dhcp6_set_publisher_node(uword node_index, uword event_type)
enum fib_protocol_t_ fib_protocol_t
Protocol Type.
Dump the DHCP client configurations.
Definition: dhcp.api:152
int dhcp_client_config(u32 is_add, u32 client_index, vlib_main_t *vm, u32 sw_if_index, u8 *hostname, u8 *client_id, dhcp_event_cb_t event_callback, u8 set_broadcast_flag, ip_dscp_t dscp, u32 pid)
Add/Delete DHCP clients.
Definition: client.c:1025
#define clib_memcpy(d, s, n)
Definition: string.h:180
void dhcp_proxy_dump(fib_protocol_t proto, void *opaque, u32 context)
Dump the proxy configs to the API.
Definition: dhcp_proxy.c:248
vl_api_dhcp_client_t client
Definition: dhcp.api:165
vl_api_dhcp_server_t servers[count]
Definition: dhcp.api:200
vl_api_interface_index_t sw_if_index
Definition: gre.api:50
static void vl_api_dhcp_proxy_config_t_handler(vl_api_dhcp_proxy_config_t *mp)
Definition: dhcp_api.c:85
static int send_dhcp_client_entry(const dhcp_client_t *client, void *arg)
Definition: dhcp_api.c:328
dhcp_event_cb_t event_callback
Definition: client.h:95
#define foreach_vpe_api_msg
Definition: dhcp_api.c:51
void dhcp6_pd_set_publisher_node(uword node_index, uword event_type)
#define VSS_TYPE_ASCII
Definition: dhcp_proxy.h:66
unsigned int u32
Definition: types.h:88
ip_dscp_t ip_dscp_decode(u8 in)
Definition: ip_types_api.c:99
u8 set_broadcast_flag
Definition: client.h:83
u32 server_fib_index
The FIB index (not the external Table-ID) in which the server is reachable.
Definition: dhcp_proxy.h:95
ip4_address_t leased_address
Definition: client.h:63
struct dhcp_client_send_walk_ctx_t_ dhcp_client_send_walk_ctx_t
long ctx[MAX_CONNS]
Definition: main.c:144
Data learned by the client during the DHCP process.
Definition: dhcp.api:120
#define REPLY_MACRO(t)
static void vl_api_dhcp_proxy_dump_t_handler(vl_api_dhcp_proxy_dump_t *mp)
Definition: dhcp_api.c:122
int dhcp_proxy_set_vss(fib_protocol_t proto, u32 tbl_id, u8 vss_type, u8 *vpn_ascii_id, u32 oui, u32 vpn_index, u8 is_del)
Configure/set a new VSS info.
Definition: dhcp_proxy.c:309
static void dhcp_client_data_encode(vl_api_dhcp_client_t *vclient, const dhcp_client_t *client)
Definition: dhcp_api.c:247
u8 client_hardware_address[6]
Definition: client.h:85
u8 len
Definition: ip_types.api:90
API main structure, used by both vpp and binary API clients.
Definition: api_common.h:203
An API client registration, only in vpp/vlib.
Definition: api_common.h:46
#define BAD_SW_IF_INDEX_LABEL
u8 * vpn_ascii_id
Type 0 ASCII VPN Identifier.
Definition: dhcp_proxy.h:77
u32 ft_table_id
Table ID (hash key) for this FIB.
Definition: fib_table.h:89
vlib_main_t * vm
Definition: buffer.c:312
ip4_address_t * domain_server_address
Definition: client.h:67
static void dhcp_compl_event_callback(u32 client_index, const dhcp_client_t *client)
Definition: dhcp_api.c:272
vl_api_registration_t * reg
Definition: dhcp_api.c:323
A representation of a single DHCP Server within a given VRF config.
Definition: dhcp_proxy.h:83
static vl_api_registration_t * vl_api_client_index_to_registration(u32 index)
Definition: api.h:56
u8 router_address[16]
Definition: dhcp.api:128
vl_api_ip_dscp_t dscp
Definition: dhcp.api:85
int dhcp4_proxy_set_server(ip46_address_t *addr, ip46_address_t *src_addr, u32 rx_table_id, u32 server_table_id, int is_del)
ip46_address_t dhcp_server
The address of the DHCP server to which to relay the client&#39;s messages.
Definition: dhcp_proxy.h:89
enum ip_dscp_t_ ip_dscp_t
VLIB_API_INIT_FUNCTION(dhcp_api_hookup)
size_t count
Definition: vapi.c:47
u32 subnet_mask_width
Definition: client.h:65
vl_api_dhcp_lease_t lease
Definition: dhcp.api:143
ip4_address_t router_address
Definition: client.h:66
static vlib_main_t * vlib_get_main(void)
Definition: global_funcs.h:23
u8 dhcp_server[16]
Definition: dhcp.api:183
Tell client about a DHCP completion event.
Definition: dhcp.api:189
#define ip46_address_reset(ip46)
Definition: ip6_packet.h:91
A DHCP proxy represenation fpr per-client VRF config.
Definition: dhcp_proxy.h:101
u8 * client_identifier
Definition: client.h:76
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
u8 vpn_id[7]
Type 1 VPN-ID.
Definition: dhcp_proxy.h:73
vl_api_dhcp_client_t client
Definition: dhcp.api:100
static void setup_message_id_table(api_main_t *am)
Definition: dhcp_api.c:377
#define VSS_TYPE_VPN_ID
Definition: dhcp_proxy.h:67
static void vl_api_dhcp_client_config_t_handler(vl_api_dhcp_client_config_t *mp)
Definition: dhcp_api.c:292
fib_table_t * fib_table_get(fib_node_index_t index, fib_protocol_t proto)
Get a pointer to a FIB table.
Definition: fib_table.c:27
static void vl_api_dhcp_proxy_set_vss_t_handler(vl_api_dhcp_proxy_set_vss_t *mp)
Definition: dhcp_api.c:66
Dump DHCP proxy table.
Definition: dhcp.api:173
u8 * hostname
Definition: client.h:75
vlib_node_registration_t dhcp6_pd_reply_process_node
(constructor) VLIB_REGISTER_NODE (dhcp6_pd_reply_process_node)
u8 ip_dscp_encode(ip_dscp_t dscp)
Definition: ip_types_api.c:105
u32 dhcp_proxy_rx_table_get_table_id(fib_protocol_t proto, u32 fib_index)
Definition: dhcp_proxy.c:46
static void dhcp_client_lease_encode(vl_api_dhcp_lease_t *lease, const dhcp_client_t *client)
Definition: dhcp_api.c:218
Tell client about a DHCP completion event.
Definition: dhcp.api:139
u32 context
Definition: gre.api:45
api_main_t api_main
Definition: api_shared.c:35
DHCP Client config add / del request.
Definition: dhcp.api:95
u8 host_address[16]
Definition: dhcp.api:127
vl_api_fib_path_nh_proto_t proto
Definition: fib_types.api:125
DHCP Proxy config add / del request.
Definition: dhcp.api:31
#define VALIDATE_SW_IF_INDEX(mp)
A protocol Independent FIB table.
Definition: fib_table.h:69