summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDonald Sharp <donaldsharp72@gmail.com>2024-11-19 09:29:24 -0500
committerGitHub <noreply@github.com>2024-11-19 09:29:24 -0500
commit321638388dacc9062c30a7658911cee2f16791f1 (patch)
tree6168cd7ce355d4da35c5c7338cf6d57a7a963732
parent81c2fe3453059e5aeb99f5a27c52e9f8c51805b6 (diff)
parent91790796bc731fa7d6f1d8a80825141adcc0593d (diff)
Merge pull request #17448 from opensourcerouting/fix/backport_65a43b57efd60c4fdf80c935750046ba861ec79f_10.0
bgpd: Validate both nexthop information (NEXTHOP and NLRI) (backport)
-rw-r--r--bgpd/bgp_route.c50
1 files changed, 22 insertions, 28 deletions
diff --git a/bgpd/bgp_route.c b/bgpd/bgp_route.c
index 1ba85adec5..3526d5b1d6 100644
--- a/bgpd/bgp_route.c
+++ b/bgpd/bgp_route.c
@@ -4030,7 +4030,7 @@ bool bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
uint8_t type, uint8_t stype, struct attr *attr,
struct bgp_dest *dest)
{
- bool ret = false;
+ bool nh_invalid = false;
bool is_bgp_static_route =
(type == ZEBRA_ROUTE_BGP && stype == BGP_ROUTE_STATIC) ? true
: false;
@@ -4052,13 +4052,15 @@ bool bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
(safi != SAFI_UNICAST && safi != SAFI_MULTICAST && safi != SAFI_EVPN))
return false;
- /* If NEXT_HOP is present, validate it. */
- if (attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEXT_HOP)) {
- if (attr->nexthop.s_addr == INADDR_ANY ||
- !ipv4_unicast_valid(&attr->nexthop) ||
- bgp_nexthop_self(bgp, afi, type, stype, attr, dest))
- return true;
- }
+ /* If NEXT_HOP is present, validate it:
+ * The route can have both nexthop + mp_nexthop encoded as multiple NLRIs,
+ * and we MUST check if at least one of them is valid.
+ * E.g.: IPv6 prefix can be with nexthop: 0.0.0.0, and mp_nexthop: fc00::1.
+ */
+ if (CHECK_FLAG(attr->flag, ATTR_FLAG_BIT(BGP_ATTR_NEXT_HOP)))
+ nh_invalid = (attr->nexthop.s_addr == INADDR_ANY ||
+ !ipv4_unicast_valid(&attr->nexthop) ||
+ bgp_nexthop_self(bgp, afi, type, stype, attr, dest));
/* If MP_NEXTHOP is present, validate it. */
/* Note: For IPv6 nexthops, we only validate the global (1st) nexthop;
@@ -4073,39 +4075,31 @@ bool bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
switch (attr->mp_nexthop_len) {
case BGP_ATTR_NHLEN_IPV4:
case BGP_ATTR_NHLEN_VPNV4:
- ret = (attr->mp_nexthop_global_in.s_addr ==
- INADDR_ANY ||
- !ipv4_unicast_valid(
- &attr->mp_nexthop_global_in) ||
- bgp_nexthop_self(bgp, afi, type, stype, attr,
- dest));
+ nh_invalid = (attr->mp_nexthop_global_in.s_addr == INADDR_ANY ||
+ !ipv4_unicast_valid(&attr->mp_nexthop_global_in) ||
+ bgp_nexthop_self(bgp, afi, type, stype, attr, dest));
break;
case BGP_ATTR_NHLEN_IPV6_GLOBAL:
case BGP_ATTR_NHLEN_VPNV6_GLOBAL:
- ret = (IN6_IS_ADDR_UNSPECIFIED(
- &attr->mp_nexthop_global)
- || IN6_IS_ADDR_LOOPBACK(&attr->mp_nexthop_global)
- || IN6_IS_ADDR_MULTICAST(
- &attr->mp_nexthop_global)
- || bgp_nexthop_self(bgp, afi, type, stype, attr,
- dest));
+ nh_invalid = (IN6_IS_ADDR_UNSPECIFIED(&attr->mp_nexthop_global) ||
+ IN6_IS_ADDR_LOOPBACK(&attr->mp_nexthop_global) ||
+ IN6_IS_ADDR_MULTICAST(&attr->mp_nexthop_global) ||
+ bgp_nexthop_self(bgp, afi, type, stype, attr, dest));
break;
case BGP_ATTR_NHLEN_IPV6_GLOBAL_AND_LL:
- ret = (IN6_IS_ADDR_LOOPBACK(&attr->mp_nexthop_global)
- || IN6_IS_ADDR_MULTICAST(
- &attr->mp_nexthop_global)
- || bgp_nexthop_self(bgp, afi, type, stype, attr,
- dest));
+ nh_invalid = (IN6_IS_ADDR_LOOPBACK(&attr->mp_nexthop_global) ||
+ IN6_IS_ADDR_MULTICAST(&attr->mp_nexthop_global) ||
+ bgp_nexthop_self(bgp, afi, type, stype, attr, dest));
break;
default:
- ret = true;
+ nh_invalid = true;
break;
}
}
- return ret;
+ return nh_invalid;
}
static void bgp_attr_add_no_export_community(struct attr *attr)