]> git.proxmox.com Git - mirror_frr.git/commitdiff
bgpd: Implementing a hash table for connected address - ipv4/ipv6
authorbisdhdh <biswajit.sadhu@gmail.com>
Wed, 30 Oct 2019 09:42:25 +0000 (15:12 +0530)
committerbisdhdh <biswajit.sadhu@gmail.com>
Tue, 19 Nov 2019 19:53:11 +0000 (01:23 +0530)
* IPv6 routes received via a ibgp session with one of its own interface as
nexthop are getting installed in the BGP table.
*A common table to be implemented should take cares of both
ipv4 and ipv6 connected addresses.

Signed-off-by: Biswajit Sadhu sadhub@vmware.com
bgpd/bgp_evpn.c
bgpd/bgp_nexthop.c
bgpd/bgp_nexthop.h
bgpd/bgp_route.c

index 35d54983dce0a9458e3fb2daf043933f99286e07..f3c514fb15f659f0cfce470eccc5cff41058d051 100644 (file)
@@ -5395,8 +5395,9 @@ int bgp_filter_evpn_routes_upon_martian_nh_change(struct bgp *bgp)
                                if (!(pi->type == ZEBRA_ROUTE_BGP
                                      && pi->sub_type == BGP_ROUTE_NORMAL))
                                        continue;
