2 * Copyright (c) 2017 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:
7 * http://www.apache.org/licenses/LICENSE-2.0
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.
15 #include <nat/dslite/dslite.h>
17 #define DSLITE_EXPECTED_ARGUMENT "expected required argument(s)"
20 dslite_add_del_pool_addr_command_fn (vlib_main_t * vm,
21 unformat_input_t * input,
22 vlib_cli_command_t * cmd)
24 dslite_main_t *dm = &dslite_main;
25 unformat_input_t _line_input, *line_input = &_line_input;
26 ip4_address_t start_addr, end_addr, this_addr;
27 u32 start_host_order, end_host_order;
30 clib_error_t *error = 0;
32 /* Get a line of input. */
33 if (!unformat_user (input, unformat_line_input, line_input))
34 return clib_error_return (0, DSLITE_EXPECTED_ARGUMENT);
36 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
38 if (unformat (line_input, "%U - %U",
39 unformat_ip4_address, &start_addr,
40 unformat_ip4_address, &end_addr))
42 else if (unformat (line_input, "%U", unformat_ip4_address, &start_addr))
43 end_addr = start_addr;
44 else if (unformat (line_input, "del"))
48 error = clib_error_return (0, "unknown input '%U'",
49 format_unformat_error, line_input);
54 start_host_order = clib_host_to_net_u32 (start_addr.as_u32);
55 end_host_order = clib_host_to_net_u32 (end_addr.as_u32);
57 if (end_host_order < start_host_order)
59 error = clib_error_return (0, "end address less than start address");
63 count = (end_host_order - start_host_order) + 1;
64 this_addr = start_addr;
66 rv = nat_add_del_ip4_pool_addrs (&dm->pool, this_addr, count, is_add, 0);
70 case VNET_API_ERROR_NO_SUCH_ENTRY:
72 clib_error_return (0, "DS-Lite pool address %U not exist.",
73 format_ip4_address, &this_addr);
75 case VNET_API_ERROR_VALUE_EXIST:
77 clib_error_return (0, "DS-Lite pool address %U exist.",
78 format_ip4_address, &this_addr);
83 unformat_free (line_input);
89 dslite_show_pool_command_fn (vlib_main_t * vm,
90 unformat_input_t * input,
91 vlib_cli_command_t * cmd)
93 dslite_main_t *dm = &dslite_main;
94 nat_ip4_pool_addr_t *a;
96 vlib_cli_output (vm, "DS-Lite pool:");
99 vec_foreach (a, dm->pool.pool_addr)
101 vlib_cli_output (vm, "%U", format_ip4_address, &a->addr);
107 static clib_error_t *
108 dslite_set_aftr_tunnel_addr_command_fn (vlib_main_t * vm,
109 unformat_input_t * input,
110 vlib_cli_command_t * cmd)
112 dslite_main_t *dm = &dslite_main;
113 unformat_input_t _line_input, *line_input = &_line_input;
114 ip6_address_t ip6_addr;
116 clib_error_t *error = 0;
118 /* Get a line of input. */
119 if (!unformat_user (input, unformat_line_input, line_input))
120 return clib_error_return (0, DSLITE_EXPECTED_ARGUMENT);
122 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
124 if (unformat (line_input, "%U", unformat_ip6_address, &ip6_addr))
128 error = clib_error_return (0, "unknown input '%U'",
129 format_unformat_error, line_input);
134 rv = dslite_set_aftr_ip6_addr (dm, &ip6_addr);
138 clib_error_return (0,
139 "Set DS-Lite AFTR tunnel endpoint address failed.");
142 unformat_free (line_input);
147 static clib_error_t *
148 dslite_show_aftr_ip6_addr_command_fn (vlib_main_t * vm,
149 unformat_input_t * input,
150 vlib_cli_command_t * cmd)
152 dslite_main_t *dm = &dslite_main;
154 vlib_cli_output (vm, "%U", format_ip6_address, &dm->aftr_ip6_addr);
158 static clib_error_t *
159 dslite_set_b4_tunnel_addr_command_fn (vlib_main_t * vm,
160 unformat_input_t * input,
161 vlib_cli_command_t * cmd)
163 dslite_main_t *dm = &dslite_main;
164 unformat_input_t _line_input, *line_input = &_line_input;
165 ip6_address_t ip6_addr;
167 clib_error_t *error = 0;
169 /* Get a line of input. */
170 if (!unformat_user (input, unformat_line_input, line_input))
171 return clib_error_return (0, DSLITE_EXPECTED_ARGUMENT);
173 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
175 if (unformat (line_input, "%U", unformat_ip6_address, &ip6_addr))
179 error = clib_error_return (0, "unknown input '%U'",
180 format_unformat_error, line_input);
185 rv = dslite_set_b4_ip6_addr (dm, &ip6_addr);
189 clib_error_return (0, "Set DS-Lite B4 tunnel endpoint address failed.");
192 unformat_free (line_input);
197 static clib_error_t *
198 dslite_show_b4_ip6_addr_command_fn (vlib_main_t * vm,
199 unformat_input_t * input,
200 vlib_cli_command_t * cmd)
202 dslite_main_t *dm = &dslite_main;
204 vlib_cli_output (vm, "%U", format_ip6_address, &dm->b4_ip6_addr);
209 format_dslite_session (u8 * s, va_list * args)
211 dslite_session_t *session = va_arg (*args, dslite_session_t *);
212 u32 indent = format_get_indent (s);
214 s = format (s, "%Uin %U:%u out %U:%u protocol %U\n",
215 format_white_space, indent + 2,
216 format_ip4_address, &session->in2out.addr,
217 clib_net_to_host_u16 (session->in2out.port),
218 format_ip4_address, &session->out2in.addr,
219 clib_net_to_host_u16 (session->out2in.port),
220 format_nat_protocol, session->in2out.proto);
221 s = format (s, "%Utotal pkts %d, total bytes %lld\n",
222 format_white_space, indent + 4,
223 session->total_pkts, session->total_bytes);
228 format_dslite_b4 (u8 * s, va_list * args)
230 dslite_per_thread_data_t *td = va_arg (*args, dslite_per_thread_data_t *);
231 dslite_b4_t *b4 = va_arg (*args, dslite_b4_t *);
232 dlist_elt_t *head, *elt;
233 u32 elt_index, head_index;
235 dslite_session_t *session;
238 format (s, "B4 %U %d sessions\n", format_ip6_address, &b4->addr,
241 if (b4->nsessions == 0)
244 head_index = b4->sessions_per_b4_list_head_index;
245 head = pool_elt_at_index (td->list_pool, head_index);
246 elt_index = head->next;
247 elt = pool_elt_at_index (td->list_pool, elt_index);
248 session_index = elt->value;
249 while (session_index != ~0)
251 session = pool_elt_at_index (td->sessions, session_index);
252 s = format (s, "%U", format_dslite_session, session);
253 elt_index = elt->next;
254 elt = pool_elt_at_index (td->list_pool, elt_index);
255 session_index = elt->value;
261 static clib_error_t *
262 dslite_show_sessions_command_fn (vlib_main_t * vm,
263 unformat_input_t * input,
264 vlib_cli_command_t * cmd)
266 dslite_main_t *dm = &dslite_main;
267 dslite_per_thread_data_t *td;
271 vec_foreach (td, dm->per_thread_data)
273 pool_foreach (b4, td->b4s)
275 vlib_cli_output (vm, "%U", format_dslite_b4, td, b4);
287 * @cliexstart{dslite add pool address}
288 * Add/delete DS-Lite pool address for AFTR element.
289 * To add DS-Lite pool address use:
290 * vpp# dslite add pool address 10.1.1.3
291 * To add DS-Lite pool address range use:
292 * vpp# dslite add pool address 10.1.1.5 - 10.1.1.7
295 VLIB_CLI_COMMAND (dslite_add_pool_address_command, static) = {
296 .path = "dslite add pool address",
297 .short_help = "dslite add pool address <ip4-range-start> [- <ip4-range-end>] "
299 .function = dslite_add_del_pool_addr_command_fn,
304 * @cliexstart{show dslite pool}
305 * Show DS-lite pool addresses.
306 * vpp# show dslite pool
314 VLIB_CLI_COMMAND (show_dslite_pool_command, static) = {
315 .path = "show dslite pool",
316 .short_help = "show dslite pool",
317 .function = dslite_show_pool_command_fn,
322 * @cliexstart{dslite set aftr-tunnel-endpoint-address}
323 * Set IPv6 tunnel endpoint address of the AFTR element.
324 * To set AFTR tunnel endpoint address use:
325 * vpp# dslite set aftr-tunnel-endpoint-address 2001:db8:85a3::8a2e:370:1
328 VLIB_CLI_COMMAND (dslite_set_aftr_tunnel_addr, static) = {
329 .path = "dslite set aftr-tunnel-endpoint-address",
330 .short_help = "dslite set aftr-tunnel-endpoint-address <ip6>",
331 .function = dslite_set_aftr_tunnel_addr_command_fn,
336 * @cliexstart{show dslite aftr-tunnel-endpoint-address}
337 * Show IPv6 tunnel endpoint address of the AFTR element.
338 * vpp# show dslite aftr-tunnel-endpoint-address
339 * 2001:db8:85a3::8a2e:370:1
342 VLIB_CLI_COMMAND (dslite_show_aftr_ip6_addr, static) = {
343 .path = "show dslite aftr-tunnel-endpoint-address",
344 .short_help = "show dslite aftr-tunnel-endpoint-address",
345 .function = dslite_show_aftr_ip6_addr_command_fn,
350 * @cliexstart{dslite set b4-tunnel-endpoint-address}
351 * Set IPv6 tunnel endpoint address of the B4 element.
352 * To set B4 tunnel endpoint address use:
353 * vpp# dslite set b4-tunnel-endpoint-address 2001:db8:62aa::375e:f4c1:1
356 VLIB_CLI_COMMAND (dslite_set_b4_tunnel_addr, static) = {
357 .path = "dslite set b4-tunnel-endpoint-address",
358 .short_help = "dslite set b4-tunnel-endpoint-address <ip6>",
359 .function = dslite_set_b4_tunnel_addr_command_fn,
364 * @cliexstart{show dslite b4-tunnel-endpoint-address}
365 * Show IPv6 tunnel endpoint address of the B4 element.
366 * vpp# show dslite b4-tunnel-endpoint-address
367 * 2001:db8:62aa::375e:f4c1:1
370 VLIB_CLI_COMMAND (dslite_show_b4_ip6_addr, static) = {
371 .path = "show dslite b4-tunnel-endpoint-address",
372 .short_help = "show dslite b4-tunnel-endpoint-address",
373 .function = dslite_show_b4_ip6_addr_command_fn,
378 * @cliexstart{show dslite sessions}
379 * Show DS-Lite sessions.
380 * vpp# show dslite sessions
381 * B4 fd01:2::2 1 sessions
382 * in 192.168.1.1:20000 out 10.0.0.3:16253 protocol udp
383 * total pkts 2, total bytes 136
384 * B4 fd01:2::3 2 sessions
385 * in 192.168.1.1:20001 out 10.0.0.3:18995 protocol tcp
386 * total pkts 2, total bytes 160
387 * in 192.168.1.1:4000 out 10.0.0.3:53893 protocol icmp
388 * total pkts 2, total bytes 136
391 VLIB_CLI_COMMAND (dslite_show_sessions, static) = {
392 .path = "show dslite sessions",
393 .short_help = "show dslite sessions",
394 .function = dslite_show_sessions_command_fn,
400 * fd.io coding-style-patch-verification: ON
403 * eval: (c-set-style "gnu")