mirror of
				https://git.proxmox.com/git/mirror_frr
				synced 2025-10-25 02:28:06 +00:00 
			
		
		
		
	 d3cc1e4518
			
		
	
	
		d3cc1e4518
		
	
	
	
	
		
			
			VRF creation can happen from either cli or from knowledged about the vrf learned from zebra. In the case where we learn about the vrf from the cli, the vrf id is UNKNOWN. Upon actual creation of the vrf, lib/vrf.c touches up the vrf_id and calls pim_vrf_enable to turn it on properly. At this point in time we have a pim->vrf_id of UNKNOWN and the vrf->vrf_id of the right value. There is no point in duplicating this data. So just remove all pim->vrf_id and use the vrf->vrf_id instead since we keep a copy of the pim->vrf pointer. This will remove some crashes where we expect the pim->vrf_id to be usable and it's not. Signed-off-by: Donald Sharp <sharpd@nvidia.com>
		
			
				
	
	
		
			863 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			863 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Multicast traceroute for FRRouting
 | |
|  * Copyright (C) 2017  Mladen Sablic
 | |
|  *
 | |
|  * This program 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 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * This program 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
 | |
|  */
 | |
| 
 | |
| /* based on draft-ietf-idmr-traceroute-ipm-07 */
 | |
| 
 | |
| #include <zebra.h>
 | |
| 
 | |
| #include "pimd.h"
 | |
| #include "pim_util.h"
 | |
| #include "pim_sock.h"
 | |
| #include "pim_rp.h"
 | |
| #include "pim_oil.h"
 | |
| #include "pim_ifchannel.h"
 | |
| #include "pim_macro.h"
 | |
| #include "pim_igmp_mtrace.h"
 | |
| 
 | |
| static struct in_addr mtrace_primary_address(struct interface *ifp)
 | |
| {
 | |
| 	struct connected *ifc;
 | |
| 	struct listnode *node;
 | |
| 	struct in_addr any;
 | |
| 	struct pim_interface *pim_ifp;
 | |
| 
 | |
| 	if (ifp->info) {
 | |
| 		pim_ifp = ifp->info;
 | |
| 		return pim_ifp->primary_address;
 | |
| 	}
 | |
| 
 | |
| 	any.s_addr = INADDR_ANY;
 | |
| 
 | |
| 	for (ALL_LIST_ELEMENTS_RO(ifp->connected, node, ifc)) {
 | |
| 		struct prefix *p = ifc->address;
 | |
| 
 | |
| 		if (p->family != AF_INET)
 | |
| 			continue;
 | |
| 
 | |
| 		if (!CHECK_FLAG(ifc->flags, ZEBRA_IFA_SECONDARY))
 | |
| 			return p->u.prefix4;
 | |
| 		/* in case no primary found, return a secondary */
 | |
| 		any = p->u.prefix4;
 | |
| 	}
 | |
| 	return any;
 | |
| }
 | |
| 
 | |
| static bool mtrace_fwd_info_weak(struct pim_instance *pim,
 | |
| 				    struct igmp_mtrace *mtracep,
 | |
| 				    struct igmp_mtrace_rsp *rspp,
 | |
| 				    struct interface **ifpp)
 | |
| {
 | |
| 	struct pim_nexthop nexthop;
 | |
| 	struct interface *ifp_in;
 | |
| 	struct in_addr nh_addr;
 | |
| 
 | |
| 	nh_addr.s_addr = INADDR_ANY;
 | |
| 
 | |
| 	memset(&nexthop, 0, sizeof(nexthop));
 | |
| 
 | |
| 	if (!pim_nexthop_lookup(pim, &nexthop, mtracep->src_addr, 1)) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("mtrace not found neighbor");
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE)
 | |
| 		zlog_debug("mtrace pim_nexthop_lookup OK");
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE)
 | |
| 		zlog_debug("mtrace next_hop=%pI4",
 | |
| 			   &nexthop.mrib_nexthop_addr.u.prefix4);
 | |
| 
 | |
| 	if (nexthop.mrib_nexthop_addr.family == AF_INET)
 | |
| 		nh_addr = nexthop.mrib_nexthop_addr.u.prefix4;
 | |
| 
 | |
| 	ifp_in = nexthop.interface;
 | |
| 
 | |
| 	/* return interface for forwarding mtrace packets */
 | |
| 	*ifpp = ifp_in;
 | |
| 
 | |
| 	/* 6.2.2. 4. Fill in the Incoming Interface Address... */
 | |
| 	rspp->incoming = mtrace_primary_address(ifp_in);
 | |
| 	rspp->prev_hop = nh_addr;
 | |
| 	rspp->in_count = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	rspp->total = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	rspp->rtg_proto = MTRACE_RTG_PROTO_PIM;
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| static bool mtrace_fwd_info(struct pim_instance *pim,
 | |
| 			    struct igmp_mtrace *mtracep,
 | |
| 			    struct igmp_mtrace_rsp *rspp,
 | |
| 			    struct interface **ifpp)
 | |
