1 /* Ethernet-VPN Attribute handling file
2 * Copyright (C) 2016 6WIND
4 * This file is part of FRRouting.
6 * FRRouting is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
11 * FRRouting is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; see the file COPYING; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
31 #include "bgpd/bgpd.h"
32 #include "bgpd/bgp_attr.h"
33 #include "bgpd/bgp_route.h"
34 #include "bgpd/bgp_attr_evpn.h"
35 #include "bgpd/bgp_ecommunity.h"
36 #include "bgpd/bgp_evpn.h"
37 #include "bgpd/bgp_evpn_private.h"
39 void bgp_add_routermac_ecom(struct attr
*attr
, struct ethaddr
*routermac
)
41 struct ecommunity_val routermac_ecom
;
43 memset(&routermac_ecom
, 0, sizeof(struct ecommunity_val
));
44 routermac_ecom
.val
[0] = ECOMMUNITY_ENCODE_EVPN
;
45 routermac_ecom
.val
[1] = ECOMMUNITY_EVPN_SUBTYPE_ROUTERMAC
;
46 memcpy(&routermac_ecom
.val
[2], routermac
->octet
, ETH_ALEN
);
47 if (!attr
->ecommunity
)
48 attr
->ecommunity
= ecommunity_new();
49 ecommunity_add_val(attr
->ecommunity
, &routermac_ecom
, false, false);
50 ecommunity_str(attr
->ecommunity
);
54 * returns 1 on success, 0 otherwise
55 * format accepted: AA:BB:CC:DD:EE:FF:GG:HH:II:JJ
56 * if id is null, check only is done
58 bool str2esi(const char *str
, esi_t
*id
)
60 unsigned int a
[ESI_BYTES
];
65 if (sscanf(str
, "%2x:%2x:%2x:%2x:%2x:%2x:%2x:%2x:%2x:%2x", a
+ 0, a
+ 1,
66 a
+ 2, a
+ 3, a
+ 4, a
+ 5, a
+ 6, a
+ 7, a
+ 8, a
+ 9)
68 /* error in incoming str length */
71 /* valid mac address */
74 for (i
= 0; i
< ESI_BYTES
; ++i
)
75 id
->val
[i
] = a
[i
] & 0xff;
79 char *ecom_mac2str(char *ecom_mac
)
86 return prefix_mac2str((struct ethaddr
*)en
, NULL
, 0);
89 /* Fetch router-mac from extended community */
90 bool bgp_attr_rmac(struct attr
*attr
, struct ethaddr
*rmac
)
93 struct ecommunity
*ecom
;
95 ecom
= attr
->ecommunity
;
96 if (!ecom
|| !ecom
->size
)
99 /* If there is a router mac extended community, set RMAC in attr */
100 for (i
= 0; i
< ecom
->size
; i
++) {
103 uint8_t sub_type
= 0;
105 pnt
= (ecom
->val
+ (i
* ECOMMUNITY_SIZE
));
109 if (!(type
== ECOMMUNITY_ENCODE_EVPN
110 && sub_type
== ECOMMUNITY_EVPN_SUBTYPE_ROUTERMAC
))
113 memcpy(rmac
, pnt
, ETH_ALEN
);
120 * return true if attr contains default gw extended community
122 uint8_t bgp_attr_default_gw(struct attr
*attr
)
124 struct ecommunity
*ecom
;
127 ecom
= attr
->ecommunity
;
128 if (!ecom
|| !ecom
->size
)
131 /* If there is a default gw extendd community return true otherwise
133 for (i
= 0; i
< ecom
->size
; i
++) {
135 uint8_t type
, sub_type
;
137 pnt
= (ecom
->val
+ (i
* ECOMMUNITY_SIZE
));
141 if ((type
== ECOMMUNITY_ENCODE_OPAQUE
142 && sub_type
== ECOMMUNITY_EVPN_SUBTYPE_DEF_GW
))
150 * Fetch and return the DF preference and algorithm from
151 * DF election extended community, if present, else 0.
153 uint16_t bgp_attr_df_pref_from_ec(struct attr
*attr
, uint8_t *alg
)
155 struct ecommunity
*ecom
;
157 uint16_t df_pref
= 0;
159 *alg
= EVPN_MH_DF_ALG_SERVICE_CARVING
;
160 ecom
= attr
->ecommunity
;
161 if (!ecom
|| !ecom
->size
)
164 for (i
= 0; i
< ecom
->size
; i
++) {
166 uint8_t type
, sub_type
;
168 pnt
= (ecom
->val
+ (i
* ECOMMUNITY_SIZE
));
171 if (!(type
== ECOMMUNITY_ENCODE_EVPN
172 && sub_type
== ECOMMUNITY_EVPN_SUBTYPE_DF_ELECTION
))
175 *alg
= (*pnt
++) & ECOMMUNITY_EVPN_SUBTYPE_DF_ALG_BITS
;
178 pnt
= ptr_get_be16(pnt
, &df_pref
);
179 (void)pnt
; /* consume value */
187 * Fetch and return the sequence number from MAC Mobility extended
188 * community, if present, else 0.
190 uint32_t bgp_attr_mac_mobility_seqnum(struct attr
*attr
, uint8_t *sticky
)
192 struct ecommunity
*ecom
;
196 ecom
= attr
->ecommunity
;
197 if (!ecom
|| !ecom
->size
)
200 /* If there is a MAC Mobility extended community, return its
202 * TODO: RFC is silent on handling of multiple MAC mobility extended
203 * communities for the same route. We will bail out upon the first
206 for (i
= 0; i
< ecom
->size
; i
++) {
208 uint8_t type
, sub_type
;
211 pnt
= (ecom
->val
+ (i
* ECOMMUNITY_SIZE
));
214 if (!(type
== ECOMMUNITY_ENCODE_EVPN
215 && sub_type
== ECOMMUNITY_EVPN_SUBTYPE_MACMOBILITY
))
219 if (flags
& ECOMMUNITY_EVPN_SUBTYPE_MACMOBILITY_FLAG_STICKY
)
225 pnt
= ptr_get_be32(pnt
, &seq_num
);
226 (void)pnt
; /* consume value */
234 * return true if attr contains router flag extended community
236 void bgp_attr_evpn_na_flag(struct attr
*attr
,
237 uint8_t *router_flag
, bool *proxy
)
239 struct ecommunity
*ecom
;
243 ecom
= attr
->ecommunity
;
244 if (!ecom
|| !ecom
->size
)
247 /* If there is a evpn na extendd community set router_flag */
248 for (i
= 0; i
< ecom
->size
; i
++) {
250 uint8_t type
, sub_type
;
252 pnt
= (ecom
->val
+ (i
* ECOMMUNITY_SIZE
));
256 if (type
== ECOMMUNITY_ENCODE_EVPN
&&
257 sub_type
== ECOMMUNITY_EVPN_SUBTYPE_ND
) {
260 if (val
& ECOMMUNITY_EVPN_SUBTYPE_ND_ROUTER_FLAG
)
263 if (val
& ECOMMUNITY_EVPN_SUBTYPE_PROXY_FLAG
)
271 /* dst prefix must be AF_INET or AF_INET6 prefix, to forge EVPN prefix */
272 extern int bgp_build_evpn_prefix(int evpn_type
, uint32_t eth_tag
,
275 struct evpn_addr
*p_evpn_p
;
277 struct prefix
*src
= &p2
;
279 if (!dst
|| dst
->family
== 0)
281 /* store initial prefix in src */
282 prefix_copy(src
, dst
);
283 memset(dst
, 0, sizeof(struct prefix
));
284 p_evpn_p
= &(dst
->u
.prefix_evpn
);
285 dst
->family
= AF_EVPN
;
286 p_evpn_p
->route_type
= evpn_type
;
287 if (evpn_type
== BGP_EVPN_IP_PREFIX_ROUTE
) {
288 p_evpn_p
->prefix_addr
.eth_tag
= eth_tag
;
289 p_evpn_p
->prefix_addr
.ip_prefix_length
= p2
.prefixlen
;
290 if (src
->family
== AF_INET
) {
291 SET_IPADDR_V4(&p_evpn_p
->prefix_addr
.ip
);
292 memcpy(&p_evpn_p
->prefix_addr
.ip
.ipaddr_v4
,
294 sizeof(struct in_addr
));
295 dst
->prefixlen
= (uint16_t)PREFIX_LEN_ROUTE_TYPE_5_IPV4
;
297 SET_IPADDR_V6(&p_evpn_p
->prefix_addr
.ip
);
298 memcpy(&p_evpn_p
->prefix_addr
.ip
.ipaddr_v6
,
300 sizeof(struct in6_addr
));
301 dst
->prefixlen
= (uint16_t)PREFIX_LEN_ROUTE_TYPE_5_IPV6
;
308 extern bool is_zero_gw_ip(const union gw_addr
*gw_ip
, const afi_t afi
)
310 if (afi
== AF_INET6
&& IN6_IS_ADDR_UNSPECIFIED(&gw_ip
->ipv6
))
313 if (afi
== AF_INET
&& gw_ip
->ipv4
.s_addr
== INADDR_ANY
)