nat: Final NAT44 EI/ED split patch
[vpp.git] / src / plugins / nat / dslite / dslite_cli.c
1 /*
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:
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
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.
14  */
15 #include <nat/dslite/dslite.h>
16
17 static clib_error_t *
18 dslite_add_del_pool_addr_command_fn (vlib_main_t * vm,
19                                      unformat_input_t * input,
20                                      vlib_cli_command_t * cmd)
21 {
22   dslite_main_t *dm = &dslite_main;
23   unformat_input_t _line_input, *line_input = &_line_input;
24   ip4_address_t start_addr, end_addr, this_addr;
25   u32 start_host_order, end_host_order;
26   int count, rv;
27   u8 is_add = 1;
28   clib_error_t *error = 0;
29
30   /* Get a line of input. */
31   if (!unformat_user (input, unformat_line_input, line_input))
32     return 0;
33
34   while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
35     {
36       if (unformat (line_input, "%U - %U",
37                     unformat_ip4_address, &start_addr,
38                     unformat_ip4_address, &end_addr))
39         ;
40       else if (unformat (line_input, "%U", unformat_ip4_address, &start_addr))
41         end_addr = start_addr;
42       else if (unformat (line_input, "del"))
43         is_add = 0;
44       else
45         {
46           error = clib_error_return (0, "unknown input '%U'",
47                                      format_unformat_error, line_input);
48           goto done;
49         }
50     }
51
52   start_host_order = clib_host_to_net_u32 (start_addr.as_u32);
53   end_host_order = clib_host_to_net_u32 (end_addr.as_u32);
54
55   if (end_host_order < start_host_order)
56     {
57       error = clib_error_return (0, "end address less than start address");
58       goto done;
59     }
60
61   count = (end_host_order - start_host_order) + 1;
62   this_addr = start_addr;
63
64   rv = nat_add_del_ip4_pool_addrs (&dm->pool, this_addr, count, is_add, 0);
65
66   switch (rv)
67     {
68     case VNET_API_ERROR_NO_SUCH_ENTRY:
69       error =
70         clib_error_return (0, "DS-Lite pool address %U not exist.",
71                            format_ip4_address, &this_addr);
72       break;
73     case VNET_API_ERROR_VALUE_EXIST:
74       error =
75         clib_error_return (0, "DS-Lite pool address %U exist.",
76                            format_ip4_address, &this_addr);
77       break;
78     }
79
80 done:
81   unformat_free (line_input);
82
83   return error;
84 }
85
86 static clib_error_t *
87 dslite_show_pool_command_fn (vlib_main_t * vm,
88                              unformat_input_t * input,
89                              vlib_cli_command_t * cmd)
90 {
91   dslite_main_t *dm = &dslite_main;
92   nat_ip4_pool_addr_t *a;
93
94   vlib_cli_output (vm, "DS-Lite pool:");
95
96   /* *INDENT-OFF* */
97   vec_foreach (a, dm->pool.pool_addr)
98     {
99       vlib_cli_output (vm, "%U", format_ip4_address, &a->addr);
100     }
101   /* *INDENT-ON* */
102   return 0;
103 }
104
105 static clib_error_t *
106 dslite_set_aftr_tunnel_addr_command_fn (vlib_main_t * vm,
107                                         unformat_input_t * input,
108                                         vlib_cli_command_t * cmd)
109 {
110   dslite_main_t *dm = &dslite_main;
111   unformat_input_t _line_input, *line_input = &_line_input;
112   ip6_address_t ip6_addr;
113   int rv;
114   clib_error_t *error = 0;
115
116   /* Get a line of input. */
117   if (!unformat_user (input, unformat_line_input, line_input))
118     return 0;
119
120   while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
121     {
122       if (unformat (line_input, "%U", unformat_ip6_address, &ip6_addr))
123         ;
124       else
125         {
126           error = clib_error_return (0, "unknown input '%U'",
127                                      format_unformat_error, line_input);
128           goto done;
129         }
130     }
131
132   rv = dslite_set_aftr_ip6_addr (dm, &ip6_addr);
133
134   if (rv)
135     error =
136       clib_error_return (0,
137                          "Set DS-Lite AFTR tunnel endpoint address failed.");
138
139 done:
140   unformat_free (line_input);
141
142   return error;
143 }
144
145 static clib_error_t *
146 dslite_show_aftr_ip6_addr_command_fn (vlib_main_t * vm,
147                                       unformat_input_t * input,
148                                       vlib_cli_command_t * cmd)
149 {
150   dslite_main_t *dm = &dslite_main;
151
152   vlib_cli_output (vm, "%U", format_ip6_address, &dm->aftr_ip6_addr);
153   return 0;
154 }
155
156 static clib_error_t *
157 dslite_set_b4_tunnel_addr_command_fn (vlib_main_t * vm,
158                                       unformat_input_t * input,
159                                       vlib_cli_command_t * cmd)
160 {
161   dslite_main_t *dm = &dslite_main;
162   unformat_input_t _line_input, *line_input = &_line_input;
163   ip6_address_t ip6_addr;
164   int rv;
165   clib_error_t *error = 0;
166
167   /* Get a line of input. */
168   if (!unformat_user (input, unformat_line_input, line_input))
169     return 0;
170
171   while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
172     {
173       if (unformat (line_input, "%U", unformat_ip6_address, &ip6_addr))
174         ;
175       else
176         {
177           error = clib_error_return (0, "unknown input '%U'",
178                                      format_unformat_error, line_input);
179           goto done;
180         }
181     }
182
183   rv = dslite_set_b4_ip6_addr (dm, &ip6_addr);
184
185   if (rv)
186     error =
187       clib_error_return (0, "Set DS-Lite B4 tunnel endpoint address failed.");
188
189 done:
190   unformat_free (line_input);
191
192   return error;
193 }
194
195 static clib_error_t *
196 dslite_show_b4_ip6_addr_command_fn (vlib_main_t * vm,
197                                     unformat_input_t * input,
198                                     vlib_cli_command_t * cmd)
199 {
200   dslite_main_t *dm = &dslite_main;
201
202   vlib_cli_output (vm, "%U", format_ip6_address, &dm->b4_ip6_addr);
203   return 0;
204 }
205
206 static u8 *
207 format_dslite_session (u8 * s, va_list * args)
208 {
209   dslite_session_t *session = va_arg (*args, dslite_session_t *);
210   u32 indent = format_get_indent (s);
211
212   s = format (s, "%Uin %U:%u out %U:%u protocol %U\n",
213               format_white_space, indent + 2,
214               format_ip4_address, &session->in2out.addr,
215               clib_net_to_host_u16 (session->in2out.port),
216               format_ip4_address, &session->out2in.addr,
217               clib_net_to_host_u16 (session->out2in.port),
218               format_nat_protocol, session->in2out.proto);
219   s = format (s, "%Utotal pkts %d, total bytes %lld\n",
220               format_white_space, indent + 4,
221               session->total_pkts, session->total_bytes);
222   return s;
223 }
224
225 static u8 *
226 format_dslite_b4 (u8 * s, va_list * args)
227 {
228   dslite_per_thread_data_t *td = va_arg (*args, dslite_per_thread_data_t *);
229   dslite_b4_t *b4 = va_arg (*args, dslite_b4_t *);
230   dlist_elt_t *head, *elt;
231   u32 elt_index, head_index;
232   u32 session_index;
233   dslite_session_t *session;
234
235   s =
236     format (s, "B4 %U %d sessions\n", format_ip6_address, &b4->addr,
237             b4->nsessions);
238
239   if (b4->nsessions == 0)
240     return s;
241
242   head_index = b4->sessions_per_b4_list_head_index;
243   head = pool_elt_at_index (td->list_pool, head_index);
244   elt_index = head->next;
245   elt = pool_elt_at_index (td->list_pool, elt_index);
246   session_index = elt->value;
247   while (session_index != ~0)
248     {
249       session = pool_elt_at_index (td->sessions, session_index);
250       s = format (s, "%U", format_dslite_session, session);
251       elt_index = elt->next;
252       elt = pool_elt_at_index (td->list_pool, elt_index);
253       session_index = elt->value;
254     }
255
256   return s;
257 }
258
259 static clib_error_t *
260 dslite_show_sessions_command_fn (vlib_main_t * vm,
261                                  unformat_input_t * input,
262                                  vlib_cli_command_t * cmd)
263 {
264   dslite_main_t *dm = &dslite_main;
265   dslite_per_thread_data_t *td;
266   dslite_b4_t *b4;
267
268   /* *INDENT-OFF* */
269   vec_foreach (td, dm->per_thread_data)
270     {
271       pool_foreach (b4, td->b4s)
272        {
273         vlib_cli_output (vm, "%U", format_dslite_b4, td, b4);
274       }
275     }
276   /* *INDENT-ON* */
277
278   return 0;
279 }
280
281 /* *INDENT-OFF* */
282
283 /*?
284  * @cliexpar
285  * @cliexstart{dslite add pool address}
286  * Add/delete DS-Lite pool address for AFTR element.
287  * To add DS-Lite pool address use:
288  *  vpp# dslite add pool address 10.1.1.3
289  * To add DS-Lite pool address range use:
290  *  vpp# dslite add pool address 10.1.1.5 - 10.1.1.7
291  * @cliexend
292 ?*/
293 VLIB_CLI_COMMAND (dslite_add_pool_address_command, static) = {
294   .path = "dslite add pool address",
295   .short_help = "dslite add pool address <ip4-range-start> [- <ip4-range-end>] "
296                 " [del]",
297   .function = dslite_add_del_pool_addr_command_fn,
298 };
299
300 /*?
301  * @cliexpar
302  * @cliexstart{show dslite pool}
303  * Show DS-lite pool addresses.
304  * vpp# show dslite pool
305  * DS-Lite pool:
306  * 10.0.0.3
307  * 10.0.0.5
308  * 10.0.0.6
309  * 10.0.0.7
310  * @cliexend
311 ?*/
312 VLIB_CLI_COMMAND (show_dslite_pool_command, static) = {
313   .path = "show dslite pool",
314   .short_help = "show dslite pool",
315   .function = dslite_show_pool_command_fn,
316 };
317
318 /*?
319  * @cliexpar
320  * @cliexstart{dslite set aftr-tunnel-endpoint-address}
321  * Set IPv6 tunnel endpoint address of the AFTR element.
322  * To set AFTR tunnel endpoint address use:
323  * vpp# dslite set aftr-tunnel-endpoint-address 2001:db8:85a3::8a2e:370:1
324  * @cliexend
325 ?*/
326 VLIB_CLI_COMMAND (dslite_set_aftr_tunnel_addr, static) = {
327   .path = "dslite set aftr-tunnel-endpoint-address",
328   .short_help = "dslite set aftr-tunnel-endpoint-address <ip6>",
329   .function = dslite_set_aftr_tunnel_addr_command_fn,
330 };
331
332 /*?
333  * @cliexpar
334  * @cliexstart{show dslite aftr-tunnel-endpoint-address}
335  * Show IPv6 tunnel endpoint address of the AFTR element.
336  * vpp# show dslite aftr-tunnel-endpoint-address
337  * 2001:db8:85a3::8a2e:370:1
338  * @cliexend
339 ?*/
340 VLIB_CLI_COMMAND (dslite_show_aftr_ip6_addr, static) = {
341   .path = "show dslite aftr-tunnel-endpoint-address",
342   .short_help = "show dslite aftr-tunnel-endpoint-address",
343   .function = dslite_show_aftr_ip6_addr_command_fn,
344 };
345
346 /*?
347  * @cliexpar
348  * @cliexstart{dslite set b4-tunnel-endpoint-address}
349  * Set IPv6 tunnel endpoint address of the B4 element.
350  * To set B4 tunnel endpoint address use:
351  * vpp# dslite set b4-tunnel-endpoint-address 2001:db8:62aa::375e:f4c1:1
352  * @cliexend
353 ?*/
354 VLIB_CLI_COMMAND (dslite_set_b4_tunnel_addr, static) = {
355   .path = "dslite set b4-tunnel-endpoint-address",
356   .short_help = "dslite set b4-tunnel-endpoint-address <ip6>",
357   .function = dslite_set_b4_tunnel_addr_command_fn,
358 };
359
360 /*?
361  * @cliexpar
362  * @cliexstart{show dslite b4-tunnel-endpoint-address}
363  * Show IPv6 tunnel endpoint address of the B4 element.
364  * vpp# show dslite b4-tunnel-endpoint-address
365  * 2001:db8:62aa::375e:f4c1:1
366  * @cliexend
367 ?*/
368 VLIB_CLI_COMMAND (dslite_show_b4_ip6_addr, static) = {
369   .path = "show dslite b4-tunnel-endpoint-address",
370   .short_help = "show dslite b4-tunnel-endpoint-address",
371   .function = dslite_show_b4_ip6_addr_command_fn,
372 };
373
374 /*?
375  * @cliexpar
376  * @cliexstart{show dslite sessions}
377  * Show DS-Lite sessions.
378  * vpp# show dslite sessions
379  * B4 fd01:2::2 1 sessions
380  *   in 192.168.1.1:20000 out 10.0.0.3:16253 protocol udp
381  *     total pkts 2, total bytes 136
382  * B4 fd01:2::3 2 sessions
383  *   in 192.168.1.1:20001 out 10.0.0.3:18995 protocol tcp
384  *     total pkts 2, total bytes 160
385  *   in 192.168.1.1:4000 out 10.0.0.3:53893 protocol icmp
386  *     total pkts 2, total bytes 136
387  * @cliexend
388 ?*/
389 VLIB_CLI_COMMAND (dslite_show_sessions, static) = {
390   .path = "show dslite sessions",
391   .short_help = "show dslite sessions",
392   .function = dslite_show_sessions_command_fn,
393 };
394
395 /* *INDENT-ON* */
396
397 /*
398  * fd.io coding-style-patch-verification: ON
399  *
400  * Local Variables:
401  * eval: (c-set-style "gnu")
402  * End:
403  */