| {
 | |
| 	struct prefix_sg sg;
 | |
| 	struct pim_upstream *up;
 | |
| 	struct interface *ifp_in;
 | |
| 	struct in_addr nh_addr;
 | |
| 	uint32_t total;
 | |
| 
 | |
| 	memset(&sg, 0, sizeof(struct prefix_sg));
 | |
| 	sg.src = mtracep->src_addr;
 | |
| 	sg.grp = mtracep->grp_addr;
 | |
| 
 | |
| 	up = pim_upstream_find(pim, &sg);
 | |
| 
 | |
| 	if (!up) {
 | |
| 		sg.src.s_addr = INADDR_ANY;
 | |
| 		up = pim_upstream_find(pim, &sg);
 | |
| 	}
 | |
| 
 | |
| 	if (!up)
 | |
| 		return false;
 | |
| 
 | |
| 	if (!up->rpf.source_nexthop.interface) {
 | |
| 		if (PIM_DEBUG_TRACE)
 | |
| 			zlog_debug("%s: up %s RPF is not present", __func__,
 | |
| 				   up->sg_str);
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	ifp_in = up->rpf.source_nexthop.interface;
 | |
| 	nh_addr = up->rpf.source_nexthop.mrib_nexthop_addr.u.prefix4;
 | |
| 	total = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE)
 | |
| 		zlog_debug("fwd_info: upstream next hop=%pI4", &nh_addr);
 | |
| 
 | |
| 	if (up->channel_oil)
 | |
| 		total = up->channel_oil->cc.pktcnt;
 | |
| 
 | |
| 	/* return interface for forwarding mtrace packets */
 | |
| 	*ifpp = ifp_in;
 | |
| 
 | |
| 	/* 6.2.2. 4. Fill in the Incoming Interface Address... */
 | |
| 	rspp->incoming = mtrace_primary_address(ifp_in);
 | |
| 	rspp->prev_hop = nh_addr;
 | |
| 	rspp->in_count = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	rspp->total = total;
 | |
| 	rspp->rtg_proto = MTRACE_RTG_PROTO_PIM;
 | |
| 
 | |
| 	/* 6.2.2. 4. Fill in ... S, and Src Mask */
 | |
| 	if (sg.src.s_addr != INADDR_ANY) {
 | |
| 		rspp->s = 1;
 | |
| 		rspp->src_mask = MTRACE_SRC_MASK_SOURCE;
 | |
| 	} else {
 | |
| 		rspp->s = 0;
 | |
| 		rspp->src_mask = MTRACE_SRC_MASK_GROUP;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| static void mtrace_rsp_set_fwd_code(struct igmp_mtrace_rsp *mtrace_rspp,
 | |
| 				    enum mtrace_fwd_code fwd_code)
 | |
| {
 | |
| 	if (mtrace_rspp->fwd_code == MTRACE_FWD_CODE_NO_ERROR)
 | |
| 		mtrace_rspp->fwd_code = fwd_code;
 | |
| }
 | |
| 
 | |
| static void mtrace_rsp_init(struct igmp_mtrace_rsp *mtrace_rspp)
 | |
| {
 | |
| 	mtrace_rspp->arrival = 0;
 | |
| 	mtrace_rspp->incoming.s_addr = INADDR_ANY;
 | |
| 	mtrace_rspp->outgoing.s_addr = INADDR_ANY;
 | |
| 	mtrace_rspp->prev_hop.s_addr = INADDR_ANY;
 | |
| 	mtrace_rspp->in_count = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	mtrace_rspp->out_count = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	mtrace_rspp->total = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	mtrace_rspp->rtg_proto = 0;
 | |
| 	mtrace_rspp->fwd_ttl = 0;
 | |
| 	mtrace_rspp->mbz = 0;
 | |
| 	mtrace_rspp->s = 0;
 | |
| 	mtrace_rspp->src_mask = 0;
 | |
| 	mtrace_rspp->fwd_code = MTRACE_FWD_CODE_NO_ERROR;
 | |
| }
 | |
| 
 | |
| static void mtrace_rsp_debug(uint32_t qry_id, int rsp,
 | |
| 			     struct igmp_mtrace_rsp *mrspp)
 | |
| {
 | |
| 	struct in_addr incoming = mrspp->incoming;
 | |
| 	struct in_addr outgoing = mrspp->outgoing;
 | |
| 	struct in_addr prev_hop = mrspp->prev_hop;
 | |
| 
 | |
| 	zlog_debug(
 | |
| 		"Rx mt(%d) qid=%ud arr=%x in=%pI4 out=%pI4 prev=%pI4 proto=%d fwd=%d",
 | |
| 		rsp, ntohl(qry_id), mrspp->arrival, &incoming, &outgoing,
 | |
| 		&prev_hop, mrspp->rtg_proto, mrspp->fwd_code);
 | |
| }
 | |
| 
 | |
| static void mtrace_debug(struct pim_interface *pim_ifp,
 | |
| 			 struct igmp_mtrace *mtracep, int mtrace_len)
 | |
| {
 | |
| 	struct in_addr ga, sa, da, ra;
 | |
| 
 | |
| 	ga = mtracep->grp_addr;
 | |
| 	sa = mtracep->src_addr;
 | |
| 	da = mtracep->dst_addr;
 | |
| 	ra = mtracep->rsp_addr;
 | |
| 
 | |
| 	zlog_debug(
 | |
| 		"Rx mtrace packet incoming on %pI4: hops=%d type=%d size=%d, grp=%pI4, src=%pI4, dst=%pI4 rsp=%pI4 ttl=%d qid=%ud",
 | |
| 		&pim_ifp->primary_address, mtracep->hops, mtracep->type,
 | |
| 		mtrace_len, &ga, &sa, &da, &ra, mtracep->rsp_ttl,
 | |
| 		ntohl(mtracep->qry_id));
 | |
| 	if (mtrace_len > (int)sizeof(struct igmp_mtrace)) {
 | |
| 
 | |
| 		int i;
 | |
| 
 | |
| 		int responses = mtrace_len - sizeof(struct igmp_mtrace);
 | |
| 
 | |
| 		if ((responses % sizeof(struct igmp_mtrace_rsp)) != 0)
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Mtrace response block of wrong length");
 | |
| 
 | |
| 		responses = responses / sizeof(struct igmp_mtrace_rsp);
 | |
| 
 | |
| 		for (i = 0; i < responses; i++)
 | |
| 			mtrace_rsp_debug(mtracep->qry_id, i, &mtracep->rsp[i]);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* 5.1 Query Arrival Time */
 | |
| static uint32_t query_arrival_time(void)
 | |
| {
 | |
| 	struct timeval tv;
 | |
| 	uint32_t qat;
 | |
| 
 | |
| 	if (gettimeofday(&tv, NULL) < 0) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("Query arrival time lookup failed: errno=%d: %s",
 | |
| 				   errno, safe_strerror(errno));
 | |
| 		return 0;
 | |
| 	}
 | |
| 	/* not sure second offset correct, as I get different value */
 | |
| 	qat = ((tv.tv_sec + 32384) << 16) + ((tv.tv_usec << 10) / 15625);
 | |
| 
 | |
| 	return qat;
 | |
| }
 | |
| 
 | |
| static int mtrace_send_packet(struct interface *ifp,
 | |
| 			      struct igmp_mtrace *mtracep,
 | |
| 			      size_t mtrace_buf_len, struct in_addr dst_addr,
 | |
| 			      struct in_addr group_addr)
 | |
| {
 | |
| 	struct sockaddr_in to;
 | |
| 	socklen_t tolen;
 | |
| 	ssize_t sent;
 | |
| 	int ret;
 | |
| 	int fd;
 | |
| 	uint8_t ttl;
 | |
| 
 | |
| 	memset(&to, 0, sizeof(to));
 | |
| 	to.sin_family = AF_INET;
 | |
| 	to.sin_addr = dst_addr;
 | |
| 	tolen = sizeof(to);
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE) {
 | |
| 		struct in_addr if_addr;
 | |
| 		struct in_addr rsp_addr = mtracep->rsp_addr;
 | |
| 
 | |
| 		if_addr = mtrace_primary_address(ifp);
 | |
| 		zlog_debug("Sending mtrace packet to %pI4 on %pI4", &rsp_addr,
 | |
| 			   &if_addr);
 | |
| 	}
 | |
| 
 | |
| 	fd = pim_socket_raw(IPPROTO_IGMP);
 | |
| 
 | |
| 	if (fd < 0)
 | |
| 		return -1;
 | |
| 
 | |
| 	ret = pim_socket_bind(fd, ifp);
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		ret = -1;
 | |
| 		goto close_fd;
 | |
| 	}
 | |
| 
 | |
| 	if (IPV4_CLASS_DE(ntohl(dst_addr.s_addr))) {
 | |
| 		if (IPV4_MC_LINKLOCAL(ntohl(dst_addr.s_addr))) {
 | |
| 			ttl = 1;
 | |
| 		} else {
 | |
| 			if (mtracep->type == PIM_IGMP_MTRACE_RESPONSE)
 | |
| 				ttl = mtracep->rsp_ttl;
 | |
| 			else
 | |
| 				ttl = 64;
 | |
| 		}
 | |
| 		ret = setsockopt(fd, IPPROTO_IP, IP_MULTICAST_TTL, &ttl,
 | |
| 				 sizeof(ttl));
 | |
| 
 | |
| 		if (ret < 0) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug("Failed to set socket multicast TTL");
 | |
| 			ret = -1;
 | |
| 			goto close_fd;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	sent = sendto(fd, (char *)mtracep, mtrace_buf_len, MSG_DONTWAIT,
 | |
| 		      (struct sockaddr *)&to, tolen);
 | |
| 
 | |
| 	if (sent != (ssize_t)mtrace_buf_len) {
 | |
| 		char dst_str[INET_ADDRSTRLEN];
 | |
| 		char group_str[INET_ADDRSTRLEN];
 | |
| 
 | |
| 		pim_inet4_dump("<dst?>", dst_addr, dst_str, sizeof(dst_str));
 | |
| 		pim_inet4_dump("<group?>", group_addr, group_str,
 | |
| 			       sizeof(group_str));
 | |
| 		if (sent < 0) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Send mtrace request failed for %s on%s: group=%s msg_size=%zd: errno=%d:  %s",
 | |
| 					dst_str, ifp->name, group_str,
 | |
| 					mtrace_buf_len, errno,
 | |
| 					safe_strerror(errno));
 | |
| 		} else {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Send mtrace request failed for %s on %s: group=%s msg_size=%zd: sent=%zd",
 | |
| 					dst_str, ifp->name, group_str,
 | |
| 					mtrace_buf_len, sent);
 | |
| 		}
 | |
| 		ret = -1;
 | |
| 		goto close_fd;
 | |
| 	}
 | |