-
-                               if (bgp_nexthop_self(bgp, pi->attr->nexthop)) {
+                               if (bgp_nexthop_self(bgp, afi,
+                                               pi->type, pi->sub_type,
+                                               pi->attr, rn)) {
 
                                        char attr_str[BUFSIZ] = {0};
                                        char pbuf[PREFIX_STRLEN];
index 812f0ce16e47429d9cd6e3f7f76984158d1fa328..7116c809412612780732da03ae82f87805eef5b3 100644 (file)
@@ -180,7 +180,7 @@ void bgp_tip_del(struct bgp *bgp, struct in_addr *tip)
 
 /* BGP own address structure */
 struct bgp_addr {
-       struct in_addr addr;
+       struct prefix *p;
        struct list *ifp_name_list;
 };
 
@@ -190,9 +190,19 @@ static void show_address_entry(struct hash_bucket *bucket, void *args)
        struct bgp_addr *addr = (struct bgp_addr *)bucket->data;
        char *name;
        struct listnode *node;
-
-       vty_out(vty, "addr: %s, count: %d : ", inet_ntoa(addr->addr),
-               addr->ifp_name_list->count);
+       char str[INET6_ADDRSTRLEN] = {0};
+
+       if (addr->p->family == AF_INET) {
+               vty_out(vty, "addr: %s, count: %d : ", inet_ntop(AF_INET,
+                               &(addr->p->u.prefix4),
+                               str, INET_ADDRSTRLEN),
+                               addr->ifp_name_list->count);
+       } else if (addr->p->family == AF_INET6) {
+               vty_out(vty, "addr: %s, count: %d : ", inet_ntop(AF_INET6,
+                               &(addr->p->u.prefix6),
+                               str, INET6_ADDRSTRLEN),
+                               addr->ifp_name_list->count);
+       }
 
        for (ALL_LIST_ELEMENTS_RO(addr->ifp_name_list, node, name)) {
                vty_out(vty, " %s,", name);
@@ -217,11 +227,12 @@ static void bgp_address_hash_string_del(void *val)
 
 static void *bgp_address_hash_alloc(void *p)
 {
-       const struct in_addr *val = (const struct in_addr *)p;
-       struct bgp_addr *addr;
+       struct bgp_addr *copy_addr = p;
+       struct bgp_addr *addr = NULL;
 
        addr = XMALLOC(MTYPE_BGP_ADDR, sizeof(struct bgp_addr));
-       addr->addr.s_addr = val->s_addr;
+       addr->p = prefix_new();
+       prefix_copy(addr->p, copy_addr->p);
 
        addr->ifp_name_list = list_new();
        addr->ifp_name_list->del = bgp_address_hash_string_del;
@@ -233,6 +244,7 @@ static void bgp_address_hash_free(void *data)
 {
        struct bgp_addr *addr = data;
 
+       prefix_free(&addr->p);
        list_delete(&addr->ifp_name_list);
        XFREE(MTYPE_BGP_ADDR, addr);
 }
@@ -241,7 +253,7 @@ static unsigned int bgp_address_hash_key_make(const void *p)
 {
        const struct bgp_addr *addr = p;
 
-       return jhash_1word(addr->addr.s_addr, 0);
+       return prefix_hash_key((const void *)(addr->p));
 }
 
 static bool bgp_address_hash_cmp(const void *p1, const void *p2)
@@ -249,14 +261,14 @@ static bool bgp_address_hash_cmp(const void *p1, const void *p2)
        const struct bgp_addr *addr1 = p1;
        const struct bgp_addr *addr2 = p2;
 
-       return addr1->addr.s_addr == addr2->addr.s_addr;
+       return prefix_same(addr1->p, addr2->p);
 }
 
 void bgp_address_init(struct bgp *bgp)
 {
        bgp->address_hash =
                hash_create(bgp_address_hash_key_make, bgp_address_hash_cmp,
-                           "BGP Address Hash");
+                               "BGP Connected Address Hash");
 }
 
 void bgp_address_destroy(struct bgp *bgp)
@@ -276,7 +288,12 @@ static void bgp_address_add(struct bgp *bgp, struct connected *ifc,
        struct listnode *node;
        char *name;
 
-       tmp.addr = p->u.prefix4;
+       tmp.p = p;
+
+       if (tmp.p->family == AF_INET)
+               tmp.p->prefixlen = IPV4_MAX_BITLEN;
+       else if (tmp.p->family == AF_INET6)
+               tmp.p->prefixlen = IPV6_MAX_BITLEN;
 
        addr = hash_get(bgp->address_hash, &tmp, bgp_address_hash_alloc);
 
@@ -298,7 +315,12 @@ static void bgp_address_del(struct bgp *bgp, struct connected *ifc,
        struct listnode *node;
        char *name;
 
-       tmp.addr = p->u.prefix4;
+       tmp.p = p;
+
+       if (tmp.p->family == AF_INET)
+               tmp.p->prefixlen = IPV4_MAX_BITLEN;
+       else if (tmp.p->family == AF_INET6)
+               tmp.p->prefixlen = IPV6_MAX_BITLEN;
 
        addr = hash_lookup(bgp->address_hash, &tmp);
        /* may have been deleted earlier by bgp_interface_down() */
@@ -379,6 +401,8 @@ void bgp_connected_add(struct bgp *bgp, struct connected *ifc)
                if (IN6_IS_ADDR_LINKLOCAL(&p.u.prefix6))
                        return;
 
+               bgp_address_add(bgp, ifc, addr);
+
                rn = bgp_node_get(bgp->connected_table[AFI_IP6],
                                  (struct prefix *)&p);
 
@@ -419,6 +443,8 @@ void bgp_connected_delete(struct bgp *bgp, struct connected *ifc)
                if (IN6_IS_ADDR_LINKLOCAL(&p.u.prefix6))
                        return;
 
+               bgp_address_del(bgp, ifc, addr);
+
                rn = bgp_node_lookup(bgp->connected_table[AFI_IP6],
                                     (struct prefix *)&p);
        }
@@ -453,21 +479,85 @@ static void bgp_connected_cleanup(struct route_table *table,
        }
 }
 
-int bgp_nexthop_self(struct bgp *bgp, struct in_addr nh_addr)
-{
-       struct bgp_addr tmp, *addr;
-       struct tip_addr tmp_tip, *tip;
-
-       tmp.addr = nh_addr;
+int bgp_nexthop_self(struct bgp *bgp, afi_t afi, uint8_t type, uint8_t sub_type,
+               struct attr *attr, struct bgp_node *rn)
+{
+       struct prefix p = {0};
+       afi_t new_afi = afi;
+       struct bgp_addr tmp_addr = {0}, *addr = NULL;
+       struct tip_addr tmp_tip, *tip = NULL;
+
+       bool is_bgp_static_route = ((type == ZEBRA_ROUTE_BGP)
+                       && (sub_type == BGP_ROUTE_STATIC))
+                       ? true
+                       : false;
+
+       if (!is_bgp_static_route)
+               new_afi = BGP_ATTR_NEXTHOP_AFI_IP6(attr) ? AFI_IP6 : AFI_IP;
+
+       switch (new_afi) {
+       case AFI_IP:
+               p.family = AF_INET;
+               if (is_bgp_static_route) {
+                       p.u.prefix4 = rn->p.u.prefix4;
+                       p.prefixlen = rn->p.prefixlen;
+               } else {
+                       /* Here we need to find out which nexthop to be used*/
+                       if (attr->flag &
+                                       ATTR_FLAG_BIT(BGP_ATTR_NEXT_HOP)) {
+
+                               p.u.prefix4 = attr->nexthop;
+                               p.prefixlen = IPV4_MAX_BITLEN;
+
+                       } else if ((attr->mp_nexthop_len) &&
+                                       ((attr->mp_nexthop_len ==
+                                         BGP_ATTR_NHLEN_IPV4) ||
+                                        (attr->mp_nexthop_len ==
+                                         BGP_ATTR_NHLEN_VPNV4))) {
+                               p.u.prefix4 =
+                                       attr->mp_nexthop_global_in;
+                               p.prefixlen = IPV4_MAX_BITLEN;
+                       } else
+                               return 0;
+               }
+               break;
+       case AFI_IP6:
+               p.family = AF_INET6;
+
+               if (is_bgp_static_route) {
+                       p.u.prefix6 = rn->p.u.prefix6;
+                       p.prefixlen = rn->p.prefixlen;
+               } else {
+                       p.u.prefix6 = attr->mp_nexthop_global;
+                       p.prefixlen = IPV6_MAX_BITLEN;
+               }
+               break;
+       default:
+               break;
+       }
 
-       addr = hash_lookup(bgp->address_hash, &tmp);
+       tmp_addr.p = &p;
+       addr = hash_lookup(bgp->address_hash, &tmp_addr);
        if (addr)
                return 1;
 
-       tmp_tip.addr = nh_addr;
-       tip = hash_lookup(bgp->tip_hash, &tmp_tip);
-       if (tip)
-               return 1;
+       if (new_afi == AFI_IP) {
+               memset(&tmp_tip, 0, sizeof(struct tip_addr));
+               tmp_tip.addr = attr->nexthop;
+
+               if (attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEXT_HOP)) {
+                       tmp_tip.addr = attr->nexthop;
+               } else if ((attr->mp_nexthop_len) &&
+                               ((attr->mp_nexthop_len == BGP_ATTR_NHLEN_IPV4)
+                                || (attr->mp_nexthop_len ==
+                                        BGP_ATTR_NHLEN_VPNV4))) {
+                       tmp_tip.addr = attr->mp_nexthop_global_in;
+               }
+
+               tip = hash_lookup(bgp->tip_hash, &tmp_tip);
+               if (tip)
+                       return 1;
+       }
 
        return 0;
 }
index d35f1ad5207d64b855cfdd3b0582c61478852276..af4c0bc0470595c6feee0af243d2516c623b3151 100644 (file)
@@ -88,7 +88,9 @@ extern int bgp_subgrp_multiaccess_check_v6(struct in6_addr nexthop,
 extern int bgp_multiaccess_check_v4(struct in_addr nexthop, struct peer *peer);
 extern int bgp_multiaccess_check_v6(struct in6_addr nexthop, struct peer *peer);
 extern int bgp_config_write_scan_time(struct vty *);
-extern int bgp_nexthop_self(struct bgp *, struct in_addr);
+extern int bgp_nexthop_self(struct bgp *bgp, afi_t afi, uint8_t type,
+                               uint8_t sub_type, struct attr *attr,
+                               struct bgp_node *rn);
 extern struct bgp_nexthop_cache *bnc_new(void);
 extern void bnc_free(struct bgp_nexthop_cache *bnc);
 extern void bnc_nexthop_free(struct bgp_nexthop_cache *bnc);
index 4fb4faebc2f14d34a18358d27b9096db3ebfae58..fb2eb10dd931fef3ff69c3094d8910884232730f 100644 (file)
@@ -2970,7 +2970,8 @@ static bool overlay_index_equal(afi_t afi, struct bgp_path_info *path,
 
 /* Check if received nexthop is valid or not. */
 static int bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
-                                     struct attr *attr)
+                               uint8_t type, uint8_t stype,
+                               struct attr *attr, struct bgp_node *rn)
 {
        int ret = 0;
 
@@ -2983,7 +2984,8 @@ static int bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
        if (attr->flag & ATTR_FLAG_BIT(BGP_ATTR_NEXT_HOP)) {
                if (attr->nexthop.s_addr == 0
                    || IPV4_CLASS_DE(ntohl(attr->nexthop.s_addr))
-                   || bgp_nexthop_self(bgp, attr->nexthop))
+                   || bgp_nexthop_self(bgp, afi, type, stype,
+                       attr, rn))
                        return 1;
        }
 
@@ -2999,8 +3001,8 @@ static int bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
                        ret = (attr->mp_nexthop_global_in.s_addr == 0
                               || IPV4_CLASS_DE(ntohl(
                                          attr->mp_nexthop_global_in.s_addr))
-                              || bgp_nexthop_self(bgp,
-                                                  attr->mp_nexthop_global_in));
+                              || bgp_nexthop_self(bgp, afi, type, stype,
+                                                  attr, rn));
                        break;
 
                case BGP_ATTR_NHLEN_IPV6_GLOBAL:
@@ -3009,7 +3011,9 @@ static int bgp_update_martian_nexthop(struct bgp *bgp, afi_t afi, safi_t safi,
                        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));
