fib: format functions for path
[vpp.git] / src / vnet / fib / fib_types.c
1 /*
2  * Copyright (c) 2016 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
16 #include <vnet/ip/ip.h>
17
18 #include <vnet/fib/fib_types.h>
19 #include <vnet/fib/fib_internal.h>
20 #include <vnet/fib/fib_table.h>
21 #include <vnet/mfib/mfib_types.h>
22 #include <vnet/mpls/mpls.h>
23
24 /*
25  * arrays of protocol and link names
26  */
27 static const char* fib_protocol_names[] = FIB_PROTOCOLS;
28 static const char* vnet_link_names[] = VNET_LINKS;
29 static const char* fib_forw_chain_names[] = FIB_FORW_CHAINS;
30 static const char* fib_mpls_lsp_mode_names[] = FIB_MPLS_LSP_MODES;
31
32 u8 *
33 format_fib_protocol (u8 * s, va_list * ap)
34 {
35     fib_protocol_t proto = va_arg(*ap, int); // fib_protocol_t promotion
36
37     return (format (s, "%s", fib_protocol_names[proto]));
38 }
39
40 u8 *
41 format_vnet_link (u8 * s, va_list * ap)
42 {
43     vnet_link_t link = va_arg(*ap, int); // vnet_link_t promotion
44
45     return (format (s, "%s", vnet_link_names[link]));
46 }
47
48 u8 *
49 format_fib_forw_chain_type (u8 * s, va_list * args)
50 {
51     fib_forward_chain_type_t fct = va_arg(*args, int);
52
53     return (format (s, "%s", fib_forw_chain_names[fct]));
54 }
55
56 u8 *
57 format_fib_mpls_lsp_mode(u8 *s, va_list *ap)
58 {
59     fib_mpls_lsp_mode_t mode = va_arg(*ap, int);
60
61     return (format (s, "%s", fib_mpls_lsp_mode_names[mode])); 
62 }
63
64 u8 *
65 format_fib_mpls_label (u8 *s, va_list *ap)
66 {
67     fib_mpls_label_t *label = va_arg(*ap, fib_mpls_label_t *);
68
69     s = format(s, "%U %U ttl:%d exp:%d",
70                format_mpls_unicast_label,
71                label->fml_value,
72                format_fib_mpls_lsp_mode,
73                label->fml_mode,
74                label->fml_ttl,
75                label->fml_exp);
76
77     return (s);
78 }
79
80 void
81 fib_prefix_from_ip46_addr (const ip46_address_t *addr,
82                            fib_prefix_t *pfx)
83 {
84     ASSERT(!ip46_address_is_zero(addr));
85
86     pfx->fp_proto = ((ip46_address_is_ip4(addr) ?
87                       FIB_PROTOCOL_IP4 :
88                       FIB_PROTOCOL_IP6));
89     pfx->fp_len = ((ip46_address_is_ip4(addr) ?
90                     32 : 128));
91     pfx->fp_addr = *addr;
92     pfx->___fp___pad = 0;
93 }
94
95 u8 *
96 format_fib_route_path_flags (u8 *s, va_list *ap)
97 {
98     fib_route_path_flags_t flags = va_arg (*ap, fib_route_path_flags_t);
99
100     if (flags & FIB_ROUTE_PATH_RESOLVE_VIA_HOST)
101         s = format (s, "via-host");
102     if (flags & FIB_ROUTE_PATH_RESOLVE_VIA_ATTACHED)
103         s = format (s, "via-attached,");
104     if (flags & FIB_ROUTE_PATH_LOCAL)
105         s = format (s, "local,");
106     if (flags & FIB_ROUTE_PATH_ATTACHED)
107         s = format (s, "attached,");
108     if (flags & FIB_ROUTE_PATH_DROP)
109          s = format (s, "drop,");
110    if (flags & FIB_ROUTE_PATH_EXCLUSIVE)
111         s = format (s, "exclusive,");
112     if (flags & FIB_ROUTE_PATH_INTF_RX)
113         s = format (s, "intf-rx,");
114     if (flags & FIB_ROUTE_PATH_RPF_ID)
115         s = format (s, "rpf-id,");
116     if (flags & FIB_ROUTE_PATH_SOURCE_LOOKUP)
117         s = format (s, "src-lkup,");
118     if (flags & FIB_ROUTE_PATH_UDP_ENCAP)
119         s = format (s, "udp-encap,");
120     if (flags & FIB_ROUTE_PATH_BIER_FMASK)
121         s = format (s, "bier-fmask,");
122     if (flags & FIB_ROUTE_PATH_BIER_TABLE)
123         s = format (s, "bier-table,");
124     if (flags & FIB_ROUTE_PATH_BIER_IMP)
125         s = format (s, "bier-imp,");
126     if (flags & FIB_ROUTE_PATH_DEAG)
127         s = format (s, "deag,");
128     if (flags & FIB_ROUTE_PATH_DVR)
129         s = format (s, "dvr,");
130     if (flags & FIB_ROUTE_PATH_ICMP_UNREACH)
131         s = format (s, "imcp-unreach,");
132     if (flags & FIB_ROUTE_PATH_ICMP_PROHIBIT)
133         s = format (s, "icmp-prohibit,");
134     if (flags & FIB_ROUTE_PATH_CLASSIFY)
135         s = format (s, "classify,");
136     if (flags & FIB_ROUTE_PATH_POP_PW_CW)
137         s = format (s, "pop-pw-cw,");
138
139     return (s);
140 }
141
142 u8 *
143 format_fib_route_path (u8 *s, va_list *ap)
144 {
145     fib_route_path_t *rpath = va_arg (*ap, fib_route_path_t*);
146
147     s = format (s, "%U %U, %U, [%U]",
148                 format_dpo_proto, rpath->frp_proto,
149                 format_ip46_address, &rpath->frp_addr, IP46_TYPE_ANY,
150                 format_vnet_sw_if_index_name, vnet_get_main (),
151                 rpath->frp_sw_if_index,
152                 format_fib_route_path_flags, rpath->frp_flags);
153
154     return (s);
155 }
156
157 void
158 fib_prefix_from_mpls_label (mpls_label_t label,
159                             mpls_eos_bit_t eos,
160                             fib_prefix_t *pfx)
161 {
162     pfx->fp_proto = FIB_PROTOCOL_MPLS;
163     pfx->fp_len = 21;
164     pfx->fp_label = label;
165     pfx->fp_eos = eos;
166     pfx->___fp___pad = 0;
167 }
168
169 void
170 fib_prefix_copy (fib_prefix_t *dst,
171                  const fib_prefix_t *src)
172 {
173     clib_memcpy(dst, src, sizeof(*dst));
174 }
175
176 int
177 fib_prefix_cmp (const fib_prefix_t *p1,
178                 const fib_prefix_t *p2)
179 {
180     int res;
181
182     res = (p1->fp_proto - p2->fp_proto);
183
184     if (0 == res)
185     {
186         switch (p1->fp_proto)
187         {
188         case FIB_PROTOCOL_IP4:
189         case FIB_PROTOCOL_IP6:
190             res = (p1->fp_len - p2->fp_len);
191
192             if (0 == res)
193             {
194                 res = ip46_address_cmp(&p1->fp_addr, &p2->fp_addr);
195             }
196             break;
197         case FIB_PROTOCOL_MPLS:
198             res = (p1->fp_label - p2->fp_label);
199
200             if (0 == res)
201             {
202                 res = (p1->fp_eos - p2->fp_eos);
203             }
204             break;
205         }
206     }
207
208     return (res);
209 }
210
211 int
212 fib_prefix_is_cover (const fib_prefix_t *p1,
213                      const fib_prefix_t *p2)
214 {
215     switch (p1->fp_proto)
216     {
217     case FIB_PROTOCOL_IP4:
218         return (ip4_destination_matches_route(&ip4_main,
219                                               &p1->fp_addr.ip4,
220                                               &p2->fp_addr.ip4,
221                                               p1->fp_len));
222     case FIB_PROTOCOL_IP6:
223         return (ip6_destination_matches_route(&ip6_main,
224                                               &p1->fp_addr.ip6,
225                                               &p2->fp_addr.ip6,
226                                               p1->fp_len));
227     case FIB_PROTOCOL_MPLS:
228         break;
229     }
230     return (0);
231 }
232
233 u8
234 fib_prefix_get_host_length (fib_protocol_t proto)
235 {
236     switch (proto)
237     {
238     case FIB_PROTOCOL_IP4:
239         return (32);
240     case FIB_PROTOCOL_IP6:
241         return (128);
242     case FIB_PROTOCOL_MPLS:
243         return (21);
244     }
245     return (0);
246 }
247
248 int
249 fib_prefix_is_host (const fib_prefix_t *prefix)
250 {
251     switch (prefix->fp_proto)
252     {
253     case FIB_PROTOCOL_IP4:
254         return (prefix->fp_len == 32);
255     case FIB_PROTOCOL_IP6:
256         return (prefix->fp_len == 128);
257     case FIB_PROTOCOL_MPLS:
258         return (!0);
259     }
260     return (0);
261 }
262
263 u8 *
264 format_fib_prefix (u8 * s, va_list * args)
265 {
266     fib_prefix_t *fp = va_arg (*args, fib_prefix_t *);
267
268     /*
269      * protocol specific so it prints ::/0 correctly.
270      */
271     switch (fp->fp_proto)
272     {
273     case FIB_PROTOCOL_IP6:
274     {
275         ip6_address_t p6 = fp->fp_addr.ip6;
276
277         ip6_address_mask(&p6, &(ip6_main.fib_masks[fp->fp_len]));
278         s = format (s, "%U", format_ip6_address, &p6);
279         break;
280     }
281     case FIB_PROTOCOL_IP4:
282     {
283         ip4_address_t p4 = fp->fp_addr.ip4;
284         p4.as_u32 &= ip4_main.fib_masks[fp->fp_len];
285
286         s = format (s, "%U", format_ip4_address, &p4);
287         break;
288     }
289     case FIB_PROTOCOL_MPLS:
290         s = format (s, "%U:%U",
291                     format_mpls_unicast_label, fp->fp_label,
292                     format_mpls_eos_bit, fp->fp_eos);
293         break;
294     }
295     s = format (s, "/%d", fp->fp_len);
296
297     return (s);
298 }
299
300 int
301 fib_route_path_cmp (const fib_route_path_t *rpath1,
302                     const fib_route_path_t *rpath2)
303 {
304     int res;
305
306     res = ip46_address_cmp(&rpath1->frp_addr,
307                            &rpath2->frp_addr);
308
309     if (0 != res) return (res);
310
311     res = (rpath1->frp_sw_if_index - rpath2->frp_sw_if_index);
312
313     if (0 != res) return (res);
314
315     if (ip46_address_is_zero(&rpath1->frp_addr))
316     {
317         res = rpath1->frp_fib_index - rpath2->frp_fib_index;
318     }
319
320     return (res);
321 }
322
323 dpo_proto_t
324 fib_proto_to_dpo (fib_protocol_t fib_proto)
325 {
326     switch (fib_proto)
327     {
328     case FIB_PROTOCOL_IP6:
329         return (DPO_PROTO_IP6);
330     case FIB_PROTOCOL_IP4:
331         return (DPO_PROTO_IP4);
332     case FIB_PROTOCOL_MPLS:
333         return (DPO_PROTO_MPLS);
334     }
335     ASSERT(0);
336     return (0);
337 }
338
339 fib_protocol_t
340 dpo_proto_to_fib (dpo_proto_t dpo_proto)
341 {
342     switch (dpo_proto)
343     {
344     case DPO_PROTO_IP6:
345         return (FIB_PROTOCOL_IP6);
346     case DPO_PROTO_IP4:
347         return (FIB_PROTOCOL_IP4);
348     case DPO_PROTO_MPLS:
349         return (FIB_PROTOCOL_MPLS);
350     default:
351         break;
352     }
353     ASSERT(0);
354     return (0);
355 }
356
357 vnet_link_t
358 fib_proto_to_link (fib_protocol_t proto)
359 {
360     switch (proto)
361     {
362     case FIB_PROTOCOL_IP4:
363         return (VNET_LINK_IP4);
364     case FIB_PROTOCOL_IP6:
365         return (VNET_LINK_IP6);
366     case FIB_PROTOCOL_MPLS:
367         return (VNET_LINK_MPLS);
368     }
369     ASSERT(0);
370     return (0);
371 }
372
373 ip46_type_t
374 fib_proto_to_ip46 (fib_protocol_t fproto)
375 {
376     switch (fproto)
377     {
378     case FIB_PROTOCOL_IP4:
379         return (IP46_TYPE_IP4);
380     case FIB_PROTOCOL_IP6:
381         return (IP46_TYPE_IP6);
382     case FIB_PROTOCOL_MPLS:
383         return (IP46_TYPE_ANY);
384     }
385     ASSERT(0);
386     return (IP46_TYPE_ANY);
387 }
388
389 fib_protocol_t
390 fib_proto_from_ip46 (ip46_type_t iproto)
391 {
392     switch (iproto)
393     {
394     case IP46_TYPE_IP4:
395         return FIB_PROTOCOL_IP4;
396     case IP46_TYPE_IP6:
397         return FIB_PROTOCOL_IP6;
398     case IP46_TYPE_ANY:
399         ASSERT(0);
400         return FIB_PROTOCOL_IP4;
401     }
402
403     ASSERT(0);
404     return FIB_PROTOCOL_IP4;
405 }
406
407 fib_forward_chain_type_t
408 fib_forw_chain_type_from_dpo_proto (dpo_proto_t proto)
409 {
410     switch (proto)
411     {
412     case DPO_PROTO_IP4:
413         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
414     case DPO_PROTO_IP6:
415         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP6);
416     case DPO_PROTO_MPLS:
417         return (FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS);
418     case DPO_PROTO_ETHERNET:
419         return (FIB_FORW_CHAIN_TYPE_ETHERNET);
420     case DPO_PROTO_NSH:
421         return (FIB_FORW_CHAIN_TYPE_NSH);
422     case DPO_PROTO_BIER:
423         return (FIB_FORW_CHAIN_TYPE_BIER);
424     }
425     ASSERT(0);
426     return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
427 }
428
429 fib_forward_chain_type_t
430 fib_forw_chain_type_from_fib_proto (fib_protocol_t proto)
431 {
432     switch (proto)
433     {
434     case FIB_PROTOCOL_IP4:
435         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
436     case FIB_PROTOCOL_IP6:
437         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP6);
438     case FIB_PROTOCOL_MPLS:
439         return (FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS);
440     }
441     ASSERT(0);
442     return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
443 }
444
445 vnet_link_t
446 fib_forw_chain_type_to_link_type (fib_forward_chain_type_t fct)
447 {
448     switch (fct)
449     {
450     case FIB_FORW_CHAIN_TYPE_UNICAST_IP4:
451     case FIB_FORW_CHAIN_TYPE_MCAST_IP4:
452         return (VNET_LINK_IP4);
453     case FIB_FORW_CHAIN_TYPE_UNICAST_IP6:
454     case FIB_FORW_CHAIN_TYPE_MCAST_IP6:
455         return (VNET_LINK_IP6);
456     case FIB_FORW_CHAIN_TYPE_ETHERNET:
457         return (VNET_LINK_ETHERNET);
458     case FIB_FORW_CHAIN_TYPE_NSH:
459         return (VNET_LINK_NSH);
460     case FIB_FORW_CHAIN_TYPE_MPLS_EOS:
461     case FIB_FORW_CHAIN_TYPE_BIER:
462         /*
463          * insufficient information to to convert
464          */
465         ASSERT(0);
466         break;
467     case FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS:
468         return (VNET_LINK_MPLS);
469     }
470     return (VNET_LINK_IP4);
471 }
472
473 fib_forward_chain_type_t
474 fib_forw_chain_type_from_link_type (vnet_link_t link_type)
475 {
476     switch (link_type)
477     {
478     case VNET_LINK_IP4:
479         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
480     case VNET_LINK_IP6:
481         return (FIB_FORW_CHAIN_TYPE_UNICAST_IP6);
482     case VNET_LINK_MPLS:
483         return (FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS);
484     case VNET_LINK_ETHERNET:
485         return (FIB_FORW_CHAIN_TYPE_ETHERNET);
486     case VNET_LINK_NSH:
487         return (FIB_FORW_CHAIN_TYPE_NSH);
488     case VNET_LINK_ARP:
489         break;
490     }
491
492     ASSERT(0);
493     return (FIB_FORW_CHAIN_TYPE_UNICAST_IP4);
494 }
495
496 dpo_proto_t
497 fib_forw_chain_type_to_dpo_proto (fib_forward_chain_type_t fct)
498 {
499     switch (fct)
500     {
501     case FIB_FORW_CHAIN_TYPE_UNICAST_IP4:
502     case FIB_FORW_CHAIN_TYPE_MCAST_IP4:
503         return (DPO_PROTO_IP4);
504     case FIB_FORW_CHAIN_TYPE_UNICAST_IP6:
505     case FIB_FORW_CHAIN_TYPE_MCAST_IP6:
506         return (DPO_PROTO_IP6);
507     case FIB_FORW_CHAIN_TYPE_ETHERNET:
508         return (DPO_PROTO_ETHERNET);
509     case FIB_FORW_CHAIN_TYPE_NSH:
510         return (DPO_PROTO_NSH);
511     case FIB_FORW_CHAIN_TYPE_BIER:
512         return (DPO_PROTO_BIER);
513     case FIB_FORW_CHAIN_TYPE_MPLS_EOS:
514     case FIB_FORW_CHAIN_TYPE_MPLS_NON_EOS:
515         return (DPO_PROTO_MPLS);
516     }
517     return (DPO_PROTO_IP4);
518 }
519
520 uword
521 unformat_fib_route_path (unformat_input_t * input, va_list * args)
522 {
523     fib_route_path_t *rpath = va_arg (*args, fib_route_path_t *);
524     dpo_proto_t *payload_proto = va_arg (*args, void*);
525     u32 weight, preference, udp_encap_id, fi;
526     mpls_label_t out_label;
527     vnet_main_t *vnm;
528
529     vnm = vnet_get_main ();
530     clib_memset(rpath, 0, sizeof(*rpath));
531     rpath->frp_weight = 1;
532     rpath->frp_sw_if_index = ~0;
533
534     while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
535     {
536         if (unformat (input, "%U %U",
537                       unformat_ip4_address,
538                       &rpath->frp_addr.ip4,
539                       unformat_vnet_sw_interface, vnm,
540                       &rpath->frp_sw_if_index))
541         {
542             rpath->frp_proto = DPO_PROTO_IP4;
543         }
544         else if (unformat (input, "%U %U",
545                            unformat_ip6_address,
546                            &rpath->frp_addr.ip6,
547                            unformat_vnet_sw_interface, vnm,
548                            &rpath->frp_sw_if_index))
549         {
550             rpath->frp_proto = DPO_PROTO_IP6;
551         }
552         else if (unformat (input, "weight %u", &weight))
553         {
554             rpath->frp_weight = weight;
555         }
556         else if (unformat (input, "preference %u", &preference))
557         {
558             rpath->frp_preference = preference;
559         }
560         else if (unformat (input, "%U next-hop-table %d",
561                            unformat_ip4_address,
562                            &rpath->frp_addr.ip4,
563                            &rpath->frp_fib_index))
564         {
565             rpath->frp_sw_if_index = ~0;
566             rpath->frp_proto = DPO_PROTO_IP4;
567
568             /*
569              * the user enter table-ids, convert to index
570              */
571             fi = fib_table_find (FIB_PROTOCOL_IP4, rpath->frp_fib_index);
572             if (~0 == fi)
573                 return 0;
574             rpath->frp_fib_index = fi;
575         }
576         else if (unformat (input, "%U next-hop-table %d",
577                            unformat_ip6_address,
578                            &rpath->frp_addr.ip6,
579                            &rpath->frp_fib_index))
580         {
581             rpath->frp_sw_if_index = ~0;
582             rpath->frp_proto = DPO_PROTO_IP6;
583             fi = fib_table_find (FIB_PROTOCOL_IP6, rpath->frp_fib_index);
584             if (~0 == fi)
585                 return 0;
586             rpath->frp_fib_index = fi;
587         }
588         else if (unformat (input, "%U",
589                            unformat_ip4_address,
590                            &rpath->frp_addr.ip4))
591         {
592             /*
593              * the recursive next-hops are by default in the default table
594              */
595             rpath->frp_fib_index = 0;
596             rpath->frp_sw_if_index = ~0;
597             rpath->frp_proto = DPO_PROTO_IP4;
598         }
599         else if (unformat (input, "%U",
600                            unformat_ip6_address,
601                            &rpath->frp_addr.ip6))
602         {
603             rpath->frp_fib_index = 0;
604             rpath->frp_sw_if_index = ~0;
605             rpath->frp_proto = DPO_PROTO_IP6;
606         }
607         else if (unformat (input, "udp-encap %d", &udp_encap_id))
608         {
609             rpath->frp_udp_encap_id = udp_encap_id;
610             rpath->frp_flags |= FIB_ROUTE_PATH_UDP_ENCAP;
611             rpath->frp_proto = *payload_proto;
612         }
613         else if (unformat (input, "lookup in table %d", &rpath->frp_fib_index))
614         {
615             rpath->frp_proto = *payload_proto;
616             rpath->frp_sw_if_index = ~0;
617             rpath->frp_flags |= FIB_ROUTE_PATH_DEAG;
618         }
619         else if (unformat (input, "resolve-via-host"))
620         {
621             rpath->frp_flags |= FIB_ROUTE_PATH_RESOLVE_VIA_HOST;
622         }
623         else if (unformat (input, "resolve-via-attached"))
624         {
625             rpath->frp_flags |= FIB_ROUTE_PATH_RESOLVE_VIA_ATTACHED;
626         }
627         else if (unformat (input, "pop-pw-cw"))
628         {
629             rpath->frp_flags |= FIB_ROUTE_PATH_POP_PW_CW;
630         }
631         else if (unformat (input,
632                            "ip4-lookup-in-table %d",
633                            &rpath->frp_fib_index))
634         {
635             rpath->frp_proto = DPO_PROTO_IP4;
636             *payload_proto = DPO_PROTO_IP4;
637             fi = fib_table_find (FIB_PROTOCOL_IP4, rpath->frp_fib_index);
638             if (~0 == fi)
639                 return 0;
640             rpath->frp_fib_index = fi;
641         }
642         else if (unformat (input,
643                            "ip6-lookup-in-table %d",
644                            &rpath->frp_fib_index))
645         {
646             rpath->frp_proto = DPO_PROTO_IP6;
647             *payload_proto = DPO_PROTO_IP6;
648             fi = fib_table_find (FIB_PROTOCOL_IP6, rpath->frp_fib_index);
649             if (~0 == fi)
650                 return 0;
651             rpath->frp_fib_index = fi;
652         }
653         else if (unformat (input,
654                            "mpls-lookup-in-table %d",
655                            &rpath->frp_fib_index))
656         {
657             rpath->frp_proto = DPO_PROTO_MPLS;
658             *payload_proto = DPO_PROTO_MPLS;
659             fi = fib_table_find (FIB_PROTOCOL_MPLS, rpath->frp_fib_index);
660             if (~0 == fi)
661                 return 0;
662             rpath->frp_fib_index = fi;
663         }
664         else if (unformat (input, "src-lookup"))
665         {
666             rpath->frp_flags |= FIB_ROUTE_PATH_SOURCE_LOOKUP;
667         }
668         else if (unformat (input,
669                            "l2-input-on %U",
670                            unformat_vnet_sw_interface, vnm,
671                            &rpath->frp_sw_if_index))
672         {
673             rpath->frp_proto = DPO_PROTO_ETHERNET;
674             *payload_proto = DPO_PROTO_ETHERNET;
675             rpath->frp_flags |= FIB_ROUTE_PATH_INTF_RX;
676         }
677         else if (unformat (input, "via-label %U",
678                            unformat_mpls_unicast_label,
679                            &rpath->frp_local_label))
680         {
681             rpath->frp_eos = MPLS_NON_EOS;
682             rpath->frp_proto = DPO_PROTO_MPLS;
683             rpath->frp_sw_if_index = ~0;
684         }
685         else if (unformat (input, "rx-ip4 %U",
686                            unformat_vnet_sw_interface, vnm,
687                            &rpath->frp_sw_if_index))
688         {
689             rpath->frp_proto = DPO_PROTO_IP4;
690             rpath->frp_flags = FIB_ROUTE_PATH_INTF_RX;
691         }
692       else if (unformat (input, "local"))
693         {
694           clib_memset (&rpath->frp_addr, 0, sizeof (rpath->frp_addr));
695           rpath->frp_sw_if_index = ~0;
696           rpath->frp_weight = 1;
697           rpath->frp_flags |= FIB_ROUTE_PATH_LOCAL;
698         }
699       else if (unformat (input, "%U",
700                          unformat_mfib_itf_flags, &rpath->frp_mitf_flags))
701         ;
702       else if (unformat (input, "out-labels"))
703         {
704             while (unformat (input, "%U",
705                              unformat_mpls_unicast_label, &out_label))
706             {
707                 fib_mpls_label_t fml = {
708                     .fml_value = out_label,
709                 };
710                 vec_add1(rpath->frp_label_stack, fml);
711             }
712         }
713         else if (unformat (input, "%U",
714                            unformat_vnet_sw_interface, vnm,
715                            &rpath->frp_sw_if_index))
716         {
717             rpath->frp_proto = *payload_proto;
718         }
719         else if (unformat (input, "via"))
720         {
721             /* new path, back up and return */
722             unformat_put_input (input);
723             unformat_put_input (input);
724             unformat_put_input (input);
725             unformat_put_input (input);
726             break;
727         }
728         else
729         {
730             return (0);
731         }
732     }
733
734     return (1);
735 }