| 	ret = 0;
 | |
| close_fd:
 | |
| 	close(fd);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int mtrace_un_forward_packet(struct pim_instance *pim, struct ip *ip_hdr,
 | |
| 				    struct interface *interface)
 | |
| {
 | |
| 	struct pim_nexthop nexthop;
 | |
| 	struct sockaddr_in to;
 | |
| 	struct interface *if_out;
 | |
| 	socklen_t tolen;
 | |
| 	int ret;
 | |
| 	int fd;
 | |
| 	int sent;
 | |
| 	uint16_t checksum;
 | |
| 
 | |
| 	checksum = ip_hdr->ip_sum;
 | |
| 
 | |
| 	ip_hdr->ip_sum = 0;
 | |
| 
 | |
| 	if (checksum != in_cksum(ip_hdr, ip_hdr->ip_hl * 4))
 | |
| 		return -1;
 | |
| 
 | |
| 	if (ip_hdr->ip_ttl-- <= 1)
 | |
| 		return -1;
 | |
| 
 | |
| 	ip_hdr->ip_sum = in_cksum(ip_hdr, ip_hdr->ip_hl * 4);
 | |
| 
 | |
| 	fd = pim_socket_raw(IPPROTO_RAW);
 | |
| 
 | |
| 	if (fd < 0)
 | |
| 		return -1;
 | |
| 
 | |
| 	pim_socket_ip_hdr(fd);
 | |
| 
 | |
| 	if (interface == NULL) {
 | |
| 		memset(&nexthop, 0, sizeof(nexthop));
 | |
| 		if (!pim_nexthop_lookup(pim, &nexthop, ip_hdr->ip_dst, 0)) {
 | |
| 			close(fd);
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Dropping mtrace packet, no route to destination");
 | |
| 			return -1;
 | |
| 		}
 | |
| 
 | |
| 		if_out = nexthop.interface;
 | |
| 	} else {
 | |
| 		if_out = interface;
 | |
| 	}
 | |
| 
 | |
| 	ret = pim_socket_bind(fd, if_out);
 | |
| 
 | |
| 	if (ret < 0) {
 | |
| 		close(fd);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	memset(&to, 0, sizeof(to));
 | |
| 	to.sin_family = AF_INET;
 | |
| 	to.sin_addr = ip_hdr->ip_dst;
 | |
| 	tolen = sizeof(to);
 | |
| 
 | |
| 	sent = sendto(fd, ip_hdr, ntohs(ip_hdr->ip_len), 0,
 | |
| 		      (struct sockaddr *)&to, tolen);
 | |
| 
 | |
| 	close(fd);
 | |
| 
 | |
| 	if (sent < 0) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Failed to forward mtrace packet: sendto errno=%d, %s",
 | |
| 				errno, safe_strerror(errno));
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE) {
 | |
| 		zlog_debug("Fwd mtrace packet len=%u to %pI4 ttl=%u",
 | |
| 			   ntohs(ip_hdr->ip_len), &ip_hdr->ip_dst,
 | |
| 			   ip_hdr->ip_ttl);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int mtrace_mc_forward_packet(struct pim_instance *pim, struct ip *ip_hdr)
 | |
| {
 | |
| 	struct prefix_sg sg;
 | |
| 	struct channel_oil *c_oil;
 | |
| 	struct listnode *chnode;
 | |
| 	struct listnode *chnextnode;
 | |
| 	struct pim_ifchannel *ch = NULL;
 | |
| 	int ret = -1;
 | |
| 
 | |
| 	memset(&sg, 0, sizeof(struct prefix_sg));
 | |
| 	sg.grp = ip_hdr->ip_dst;
 | |
| 
 | |
| 	c_oil = pim_find_channel_oil(pim, &sg);
 | |
| 
 | |
| 	if (c_oil == NULL) {
 | |
| 		if (PIM_DEBUG_MTRACE) {
 | |
| 			zlog_debug(
 | |
| 				"Dropping mtrace multicast packet len=%u to %pI4 ttl=%u",
 | |
| 				ntohs(ip_hdr->ip_len),
 | |
| 				&ip_hdr->ip_dst, ip_hdr->ip_ttl);
 | |
| 		}
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (c_oil->up == NULL)
 | |
| 		return -1;
 | |
| 	if (c_oil->up->ifchannels == NULL)
 | |
| 		return -1;
 | |
| 	for (ALL_LIST_ELEMENTS(c_oil->up->ifchannels, chnode, chnextnode, ch)) {
 | |
| 		if (pim_macro_chisin_oiflist(ch)) {
 | |
| 			int r;
 | |
| 
 | |
| 			r = mtrace_un_forward_packet(pim, ip_hdr,
 | |
| 						     ch->interface);
 | |
| 			if (r == 0)
 | |
| 				ret = 0;
 | |
| 		}
 | |
| 	}
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int mtrace_forward_packet(struct pim_instance *pim, struct ip *ip_hdr)
 | |
| {
 | |
| 	if (IPV4_CLASS_DE(ntohl(ip_hdr->ip_dst.s_addr)))
 | |
| 		return mtrace_mc_forward_packet(pim, ip_hdr);
 | |
| 	else
 | |
| 		return mtrace_un_forward_packet(pim, ip_hdr, NULL);
 | |
| }
 | |
| 
 | |
| static int mtrace_send_mc_response(struct pim_instance *pim,
 | |
| 				   struct igmp_mtrace *mtracep,
 | |
| 				   size_t mtrace_len)
 | |
| {
 | |
| 	struct prefix_sg sg;
 | |
| 	struct channel_oil *c_oil;
 | |
| 	struct listnode *chnode;
 | |
| 	struct listnode *chnextnode;
 | |
| 	struct pim_ifchannel *ch = NULL;
 | |
| 	int ret = -1;
 | |
| 
 | |
| 	memset(&sg, 0, sizeof(struct prefix_sg));
 | |
| 	sg.grp = mtracep->rsp_addr;
 | |
| 
 | |
| 	c_oil = pim_find_channel_oil(pim, &sg);
 | |
| 
 | |
| 	if (c_oil == NULL) {
 | |
| 		if (PIM_DEBUG_MTRACE) {
 | |
| 			struct in_addr rsp_addr = mtracep->rsp_addr;
 | |
| 
 | |
| 			zlog_debug(
 | |
| 				"Dropping mtrace multicast response packet len=%u to %pI4",
 | |
| 				(unsigned int)mtrace_len, &rsp_addr);
 | |
| 		}
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (c_oil->up == NULL)
 | |
| 		return -1;
 | |
| 	if (c_oil->up->ifchannels == NULL)
 | |
| 		return -1;
 | |
| 	for (ALL_LIST_ELEMENTS(c_oil->up->ifchannels, chnode, chnextnode, ch)) {
 | |
| 		if (pim_macro_chisin_oiflist(ch)) {
 | |
| 			int r;
 | |
| 
 | |
| 			r = mtrace_send_packet(ch->interface, mtracep,
 | |
| 					       mtrace_len, mtracep->rsp_addr,
 | |
| 					       mtracep->grp_addr);
 | |
| 			if (r == 0)
 | |
| 				ret = 0;
 | |
| 		}
 | |
| 	}
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /* 6.5 Sending Traceroute Responses */
 | |
| static int mtrace_send_response(struct pim_instance *pim,
 | |
| 				struct igmp_mtrace *mtracep, size_t mtrace_len)
 | |
| {
 | |
| 	struct pim_nexthop nexthop;
 | |
| 
 | |
| 	mtracep->type = PIM_IGMP_MTRACE_RESPONSE;
 | |
| 
 | |
| 	mtracep->checksum = 0;
 | |
| 	mtracep->checksum = in_cksum((char *)mtracep, mtrace_len);
 | |
| 
 | |
| 	if (IPV4_CLASS_DE(ntohl(mtracep->rsp_addr.s_addr))) {
 | |
| 		struct pim_rpf *p_rpf;
 | |
| 
 | |
| 		if (pim_rp_i_am_rp(pim, mtracep->rsp_addr))
 | |
| 			return mtrace_send_mc_response(pim, mtracep,
 | |
| 						       mtrace_len);
 | |
| 
 | |
| 		p_rpf = pim_rp_g(pim, mtracep->rsp_addr);
 | |
| 
 | |
| 		if (p_rpf == NULL) {
 | |
| 			if (PIM_DEBUG_MTRACE) {
 | |
| 				struct in_addr rsp_addr = mtracep->rsp_addr;
 | |
| 
 | |
| 				zlog_debug("mtrace no RP for %pI4", &rsp_addr);
 | |
| 			}
 | |
| 			return -1;
 | |
| 		}
 | |
| 		nexthop = p_rpf->source_nexthop;
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("mtrace response to RP");
 | |
| 	} else {
 | |
| 		memset(&nexthop, 0, sizeof(nexthop));
 | |
| 		/* TODO: should use unicast rib lookup */
 | |
| 		if (!pim_nexthop_lookup(pim, &nexthop, mtracep->rsp_addr, 1)) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Dropped response qid=%ud, no route to response address",
 | |
| 					mtracep->qry_id);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return mtrace_send_packet(nexthop.interface, mtracep, mtrace_len,
 | |
| 				  mtracep->rsp_addr, mtracep->grp_addr);
 | |
| }
 | |
| 
 | |
| int igmp_mtrace_recv_qry_req(struct igmp_sock *igmp, struct ip *ip_hdr,
 | |
| 			     struct in_addr from, const char *from_str,
 | |
| 			     char *igmp_msg, int igmp_msg_len)
 | |
| {
 | |
| 	static uint32_t qry_id, qry_src;
 | |
| 	char mtrace_buf[MTRACE_HDR_SIZE + MTRACE_MAX_HOPS * MTRACE_RSP_SIZE];
 | |
| 	struct interface *ifp;
 | |
| 	struct interface *out_ifp = NULL;
 | |
| 	struct pim_interface *pim_ifp;
 | |
| 	struct pim_instance *pim;
 | |
| 	struct igmp_mtrace *mtracep;
 | |
| 	struct igmp_mtrace_rsp *rspp;
 | |
| 	struct in_addr nh_addr;
 | |
| 	enum mtrace_fwd_code fwd_code = MTRACE_FWD_CODE_NO_ERROR;
 | |
| 	size_t r_len;
 | |
| 	int last_rsp_ind = 0;
 | |
| 	size_t mtrace_len;
 | |
| 	uint16_t recv_checksum;
 | |
| 	uint16_t checksum;
 | |
| 	bool reached_source;
 | |
| 	bool fwd_info;
 | |
| 
 | |
| 	ifp = igmp->interface;
 | |
| 	pim_ifp = ifp->info;
 | |
| 	pim = pim_ifp->pim;
 | |
| 
 | |
| 	/*
 | |
| 	 * 6. Router Behaviour
 | |
| 	 * Check if mtrace packet is addressed elsewhere and forward,
 | |
| 	 * if applicable
 | |
| 	 */
 | |
| 	if (!IPV4_CLASS_DE(ntohl(ip_hdr->ip_dst.s_addr)))
 | |
| 		if (!if_lookup_exact_address(&ip_hdr->ip_dst, AF_INET,
 | |
| 					     pim->vrf->vrf_id))
 | |
| 			return mtrace_forward_packet(pim, ip_hdr);
 | |
| 
 | |
| 	if (igmp_msg_len < (int)sizeof(struct igmp_mtrace)) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace packet from %s on %s: too short, len=%d, min=%zu",
 | |
| 				from_str, ifp->name, igmp_msg_len,
 | |
| 				sizeof(struct igmp_mtrace));
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	mtracep = (struct igmp_mtrace *)igmp_msg;
 | |
| 
 | |
| 	recv_checksum = mtracep->checksum;
 | |
| 
 | |
| 	mtracep->checksum = 0;
 | |
| 
 | |
| 	checksum = in_cksum(igmp_msg, igmp_msg_len);
 | |
| 
 | |
| 	if (recv_checksum != checksum) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace packet from %s on %s: checksum mismatch: received=%x computed=%x",
 | |
| 				from_str, ifp->name, recv_checksum, checksum);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* Collecting IGMP Rx stats */
 | |
| 	igmp->rx_stats.mtrace_req++;
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE)
 | |
| 		mtrace_debug(pim_ifp, mtracep, igmp_msg_len);
 | |
| 
 | |
| 	/* subtract header from message length */
 | |
| 	r_len = igmp_msg_len - sizeof(struct igmp_mtrace);
 | |
| 
 | |
| 	/* Classify mtrace packet, check if it is a query */
 | |
| 	if (!r_len) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("Received IGMP multicast traceroute query");
 | |
| 
 | |
| 		/* 6.1.1  Packet verification */
 | |
| 		if (!pim_if_connected_to_source(ifp, mtracep->dst_addr)) {
 | |
| 			if (IPV4_CLASS_DE(ntohl(ip_hdr->ip_dst.s_addr))) {
 | |
| 				if (PIM_DEBUG_MTRACE)
 | |
| 					zlog_debug(
 | |
| 						"Dropping multicast query on wrong interface");
 | |
| 				return -1;
 | |
| 			}
 | |
| 			/* Unicast query on wrong interface */
 | |
| 			fwd_code = MTRACE_FWD_CODE_WRONG_IF;
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug("Multicast query on wrong interface");
 | |
| 		}
 | |
| 		if (qry_id == mtracep->qry_id && qry_src == from.s_addr) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug(
 | |
| 					"Dropping multicast query with duplicate source and id");
 | |
| 			return -1;
 | |
| 		}
 | |
| 		qry_id = mtracep->qry_id;
 | |
| 		qry_src = from.s_addr;
 | |
| 	}
 | |
| 	/* if response fields length is equal to a whole number of responses */
 | |
| 	else if ((r_len % sizeof(struct igmp_mtrace_rsp)) == 0) {
 | |
| 		r_len = igmp_msg_len - sizeof(struct igmp_mtrace);
 | |
| 
 | |
| 		if (r_len != 0)
 | |
| 			last_rsp_ind = r_len / sizeof(struct igmp_mtrace_rsp);
 | |
| 		if (last_rsp_ind > MTRACE_MAX_HOPS) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug("Mtrace request of excessive size");
 | |
| 			return -1;
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace packet from %s on %s: invalid length %d",
 | |
| 				from_str, ifp->name, igmp_msg_len);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* 6.2.1 Packet Verification - drop not link-local multicast */
 | |
| 	if (IPV4_CLASS_DE(ntohl(ip_hdr->ip_dst.s_addr))
 | |
| 	    && !IPV4_MC_LINKLOCAL(ntohl(ip_hdr->ip_dst.s_addr))) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace packet from %s on %s: not link-local multicast %pI4",
 | |
| 				from_str, ifp->name, &ip_hdr->ip_dst);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	/* 6.2.2. Normal Processing */
 | |