+                                         &attr->mp_nexthop_global)
+                              || bgp_nexthop_self(bgp, afi, type, stype,
+                                                  attr, rn));
                        break;
 
                default:
@@ -3042,6 +3046,9 @@ int bgp_update(struct peer *peer, struct prefix *p, uint32_t addpath_id,
        int do_loop_check = 1;
        int has_valid_label = 0;
        afi_t nh_afi;
+       uint8_t pi_type = 0;
+       uint8_t pi_sub_type = 0;
+
 #if ENABLE_BGP_VNC
        int vnc_implicit_withdraw = 0;
 #endif
@@ -3187,9 +3194,15 @@ int bgp_update(struct peer *peer, struct prefix *p, uint32_t addpath_id,
                }
        }
 
+       if (pi) {
+               pi_type = pi->type;
+               pi_sub_type = pi->sub_type;
+       }
+
        /* next hop check.  */
        if (!CHECK_FLAG(peer->flags, PEER_FLAG_IS_RFAPI_HD)
-           && bgp_update_martian_nexthop(bgp, afi, safi, &new_attr)) {
+               && bgp_update_martian_nexthop(bgp, afi, safi, pi_type,
+               pi_sub_type, &new_attr, rn)) {
                peer->stat_pfx_nh_invalid++;
                reason = "martian or self next-hop;";
                bgp_attr_flush(&new_attr);