/* * Zebra connect library for OSPFd * Copyright (C) 1997, 98, 99, 2000 Kunihiro Ishiguro, Toshiaki Takada * * This file is part of GNU Zebra. * * GNU Zebra is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2, or (at your option) any * later version. * * GNU Zebra is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; see the file COPYING; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include "thread.h" #include "command.h" #include "network.h" #include "prefix.h" #include "routemap.h" #include "table.h" #include "stream.h" #include "memory.h" #include "zclient.h" #include "filter.h" #include "plist.h" #include "log.h" #include "lib/bfd.h" #include "nexthop.h" #include "ospfd/ospfd.h" #include "ospfd/ospf_interface.h" #include "ospfd/ospf_ism.h" #include "ospfd/ospf_asbr.h" #include "ospfd/ospf_asbr.h" #include "ospfd/ospf_abr.h" #include "ospfd/ospf_lsa.h" #include "ospfd/ospf_dump.h" #include "ospfd/ospf_route.h" #include "ospfd/ospf_lsdb.h" #include "ospfd/ospf_neighbor.h" #include "ospfd/ospf_nsm.h" #include "ospfd/ospf_zebra.h" #include "ospfd/ospf_te.h" DEFINE_MTYPE_STATIC(OSPFD, OSPF_EXTERNAL, "OSPF External route table") DEFINE_MTYPE_STATIC(OSPFD, OSPF_REDISTRIBUTE, "OSPF Redistriute") DEFINE_MTYPE_STATIC(OSPFD, OSPF_DIST_ARGS, "OSPF Distribute arguments") DEFINE_HOOK(ospf_if_update, (struct interface * ifp), (ifp)) DEFINE_HOOK(ospf_if_delete, (struct interface * ifp), (ifp)) /* Zebra structure to hold current status. */ struct zclient *zclient = NULL; /* For registering threads. */ extern struct thread_master *master; /* Router-id update message from zebra. */ static int ospf_router_id_update_zebra(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct ospf *ospf = NULL; struct prefix router_id; zebra_router_id_update_read(zclient->ibuf, &router_id); if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) { char buf[PREFIX2STR_BUFFER]; prefix2str(&router_id, buf, sizeof(buf)); zlog_debug("Zebra rcvd: router id update %s vrf %s id %u", buf, ospf_vrf_id_to_name(vrf_id), vrf_id); } ospf = ospf_lookup_by_vrf_id(vrf_id); if (ospf != NULL) { ospf->router_id_zebra = router_id.u.prefix4; ospf_router_id_update(ospf); } else { if (IS_DEBUG_OSPF_EVENT) { char buf[PREFIX2STR_BUFFER]; prefix2str(&router_id, buf, sizeof(buf)); zlog_debug( "%s: ospf instance not found for vrf %s id %u router_id %s", __PRETTY_FUNCTION__, ospf_vrf_id_to_name(vrf_id), vrf_id, buf); } } return 0; } /* Inteface addition message from zebra. */ static int ospf_interface_add(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct interface *ifp = NULL; struct ospf *ospf = NULL; ifp = zebra_interface_add_read(zclient->ibuf, vrf_id); if (ifp == NULL) return 0; if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug( "Zebra: interface add %s vrf %s[%u] index %d flags %llx metric %d mtu %d", ifp->name, ospf_vrf_id_to_name(ifp->vrf_id), ifp->vrf_id, ifp->ifindex, (unsigned long long)ifp->flags, ifp->metric, ifp->mtu); assert(ifp->info); if (!OSPF_IF_PARAM_CONFIGURED(IF_DEF_PARAMS(ifp), type)) { SET_IF_PARAM(IF_DEF_PARAMS(ifp), type); IF_DEF_PARAMS(ifp)->type = ospf_default_iftype(ifp); } ospf = ospf_lookup_by_vrf_id(vrf_id); if (!ospf) return 0; ospf_if_update(ospf, ifp); hook_call(ospf_if_update, ifp); return 0; } static int ospf_interface_delete(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct interface *ifp; struct stream *s; struct route_node *rn; s = zclient->ibuf; /* zebra_interface_state_read() updates interface structure in iflist */ ifp = zebra_interface_state_read(s, vrf_id); if (ifp == NULL) return 0; if (if_is_up(ifp)) zlog_warn("Zebra: got delete of %s, but interface is still up", ifp->name); if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug( "Zebra: interface delete %s vrf %s[%u] index %d flags %llx metric %d mtu %d", ifp->name, ospf_vrf_id_to_name(ifp->vrf_id), ifp->vrf_id, ifp->ifindex, (unsigned long long)ifp->flags, ifp->metric, ifp->mtu); hook_call(ospf_if_delete, ifp); for (rn = route_top(IF_OIFS(ifp)); rn; rn = route_next(rn)) if (rn->info) ospf_if_free((struct ospf_interface *)rn->info); if_set_index(ifp, IFINDEX_INTERNAL); return 0; } static struct interface *zebra_interface_if_lookup(struct stream *s, vrf_id_t vrf_id) { char ifname_tmp[INTERFACE_NAMSIZ]; /* Read interface name. */ stream_get(ifname_tmp, s, INTERFACE_NAMSIZ); /* And look it up. */ return if_lookup_by_name(ifname_tmp, vrf_id); } static int ospf_interface_state_up(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct interface *ifp; struct ospf_interface *oi; struct route_node *rn; ifp = zebra_interface_if_lookup(zclient->ibuf, vrf_id); if (ifp == NULL) return 0; /* Interface is already up. */ if (if_is_operative(ifp)) { /* Temporarily keep ifp values. */ struct interface if_tmp; memcpy(&if_tmp, ifp, sizeof(struct interface)); zebra_interface_if_set_value(zclient->ibuf, ifp); if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug( "Zebra: Interface[%s] state update speed %u -> %u, bw %d -> %d", ifp->name, if_tmp.speed, ifp->speed, if_tmp.bandwidth, ifp->bandwidth); ospf_if_recalculate_output_cost(ifp); if (if_tmp.mtu != ifp->mtu) { if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug( "Zebra: Interface[%s] MTU change %u -> %u.", ifp->name, if_tmp.mtu, ifp->mtu); /* Must reset the interface (simulate down/up) when MTU * changes. */ ospf_if_reset(ifp); } return 0; } zebra_interface_if_set_value(zclient->ibuf, ifp); if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug("Zebra: Interface[%s] state change to up.", ifp->name); for (rn = route_top(IF_OIFS(ifp)); rn; rn = route_next(rn)) { if ((oi = rn->info) == NULL) continue; ospf_if_up(oi); } return 0; } static int ospf_interface_state_down(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct interface *ifp; struct ospf_interface *oi; struct route_node *node; ifp = zebra_interface_state_read(zclient->ibuf, vrf_id); if (ifp == NULL) return 0; if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) zlog_debug("Zebra: Interface[%s] state change to down.", ifp->name); for (node = route_top(IF_OIFS(ifp)); node; node = route_next(node)) { if ((oi = node->info) == NULL) continue; ospf_if_down(oi); } return 0; } static int ospf_interface_address_add(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct connected *c; struct ospf *ospf = NULL; c = zebra_interface_address_read(command, zclient->ibuf, vrf_id); if (c == NULL) return 0; if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) { char buf[PREFIX2STR_BUFFER]; prefix2str(c->address, buf, sizeof(buf)); zlog_debug("Zebra: interface %s address add %s vrf %s id %u", c->ifp->name, buf, ospf_vrf_id_to_name(vrf_id), vrf_id); } ospf = ospf_lookup_by_vrf_id(vrf_id); if (!ospf) return 0; ospf_if_update(ospf, c->ifp); hook_call(ospf_if_update, c->ifp); return 0; } static int ospf_interface_address_delete(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct connected *c; struct interface *ifp; struct ospf_interface *oi; struct route_node *rn; struct prefix p; c = zebra_interface_address_read(command, zclient->ibuf, vrf_id); if (c == NULL) return 0; if (IS_DEBUG_OSPF(zebra, ZEBRA_INTERFACE)) { char buf[PREFIX2STR_BUFFER]; prefix2str(c->address, buf, sizeof(buf)); zlog_debug("Zebra: interface %s address delete %s", c->ifp->name, buf); } ifp = c->ifp; p = *c->address; p.prefixlen = IPV4_MAX_PREFIXLEN; rn = route_node_lookup(IF_OIFS(ifp), &p); if (!rn) { connected_free(c); return 0; } assert(rn->info); oi = rn->info; route_unlock_node(rn); /* Call interface hook functions to clean up */ ospf_if_free(oi); hook_call(ospf_if_update, c->ifp); connected_free(c); return 0; } static int ospf_interface_link_params(int command, struct zclient *zclient, zebra_size_t length) { struct interface *ifp; ifp = zebra_interface_link_params_read(zclient->ibuf); if (ifp == NULL) return 0; /* Update TE TLV */ ospf_mpls_te_update_if(ifp); return 0; } /* VRF update for an interface. */ static int ospf_interface_vrf_update(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct interface *ifp = NULL; vrf_id_t new_vrf_id; ifp = zebra_interface_vrf_update_read(zclient->ibuf, vrf_id, &new_vrf_id); if (!ifp) return 0; if (IS_DEBUG_OSPF_EVENT) zlog_debug( "%s: Rx Interface %s VRF change vrf_id %u New vrf %s id %u", __PRETTY_FUNCTION__, ifp->name, vrf_id, ospf_vrf_id_to_name(new_vrf_id), new_vrf_id); /*if_update(ifp, ifp->name, strlen(ifp->name), new_vrf_id);*/ if_update_to_new_vrf(ifp, new_vrf_id); return 0; } void ospf_zebra_add(struct ospf *ospf, struct prefix_ipv4 *p, struct ospf_route * or) { struct zapi_route api; struct zapi_nexthop *api_nh; u_char distance; struct ospf_path *path; struct listnode *node; int count = 0; memset(&api, 0, sizeof(api)); api.vrf_id = ospf->vrf_id; api.type = ZEBRA_ROUTE_OSPF; api.instance = ospf->instance; api.safi = SAFI_UNICAST; memcpy(&api.prefix, p, sizeof(*p)); SET_FLAG(api.message, ZAPI_MESSAGE_NEXTHOP); /* Metric value. */ SET_FLAG(api.message, ZAPI_MESSAGE_METRIC); if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL) api.metric = or->cost + or->u.ext.type2_cost; else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL) api.metric = or->u.ext.type2_cost; else api.metric = or->cost; /* Check if path type is ASE */ if (((or->path_type == OSPF_PATH_TYPE1_EXTERNAL) || (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)) && (or->u.ext.tag > 0) && (or->u.ext.tag <= ROUTE_TAG_MAX)) { SET_FLAG(api.message, ZAPI_MESSAGE_TAG); api.tag = or->u.ext.tag; } /* Distance value. */ distance = ospf_distance_apply(ospf, p, or); if (distance) { SET_FLAG(api.message, ZAPI_MESSAGE_DISTANCE); api.distance = distance; } /* Nexthop, ifindex, distance and metric information. */ for (ALL_LIST_ELEMENTS_RO(or->paths, node, path)) { if (count >= MULTIPATH_NUM) break; api_nh = &api.nexthops[count]; #ifdef HAVE_NETLINK if (path->unnumbered || (path->nexthop.s_addr != INADDR_ANY && path->ifindex != 0)) { #else /* HAVE_NETLINK */ if (path->nexthop.s_addr != INADDR_ANY && path->ifindex != 0) { #endif /* HAVE_NETLINK */ api_nh->gate.ipv4 = path->nexthop; api_nh->ifindex = path->ifindex; api_nh->type = NEXTHOP_TYPE_IPV4_IFINDEX; } else if (path->nexthop.s_addr != INADDR_ANY) { api_nh->gate.ipv4 = path->nexthop; api_nh->type = NEXTHOP_TYPE_IPV4; } else { api_nh->ifindex = path->ifindex; api_nh->type = NEXTHOP_TYPE_IFINDEX; } api_nh->vrf_id = ospf->vrf_id; count++; if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) { char buf[2][INET_ADDRSTRLEN]; zlog_debug( "Zebra: Route add %s/%d nexthop %s, ifindex=%d", inet_ntop(AF_INET, &p->prefix, buf[0], sizeof(buf[0])), p->prefixlen, inet_ntop(AF_INET, &path->nexthop, buf[1], sizeof(buf[1])), path->ifindex); } } api.nexthop_num = count; zclient_route_send(ZEBRA_ROUTE_ADD, zclient, &api); } void ospf_zebra_delete(struct ospf *ospf, struct prefix_ipv4 *p, struct ospf_route * or) { struct zapi_route api; memset(&api, 0, sizeof(api)); api.vrf_id = ospf->vrf_id; api.type = ZEBRA_ROUTE_OSPF; api.instance = ospf->instance; api.safi = SAFI_UNICAST; memcpy(&api.prefix, p, sizeof(*p)); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) { char buf[INET_ADDRSTRLEN]; zlog_debug("Zebra: Route delete %s/%d", inet_ntop(AF_INET, &p->prefix, buf, sizeof(buf[0])), p->prefixlen); } zclient_route_send(ZEBRA_ROUTE_DELETE, zclient, &api); } void ospf_zebra_add_discard(struct ospf *ospf, struct prefix_ipv4 *p) { struct zapi_route api; memset(&api, 0, sizeof(api)); api.vrf_id = ospf->vrf_id; api.type = ZEBRA_ROUTE_OSPF; api.instance = ospf->instance; api.safi = SAFI_UNICAST; memcpy(&api.prefix, p, sizeof(*p)); zapi_route_set_blackhole(&api, BLACKHOLE_NULL); zclient_route_send(ZEBRA_ROUTE_ADD, zclient, &api); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug("Zebra: Route add discard %s/%d", inet_ntoa(p->prefix), p->prefixlen); } void ospf_zebra_delete_discard(struct ospf *ospf, struct prefix_ipv4 *p) { struct zapi_route api; memset(&api, 0, sizeof(api)); api.vrf_id = ospf->vrf_id; api.type = ZEBRA_ROUTE_OSPF; api.instance = ospf->instance; api.safi = SAFI_UNICAST; memcpy(&api.prefix, p, sizeof(*p)); zapi_route_set_blackhole(&api, BLACKHOLE_NULL); zclient_route_send(ZEBRA_ROUTE_DELETE, zclient, &api); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug("Zebra: Route delete discard %s/%d", inet_ntoa(p->prefix), p->prefixlen); } struct ospf_external *ospf_external_lookup(struct ospf *ospf, u_char type, u_short instance) { struct list *ext_list; struct listnode *node; struct ospf_external *ext; ext_list = ospf->external[type]; if (!ext_list) return (NULL); for (ALL_LIST_ELEMENTS_RO(ext_list, node, ext)) if (ext->instance == instance) return ext; return NULL; } struct ospf_external *ospf_external_add(struct ospf *ospf, u_char type, u_short instance) { struct list *ext_list; struct ospf_external *ext; ext = ospf_external_lookup(ospf, type, instance); if (ext) return ext; if (!ospf->external[type]) ospf->external[type] = list_new(); ext_list = ospf->external[type]; ext = (struct ospf_external *)XCALLOC(MTYPE_OSPF_EXTERNAL, sizeof(struct ospf_external)); ext->instance = instance; EXTERNAL_INFO(ext) = route_table_init(); listnode_add(ext_list, ext); return ext; } void ospf_external_del(struct ospf *ospf, u_char type, u_short instance) { struct ospf_external *ext; ext = ospf_external_lookup(ospf, type, instance); if (ext) { if (EXTERNAL_INFO(ext)) route_table_finish(EXTERNAL_INFO(ext)); listnode_delete(ospf->external[type], ext); if (!ospf->external[type]->count) list_delete_and_null(&ospf->external[type]); XFREE(MTYPE_OSPF_EXTERNAL, ext); } } struct ospf_redist *ospf_redist_lookup(struct ospf *ospf, u_char type, u_short instance) { struct list *red_list; struct listnode *node; struct ospf_redist *red; red_list = ospf->redist[type]; if (!red_list) return (NULL); for (ALL_LIST_ELEMENTS_RO(red_list, node, red)) if (red->instance == instance) return red; return NULL; } struct ospf_redist *ospf_redist_add(struct ospf *ospf, u_char type, u_short instance) { struct list *red_list; struct ospf_redist *red; red = ospf_redist_lookup(ospf, type, instance); if (red) return red; if (!ospf->redist[type]) ospf->redist[type] = list_new(); red_list = ospf->redist[type]; red = (struct ospf_redist *)XCALLOC(MTYPE_OSPF_REDISTRIBUTE, sizeof(struct ospf_redist)); red->instance = instance; red->dmetric.type = -1; red->dmetric.value = -1; listnode_add(red_list, red); return red; } void ospf_redist_del(struct ospf *ospf, u_char type, u_short instance) { struct ospf_redist *red; red = ospf_redist_lookup(ospf, type, instance); if (red) { listnode_delete(ospf->redist[type], red); if (!ospf->redist[type]->count) { list_delete_and_null(&ospf->redist[type]); } ospf_routemap_unset(red); XFREE(MTYPE_OSPF_REDISTRIBUTE, red); } } int ospf_is_type_redistributed(struct ospf *ospf, int type, u_short instance) { return (DEFAULT_ROUTE_TYPE(type) ? vrf_bitmap_check(zclient->default_information, ospf->vrf_id) : ((instance && redist_check_instance( &zclient->mi_redist[AFI_IP][type], instance)) || (!instance && vrf_bitmap_check( zclient->redist[AFI_IP][type], ospf->vrf_id)))); } int ospf_redistribute_set(struct ospf *ospf, int type, u_short instance, int mtype, int mvalue) { int force = 0; struct ospf_redist *red; red = ospf_redist_lookup(ospf, type, instance); if (ospf_is_type_redistributed(ospf, type, instance)) { if (mtype != red->dmetric.type) { red->dmetric.type = mtype; force = LSA_REFRESH_FORCE; } if (mvalue != red->dmetric.value) { red->dmetric.value = mvalue; force = LSA_REFRESH_FORCE; } ospf_external_lsa_refresh_type(ospf, type, instance, force); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "Redistribute[%s][%d]: Refresh Type[%d], Metric[%d]", ospf_redist_string(type), instance, metric_type(ospf, type, instance), metric_value(ospf, type, instance)); return CMD_SUCCESS; } red->dmetric.type = mtype; red->dmetric.value = mvalue; ospf_external_add(ospf, type, instance); zclient_redistribute(ZEBRA_REDISTRIBUTE_ADD, zclient, AFI_IP, type, instance, ospf->vrf_id); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "Redistribute[%s][%d] vrf id %u: Start Type[%d], Metric[%d]", ospf_redist_string(type), instance, ospf->vrf_id, metric_type(ospf, type, instance), metric_value(ospf, type, instance)); ospf_asbr_status_update(ospf, ++ospf->redistribute); return CMD_SUCCESS; } int ospf_redistribute_unset(struct ospf *ospf, int type, u_short instance) { if (type == zclient->redist_default && instance == zclient->instance) return CMD_SUCCESS; if (!ospf_is_type_redistributed(ospf, type, instance)) return CMD_SUCCESS; zclient_redistribute(ZEBRA_REDISTRIBUTE_DELETE, zclient, AFI_IP, type, instance, ospf->vrf_id); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug("Redistribute[%s][%d] vrf id %u: Stop", ospf_redist_string(type), instance, ospf->vrf_id); ospf_redist_del(ospf, type, instance); /* Remove the routes from OSPF table. */ ospf_redistribute_withdraw(ospf, type, instance); ospf_external_del(ospf, type, instance); ospf_asbr_status_update(ospf, --ospf->redistribute); return CMD_SUCCESS; } int ospf_redistribute_default_set(struct ospf *ospf, int originate, int mtype, int mvalue) { struct ospf_redist *red; ospf->default_originate = originate; red = ospf_redist_add(ospf, DEFAULT_ROUTE, 0); red->dmetric.type = mtype; red->dmetric.value = mvalue; ospf_external_add(ospf, DEFAULT_ROUTE, 0); if (ospf_is_type_redistributed(ospf, DEFAULT_ROUTE, 0)) { /* if ospf->default_originate changes value, is calling ospf_external_lsa_refresh_default sufficient to implement the change? */ ospf_external_lsa_refresh_default(ospf); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "Redistribute[%s]: Refresh Type[%d], Metric[%d]", ospf_redist_string(DEFAULT_ROUTE), metric_type(ospf, DEFAULT_ROUTE, 0), metric_value(ospf, DEFAULT_ROUTE, 0)); return CMD_SUCCESS; } zclient_redistribute_default(ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient, ospf->vrf_id); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]", metric_type(ospf, DEFAULT_ROUTE, 0), metric_value(ospf, DEFAULT_ROUTE, 0)); if (ospf->router_id.s_addr == 0) ospf->external_origin |= (1 << DEFAULT_ROUTE); else thread_add_timer(master, ospf_default_originate_timer, ospf, 1, NULL); ospf_asbr_status_update(ospf, ++ospf->redistribute); return CMD_SUCCESS; } int ospf_redistribute_default_unset(struct ospf *ospf) { if (!ospf_is_type_redistributed(ospf, DEFAULT_ROUTE, 0)) return CMD_SUCCESS; ospf->default_originate = DEFAULT_ORIGINATE_NONE; ospf_redist_del(ospf, DEFAULT_ROUTE, 0); zclient_redistribute_default(ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient, ospf->vrf_id); if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug("Redistribute[DEFAULT]: Stop"); // Pending: how does the external_info cleanup work in this case? ospf_asbr_status_update(ospf, --ospf->redistribute); return CMD_SUCCESS; } static int ospf_external_lsa_originate_check(struct ospf *ospf, struct external_info *ei) { /* If prefix is multicast, then do not originate LSA. */ if (IN_MULTICAST(htonl(ei->p.prefix.s_addr))) { zlog_info( "LSA[Type5:%s]: Not originate AS-external-LSA, " "Prefix belongs multicast", inet_ntoa(ei->p.prefix)); return 0; } /* Take care of default-originate. */ if (is_prefix_default(&ei->p)) if (ospf->default_originate == DEFAULT_ORIGINATE_NONE) { zlog_info( "LSA[Type5:0.0.0.0]: Not originate AS-external-LSA " "for default"); return 0; } return 1; } /* If connected prefix is OSPF enable interface, then do not announce. */ int ospf_distribute_check_connected(struct ospf *ospf, struct external_info *ei) { struct listnode *node; struct ospf_interface *oi; for (ALL_LIST_ELEMENTS_RO(ospf->oiflist, node, oi)) if (prefix_match(oi->address, (struct prefix *)&ei->p)) return 0; return 1; } /* return 1 if external LSA must be originated, 0 otherwise */ int ospf_redistribute_check(struct ospf *ospf, struct external_info *ei, int *changed) { struct route_map_set_values save_values; struct prefix_ipv4 *p = &ei->p; struct ospf_redist *red; u_char type = is_prefix_default(&ei->p) ? DEFAULT_ROUTE : ei->type; u_short instance = is_prefix_default(&ei->p) ? 0 : ei->instance; if (changed) *changed = 0; if (!ospf_external_lsa_originate_check(ospf, ei)) return 0; /* Take care connected route. */ if (type == ZEBRA_ROUTE_CONNECT && !ospf_distribute_check_connected(ospf, ei)) return 0; if (!DEFAULT_ROUTE_TYPE(type) && DISTRIBUTE_NAME(ospf, type)) /* distirbute-list exists, but access-list may not? */ if (DISTRIBUTE_LIST(ospf, type)) if (access_list_apply(DISTRIBUTE_LIST(ospf, type), p) == FILTER_DENY) { if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "Redistribute[%s]: %s/%d filtered by ditribute-list.", ospf_redist_string(type), inet_ntoa(p->prefix), p->prefixlen); return 0; } save_values = ei->route_map_set; ospf_reset_route_map_set_values(&ei->route_map_set); /* apply route-map if needed */ red = ospf_redist_lookup(ospf, type, instance); if (red && ROUTEMAP_NAME(red)) { int ret; ret = route_map_apply(ROUTEMAP(red), (struct prefix *)p, RMAP_OSPF, ei); if (ret == RMAP_DENYMATCH) { ei->route_map_set = save_values; if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "Redistribute[%s]: %s/%d filtered by route-map.", ospf_redist_string(type), inet_ntoa(p->prefix), p->prefixlen); return 0; } /* check if 'route-map set' changed something */ if (changed) *changed = !ospf_route_map_set_compare( &ei->route_map_set, &save_values); } return 1; } /* OSPF route-map set for redistribution */ void ospf_routemap_set(struct ospf_redist *red, const char *name) { if (ROUTEMAP_NAME(red)) free(ROUTEMAP_NAME(red)); ROUTEMAP_NAME(red) = strdup(name); ROUTEMAP(red) = route_map_lookup_by_name(name); } void ospf_routemap_unset(struct ospf_redist *red) { if (ROUTEMAP_NAME(red)) free(ROUTEMAP_NAME(red)); ROUTEMAP_NAME(red) = NULL; ROUTEMAP(red) = NULL; } /* Zebra route add and delete treatment. */ static int ospf_zebra_read_route(int command, struct zclient *zclient, zebra_size_t length, vrf_id_t vrf_id) { struct zapi_route api; struct prefix_ipv4 p; unsigned long ifindex; struct in_addr nexthop; struct external_info *ei; struct ospf *ospf; int i; ospf = ospf_lookup_by_vrf_id(vrf_id); if (ospf == NULL) return 0; if (zapi_route_decode(zclient->ibuf, &api) < 0) return -1; ifindex = api.nexthops[0].ifindex; nexthop = api.nexthops[0].gate.ipv4; memcpy(&p, &api.prefix, sizeof(p)); if (IPV4_NET127(ntohl(p.prefix.s_addr))) return 0; if (IS_DEBUG_OSPF(zebra, ZEBRA_REDISTRIBUTE)) { char buf_prefix[PREFIX_STRLEN]; prefix2str(&api.prefix, buf_prefix, sizeof(buf_prefix)); zlog_debug("%s: from client %s: vrf_id %d, p %s", __func__, zebra_route_string(api.type), vrf_id, buf_prefix); } if (command == ZEBRA_REDISTRIBUTE_ROUTE_ADD) { /* XXX|HACK|TODO|FIXME: * Maybe we should ignore reject/blackhole routes? Testing * shows that there is no problems though and this is only way * to "summarize" routes in ASBR at the moment. Maybe we need * just a better generalised solution for these types? */ /* Protocol tag overwrites all other tag value sent by zebra */ if (ospf->dtag[api.type] > 0) api.tag = ospf->dtag[api.type]; /* * Given zebra sends update for a prefix via ADD message, it * should * be considered as an implicit DEL for that prefix with other * source * types. */ for (i = 0; i < ZEBRA_ROUTE_MAX; i++) if (i != api.type) ospf_external_info_delete(ospf, i, api.instance, p); ei = ospf_external_info_add(ospf, api.type, api.instance, p, ifindex, nexthop, api.tag); if (ei == NULL) { /* Nothing has changed, so nothing to do; return */ return 0; } if (ospf->router_id.s_addr == 0) /* Set flags to generate AS-external-LSA originate event for each redistributed protocols later. */ ospf->external_origin |= (1 << api.type); else { if (ei) { if (is_prefix_default(&p)) ospf_external_lsa_refresh_default(ospf); else { struct ospf_lsa *current; current = ospf_external_info_find_lsa( ospf, &ei->p); if (!current) ospf_external_lsa_originate( ospf, ei); else { if (IS_DEBUG_OSPF( zebra, ZEBRA_REDISTRIBUTE)) zlog_debug( "ospf_zebra_read_route() : %s refreshing LSA", inet_ntoa( p.prefix)); ospf_external_lsa_refresh( ospf, current, ei, LSA_REFRESH_FORCE); } } } } } else /* if (command == ZEBRA_REDISTRIBUTE_ROUTE_DEL) */ { ospf_external_info_delete(ospf, api.type, api.instance, p); if (is_prefix_default(&p)) ospf_external_lsa_refresh_default(ospf); else ospf_external_lsa_flush(ospf, api.type, &p, ifindex /*, nexthop */); } return 0; } int ospf_distribute_list_out_set(struct ospf *ospf, int type, const char *name) { /* Lookup access-list for distribute-list. */ DISTRIBUTE_LIST(ospf, type) = access_list_lookup(AFI_IP, name); /* Clear previous distribute-name. */ if (DISTRIBUTE_NAME(ospf, type)) free(DISTRIBUTE_NAME(ospf, type)); /* Set distribute-name. */ DISTRIBUTE_NAME(ospf, type) = strdup(name); /* If access-list have been set, schedule update timer. */ if (DISTRIBUTE_LIST(ospf, type)) ospf_distribute_list_update(ospf, type, 0); return CMD_SUCCESS; } int ospf_distribute_list_out_unset(struct ospf *ospf, int type, const char *name) { /* Schedule update timer. */ if (DISTRIBUTE_LIST(ospf, type)) ospf_distribute_list_update(ospf, type, 0); /* Unset distribute-list. */ DISTRIBUTE_LIST(ospf, type) = NULL; /* Clear distribute-name. */ if (DISTRIBUTE_NAME(ospf, type)) free(DISTRIBUTE_NAME(ospf, type)); DISTRIBUTE_NAME(ospf, type) = NULL; return CMD_SUCCESS; } /* distribute-list update timer. */ static int ospf_distribute_list_update_timer(struct thread *thread) { struct route_node *rn; struct external_info *ei; struct route_table *rt; struct ospf_lsa *lsa; int type, default_refresh = 0, arg_type; struct ospf *ospf = NULL; void **arg = THREAD_ARG(thread); ospf = (struct ospf *)arg[0]; arg_type = (int)(intptr_t)arg[1]; if (ospf == NULL) return 0; ospf->t_distribute_update = NULL; zlog_info("Zebra[Redistribute]: distribute-list update timer fired!"); if (IS_DEBUG_OSPF_EVENT) { zlog_debug( "%s: ospf distribute-list update arg_type %d vrf %s id %d", __PRETTY_FUNCTION__, arg_type, ospf_vrf_id_to_name(ospf->vrf_id), ospf->vrf_id); } /* foreach all external info. */ for (type = 0; type <= ZEBRA_ROUTE_MAX; type++) { struct list *ext_list; struct listnode *node; struct ospf_external *ext; ext_list = ospf->external[type]; if (!ext_list) continue; for (ALL_LIST_ELEMENTS_RO(ext_list, node, ext)) { rt = ext->external_info; if (!rt) continue; for (rn = route_top(rt); rn; rn = route_next(rn)) if ((ei = rn->info) != NULL) { if (is_prefix_default(&ei->p)) default_refresh = 1; else if ( (lsa = ospf_external_info_find_lsa( ospf, &ei->p))) ospf_external_lsa_refresh( ospf, lsa, ei, LSA_REFRESH_IF_CHANGED); else ospf_external_lsa_originate( ospf, ei); } } } if (default_refresh) ospf_external_lsa_refresh_default(ospf); XFREE(MTYPE_OSPF_DIST_ARGS, arg); return 0; } /* Update distribute-list and set timer to apply access-list. */ void ospf_distribute_list_update(struct ospf *ospf, int type, u_short instance) { struct route_table *rt; struct ospf_external *ext; void **args = XCALLOC(MTYPE_OSPF_DIST_ARGS, sizeof(void *) * 2); args[0] = ospf; args[1] = (void *)((ptrdiff_t)type); /* External info does not exist. */ ext = ospf_external_lookup(ospf, type, instance); if (!ext || !(rt = EXTERNAL_INFO(ext))) { XFREE(MTYPE_OSPF_DIST_ARGS, args); return; } /* If exists previously invoked thread, then let it continue. */ if (ospf->t_distribute_update) { XFREE(MTYPE_OSPF_DIST_ARGS, args); return; } /* Set timer. */ ospf->t_distribute_update = NULL; thread_add_timer_msec(master, ospf_distribute_list_update_timer, (void **)args, ospf->min_ls_interval, &ospf->t_distribute_update); } /* If access-list is updated, apply some check. */ static void ospf_filter_update(struct access_list *access) { struct ospf *ospf; int type; int abr_inv = 0; struct ospf_area *area; struct listnode *node, *n1; /* If OSPF instance does not exist, return right now. */ if (listcount(om->ospf) == 0) return; /* Iterate all ospf [VRF] instances */ for (ALL_LIST_ELEMENTS_RO(om->ospf, n1, ospf)) { /* Update distribute-list, and apply filter. */ for (type = 0; type <= ZEBRA_ROUTE_MAX; type++) { struct list *red_list; struct listnode *node; struct ospf_redist *red; red_list = ospf->redist[type]; if (red_list) for (ALL_LIST_ELEMENTS_RO(red_list, node, red)) { if (ROUTEMAP(red)) { /* if route-map is not NULL it * may be * using this access list */ ospf_distribute_list_update( ospf, type, red->instance); } } /* There is place for route-map for default-information * (ZEBRA_ROUTE_MAX), * but no distribute list. */ if (type == ZEBRA_ROUTE_MAX) break; if (DISTRIBUTE_NAME(ospf, type)) { /* Keep old access-list for distribute-list. */ struct access_list *old = DISTRIBUTE_LIST(ospf, type); /* Update access-list for distribute-list. */ DISTRIBUTE_LIST(ospf, type) = access_list_lookup( AFI_IP, DISTRIBUTE_NAME(ospf, type)); /* No update for this distribute type. */ if (old == NULL && DISTRIBUTE_LIST(ospf, type) == NULL) continue; /* Schedule distribute-list update timer. */ if (DISTRIBUTE_LIST(ospf, type) == NULL || strcmp(DISTRIBUTE_NAME(ospf, type), access->name) == 0) ospf_distribute_list_update(ospf, type, 0); } } /* Update Area access-list. */ for (ALL_LIST_ELEMENTS_RO(ospf->areas, node, area)) { if (EXPORT_NAME(area)) { EXPORT_LIST(area) = NULL; abr_inv++; } if (IMPORT_NAME(area)) { IMPORT_LIST(area) = NULL; abr_inv++; } } /* Schedule ABR tasks -- this will be changed -- takada. */ if (IS_OSPF_ABR(ospf) && abr_inv) ospf_schedule_abr_task(ospf); } } /* If prefix-list is updated, do some updates. */ void ospf_prefix_list_update(struct prefix_list *plist) { struct ospf *ospf = NULL; int type; int abr_inv = 0; struct ospf_area *area; struct listnode *node, *n1; /* If OSPF instatnce does not exist, return right now. */ if (listcount(om->ospf) == 0) return; /* Iterate all ospf [VRF] instances */ for (ALL_LIST_ELEMENTS_RO(om->ospf, n1, ospf)) { /* Update all route-maps which are used * as redistribution filters. * They might use prefix-list. */ for (type = 0; type <= ZEBRA_ROUTE_MAX; type++) { struct list *red_list; struct listnode *node; struct ospf_redist *red; red_list = ospf->redist[type]; if (red_list) { for (ALL_LIST_ELEMENTS_RO(red_list, node, red)) { if (ROUTEMAP(red)) { /* if route-map is not NULL * it may be using * this prefix list */ ospf_distribute_list_update( ospf, type, red->instance); } } } } /* Update area filter-lists. */ for (ALL_LIST_ELEMENTS_RO(ospf->areas, node, area)) { /* Update filter-list in. */ if (PREFIX_NAME_IN(area)) if (strcmp(PREFIX_NAME_IN(area), prefix_list_name(plist)) == 0) { PREFIX_LIST_IN(area) = prefix_list_lookup( AFI_IP, PREFIX_NAME_IN(area)); abr_inv++; } /* Update filter-list out. */ if (PREFIX_NAME_OUT(area)) if (strcmp(PREFIX_NAME_OUT(area), prefix_list_name(plist)) == 0) { PREFIX_LIST_IN(area) = prefix_list_lookup( AFI_IP, PREFIX_NAME_OUT(area)); abr_inv++; } } /* Schedule ABR task. */ if (IS_OSPF_ABR(ospf) && abr_inv) ospf_schedule_abr_task(ospf); } } static struct ospf_distance *ospf_distance_new(void) { return XCALLOC(MTYPE_OSPF_DISTANCE, sizeof(struct ospf_distance)); } static void ospf_distance_free(struct ospf_distance *odistance) { XFREE(MTYPE_OSPF_DISTANCE, odistance); } int ospf_distance_set(struct vty *vty, struct ospf *ospf, const char *distance_str, const char *ip_str, const char *access_list_str) { int ret; struct prefix_ipv4 p; u_char distance; struct route_node *rn; struct ospf_distance *odistance; ret = str2prefix_ipv4(ip_str, &p); if (ret == 0) { vty_out(vty, "Malformed prefix\n"); return CMD_WARNING_CONFIG_FAILED; } distance = atoi(distance_str); /* Get OSPF distance node. */ rn = route_node_get(ospf->distance_table, (struct prefix *)&p); if (rn->info) { odistance = rn->info; route_unlock_node(rn); } else { odistance = ospf_distance_new(); rn->info = odistance; } /* Set distance value. */ odistance->distance = distance; /* Reset access-list configuration. */ if (odistance->access_list) { free(odistance->access_list); odistance->access_list = NULL; } if (access_list_str) odistance->access_list = strdup(access_list_str); return CMD_SUCCESS; } int ospf_distance_unset(struct vty *vty, struct ospf *ospf, const char *distance_str, const char *ip_str, char const *access_list_str) { int ret; struct prefix_ipv4 p; struct route_node *rn; struct ospf_distance *odistance; ret = str2prefix_ipv4(ip_str, &p); if (ret == 0) { vty_out(vty, "Malformed prefix\n"); return CMD_WARNING_CONFIG_FAILED; } rn = route_node_lookup(ospf->distance_table, (struct prefix *)&p); if (!rn) { vty_out(vty, "Can't find specified prefix\n"); return CMD_WARNING_CONFIG_FAILED; } odistance = rn->info; if (odistance->access_list) free(odistance->access_list); ospf_distance_free(odistance); rn->info = NULL; route_unlock_node(rn); route_unlock_node(rn); return CMD_SUCCESS; } void ospf_distance_reset(struct ospf *ospf) { struct route_node *rn; struct ospf_distance *odistance; for (rn = route_top(ospf->distance_table); rn; rn = route_next(rn)) if ((odistance = rn->info) != NULL) { if (odistance->access_list) free(odistance->access_list); ospf_distance_free(odistance); rn->info = NULL; route_unlock_node(rn); } } u_char ospf_distance_apply(struct ospf *ospf, struct prefix_ipv4 *p, struct ospf_route * or) { if (ospf == NULL) return 0; if (ospf->distance_intra) if (or->path_type == OSPF_PATH_INTRA_AREA) return ospf->distance_intra; if (ospf->distance_inter) if (or->path_type == OSPF_PATH_INTER_AREA) return ospf->distance_inter; if (ospf->distance_external) if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL || or->path_type == OSPF_PATH_TYPE2_EXTERNAL) return ospf->distance_external; if (ospf->distance_all) return ospf->distance_all; return 0; } void ospf_zebra_vrf_register(struct ospf *ospf) { if (!zclient || zclient->sock < 0 || !ospf) return; if (ospf->vrf_id != VRF_UNKNOWN) { if (IS_DEBUG_OSPF_EVENT) zlog_debug("%s: Register VRF %s id %u", __PRETTY_FUNCTION__, ospf_vrf_id_to_name(ospf->vrf_id), ospf->vrf_id); zclient_send_reg_requests(zclient, ospf->vrf_id); } } void ospf_zebra_vrf_deregister(struct ospf *ospf) { if (!zclient || zclient->sock < 0 || !ospf) return; if (ospf->vrf_id != VRF_DEFAULT && ospf->vrf_id != VRF_UNKNOWN) { if (IS_DEBUG_OSPF_EVENT) zlog_debug("%s: De-Register VRF %s id %u to Zebra.", __PRETTY_FUNCTION__, ospf_vrf_id_to_name(ospf->vrf_id), ospf->vrf_id); /* Deregister for router-id, interfaces, * redistributed routes. */ zclient_send_dereg_requests(zclient, ospf->vrf_id); } } static void ospf_zebra_connected(struct zclient *zclient) { /* Send the client registration */ bfd_client_sendmsg(zclient, ZEBRA_BFD_CLIENT_REGISTER); zclient_send_reg_requests(zclient, VRF_DEFAULT); } void ospf_zebra_init(struct thread_master *master, u_short instance) { /* Allocate zebra structure. */ zclient = zclient_new_notify(master, &zclient_options_default); zclient_init(zclient, ZEBRA_ROUTE_OSPF, instance, &ospfd_privs); zclient->zebra_connected = ospf_zebra_connected; zclient->router_id_update = ospf_router_id_update_zebra; zclient->interface_add = ospf_interface_add; zclient->interface_delete = ospf_interface_delete; zclient->interface_up = ospf_interface_state_up; zclient->interface_down = ospf_interface_state_down; zclient->interface_address_add = ospf_interface_address_add; zclient->interface_address_delete = ospf_interface_address_delete; zclient->interface_link_params = ospf_interface_link_params; zclient->interface_vrf_update = ospf_interface_vrf_update; zclient->redistribute_route_add = ospf_zebra_read_route; zclient->redistribute_route_del = ospf_zebra_read_route; access_list_add_hook(ospf_filter_update); access_list_delete_hook(ospf_filter_update); prefix_list_add_hook(ospf_prefix_list_update); prefix_list_delete_hook(ospf_prefix_list_update); }