| 
 | |
| 	/* 6.2.2. 1. If there is room in the current buffer? */
 | |
| 
 | |
| 	if (last_rsp_ind == MTRACE_MAX_HOPS) {
 | |
| 		/* ...there was no room... */
 | |
| 		mtracep->rsp[MTRACE_MAX_HOPS - 1].fwd_code =
 | |
| 			MTRACE_FWD_CODE_NO_SPACE;
 | |
| 		return mtrace_send_response(pim_ifp->pim, mtracep,
 | |
| 					    igmp_msg_len);
 | |
| 	}
 | |
| 
 | |
| 	/* ...insert new response block... */
 | |
| 
 | |
| 	/* calculate new mtrace lenght with extra response */
 | |
| 	mtrace_len = igmp_msg_len + sizeof(struct igmp_mtrace_rsp);
 | |
| 
 | |
| 	/* copy received query/request */
 | |
| 	memcpy(mtrace_buf, igmp_msg, igmp_msg_len);
 | |
| 
 | |
| 	/* repoint mtracep pointer to copy */
 | |
| 	mtracep = (struct igmp_mtrace *)mtrace_buf;
 | |
| 
 | |
| 	/* pointer for extra response field to be filled in */
 | |
| 	rspp = &mtracep->rsp[last_rsp_ind];
 | |
| 
 | |
| 	/* initialize extra response field */
 | |
| 	mtrace_rsp_init(rspp);
 | |
| 
 | |
| 	/* carry over any error noted when receiving the query */
 | |
| 	rspp->fwd_code = fwd_code;
 | |
| 
 | |
| 	/* ...and fill in Query Arrival Time... */
 | |
| 	rspp->arrival = htonl(query_arrival_time());
 | |
| 	rspp->outgoing = pim_ifp->primary_address;
 | |
| 	rspp->out_count = htonl(MTRACE_UNKNOWN_COUNT);
 | |
| 	rspp->fwd_ttl = 1;
 | |
| 
 | |
| 	/* 6.2.2. 2. Attempt to determine the forwarding information... */
 | |
| 
 | |
| 	if (mtracep->grp_addr.s_addr != INADDR_ANY)
 | |
| 		fwd_info = mtrace_fwd_info(pim, mtracep, rspp, &out_ifp);
 | |
| 	else
 | |
| 		fwd_info = mtrace_fwd_info_weak(pim, mtracep, rspp, &out_ifp);
 | |
| 
 | |
| 	/* 6.2.2 3. If no forwarding information... */
 | |
| 	if (!fwd_info) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("mtrace not found multicast state");
 | |
| 		mtrace_rsp_set_fwd_code(rspp, MTRACE_FWD_CODE_NO_ROUTE);
 | |
| 		/* 6.2.2. 3. forward the packet to requester */
 | |
| 		return mtrace_send_response(pim, mtracep, mtrace_len);
 | |
| 	}
 | |
| 
 | |
| 	nh_addr = rspp->prev_hop;
 | |
| 
 | |
| 	reached_source = false;
 | |
| 
 | |
| 	if (nh_addr.s_addr == INADDR_ANY) {
 | |
| 		/* no pim? i.e. 7.5.3. No Previous Hop */
 | |
| 		if (!out_ifp->info) {
 | |
| 			if (PIM_DEBUG_MTRACE)
 | |
| 				zlog_debug("mtrace not found incoming if w/ pim");
 | |
| 			mtrace_rsp_set_fwd_code(rspp,
 | |
| 						MTRACE_FWD_CODE_NO_MULTICAST);
 | |
| 			return mtrace_send_response(pim, mtracep, mtrace_len);
 | |
| 		}
 | |
| 		/* reached source? i.e. 7.5.1 Arriving at source */
 | |
| 		if (pim_if_connected_to_source(out_ifp, mtracep->src_addr)) {
 | |
| 			reached_source = true;
 | |
| 			rspp->prev_hop = mtracep->src_addr;
 | |
| 		}
 | |
| 		/*
 | |
| 		 * 6.4 Forwarding Traceroute Requests:
 | |
| 		 * Previous-hop router not known,
 | |
| 		 * packet is sent to an appropriate multicast address
 | |
| 		 */
 | |
| 		(void)inet_aton(MCAST_ALL_ROUTERS, &nh_addr);
 | |
| 	}
 | |
| 
 | |
| 	/* 6.2.2 8. If this router is the Rendez-vous Point */
 | |
| 	if (pim_rp_i_am_rp(pim, mtracep->grp_addr)) {
 | |
| 		mtrace_rsp_set_fwd_code(rspp, MTRACE_FWD_CODE_REACHED_RP);
 | |
| 		/* 7.7.1. PIM-SM ...RP has not performed source-specific join */
 | |
| 		if (rspp->src_mask == MTRACE_SRC_MASK_GROUP)
 | |
| 			return mtrace_send_response(pim, mtracep, mtrace_len);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * 6.4 Forwarding Traceroute Requests: the number of response
 | |
| 	 * blocks exceeds number of responses, so forward to the requester.
 | |
| 	 */
 | |
| 	if (mtracep->hops <= (last_rsp_ind + 1))
 | |
| 		return mtrace_send_response(pim, mtracep, mtrace_len);
 | |
| 
 | |
| 	/* 7.5.1. Arriving at source: terminate trace */
 | |
| 	if (reached_source)
 | |
| 		return mtrace_send_response(pim, mtracep, mtrace_len);
 | |
| 
 | |
| 	mtracep->checksum = 0;
 | |
| 
 | |
| 	mtracep->checksum = in_cksum(mtrace_buf, mtrace_len);
 | |
| 
 | |
| 	/* 6.4 Forwarding Traceroute Requests: response blocks less than req. */
 | |
| 	return mtrace_send_packet(out_ifp, mtracep, mtrace_len, nh_addr,
 | |
| 				  mtracep->grp_addr);
 | |
| }
 | |
| 
 | |
| /* 6.3. Traceroute responses */
 | |
| int igmp_mtrace_recv_response(struct igmp_sock *igmp, struct ip *ip_hdr,
 | |
| 			      struct in_addr from, const char *from_str,
 | |
| 			      char *igmp_msg, int igmp_msg_len)
 | |
| {
 | |
| 	static uint32_t qry_id, rsp_dst;
 | |
| 	struct interface *ifp;
 | |
| 	struct pim_interface *pim_ifp;
 | |
| 	struct pim_instance *pim;
 | |
| 	struct igmp_mtrace *mtracep;
 | |
| 	uint16_t recv_checksum;
 | |
| 	uint16_t checksum;
 | |
| 
 | |
| 	ifp = igmp->interface;
 | |
| 	pim_ifp = ifp->info;
 | |
| 	pim = pim_ifp->pim;
 | |
| 
 | |
| 	if (igmp_msg_len < (int)sizeof(struct igmp_mtrace)) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace packet from %s on %s: too short, len=%d, min=%zu",
 | |
| 				from_str, ifp->name, igmp_msg_len,
 | |
| 				sizeof(struct igmp_mtrace));
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	mtracep = (struct igmp_mtrace *)igmp_msg;
 | |
| 
 | |
| 	recv_checksum = mtracep->checksum;
 | |
| 
 | |
| 	mtracep->checksum = 0;
 | |
| 
 | |
| 	checksum = in_cksum(igmp_msg, igmp_msg_len);
 | |
| 
 | |
| 	if (recv_checksum != checksum) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug(
 | |
| 				"Recv mtrace response from %s on %s: checksum mismatch: received=%x computed=%x",
 | |
| 				from_str, ifp->name, recv_checksum, checksum);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	mtracep->checksum = checksum;
 | |
| 
 | |
| 	/* Collecting IGMP Rx stats */
 | |
| 	igmp->rx_stats.mtrace_rsp++;
 | |
| 
 | |
| 	if (PIM_DEBUG_MTRACE)
 | |
| 		mtrace_debug(pim_ifp, mtracep, igmp_msg_len);
 | |
| 
 | |
| 	/* Drop duplicate packets */
 | |
| 	if (qry_id == mtracep->qry_id && rsp_dst == ip_hdr->ip_dst.s_addr) {
 | |
| 		if (PIM_DEBUG_MTRACE)
 | |
| 			zlog_debug("duplicate mtrace response packet dropped");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	qry_id = mtracep->qry_id;
 | |
| 	rsp_dst = ip_hdr->ip_dst.s_addr;
 | |
| 
 | |
| 	return mtrace_forward_packet(pim, ip_hdr);
 | |
| }
 |