git: 9742519b2229 - main - netlink: fix operations with link-local routes/gateways.

From: Alexander V. Chernikov <melifaro_at_FreeBSD.org>
Date: Mon, 17 Apr 2023 12:05:25 UTC
The branch main has been updated by melifaro:

URL: https://cgit.FreeBSD.org/src/commit/?id=9742519b2229c1035f5c81f478e558c951ddd1ea

commit 9742519b2229c1035f5c81f478e558c951ddd1ea
Author:     Alexander V. Chernikov <melifaro@FreeBSD.org>
AuthorDate: 2023-04-17 12:04:43 +0000
Commit:     Alexander V. Chernikov <melifaro@FreeBSD.org>
CommitDate: 2023-04-17 12:04:43 +0000

    netlink: fix operations with link-local routes/gateways.
    
    MFC after:      3 days
---
 sys/netlink/route/rt.c | 30 ++++++++++++++++++++++++++++++
 1 file changed, 30 insertions(+)

diff --git a/sys/netlink/route/rt.c b/sys/netlink/route/rt.c
index db535cb676e4..cbe9ee001dfd 100644
--- a/sys/netlink/route/rt.c
+++ b/sys/netlink/route/rt.c
@@ -406,6 +406,25 @@ const static struct nlfield_parser nlf_p_rtnh[] = {
 #undef _OUT
 NL_DECLARE_PARSER(mpath_parser, struct rtnexthop, nlf_p_rtnh, nla_p_rtnh);
 
+static void
+set_scope6(struct sockaddr *sa, struct ifnet *ifp)
+{
+#ifdef INET6
+	if (sa != NULL && sa->sa_family == AF_INET6 && ifp != NULL) {
+		struct sockaddr_in6 *sa6 = (struct sockaddr_in6 *)sa;
+
+		if (IN6_IS_ADDR_LINKLOCAL(&sa6->sin6_addr))
+			in6_set_unicast_scopeid(&sa6->sin6_addr, ifp->if_index);
+	}
+#endif
+}
+
+static void
+post_p_mpath(struct rta_mpath_nh *mpnh)
+{
+	set_scope6(mpnh->gw, mpnh->ifp);
+}
+
 struct rta_mpath {
 	int num_nhops;
 	struct rta_mpath_nh nhops[0];
@@ -432,6 +451,7 @@ nlattr_get_multipath(struct nlattr *nla, struct nl_pstate *npt, const void *arg,
 			    mp->num_nhops - 1);
 			return (error);
 		}
+		post_p_mpath(mpnh);
 
 		int len = NL_ITEM_ALIGN(rtnh->rtnh_len);
 		data_len -= len;
@@ -495,6 +515,13 @@ static const struct nlfield_parser nlf_p_rtmsg[] = {
 #undef _OUT
 NL_DECLARE_PARSER(rtm_parser, struct rtmsg, nlf_p_rtmsg, nla_p_rtmsg);
 
+static void
+post_p_rtmsg(struct nl_parsed_route *r)
+{
+	set_scope6(r->rta_dst, r->rta_oif);
+	set_scope6(r->rta_gw, r->rta_oif);
+}
+
 struct netlink_walkargs {
 	struct nl_writer *nw;
 	struct route_nhop_data rnd;
@@ -899,6 +926,7 @@ rtnl_handle_newroute(struct nlmsghdr *hdr, struct nlpcb *nlp,
 	error = nl_parse_nlmsg(hdr, &rtm_parser, npt, &attrs);
 	if (error != 0)
 		return (error);
+	post_p_rtmsg(&attrs);
 
 	/* Check if we have enough data */
 	if (attrs.rta_dst == NULL) {
@@ -963,6 +991,7 @@ rtnl_handle_delroute(struct nlmsghdr *hdr, struct nlpcb *nlp,
 	error = nl_parse_nlmsg(hdr, &rtm_parser, npt, &attrs);
 	if (error != 0)
 		return (error);
+	post_p_rtmsg(&attrs);
 
 	if (attrs.rta_dst == NULL) {
 		NLMSG_REPORT_ERR_MSG(npt, "RTA_DST is not set");
@@ -990,6 +1019,7 @@ rtnl_handle_getroute(struct nlmsghdr *hdr, struct nlpcb *nlp, struct nl_pstate *
 	error = nl_parse_nlmsg(hdr, &rtm_parser, npt, &attrs);
 	if (error != 0)
 		return (error);
+	post_p_rtmsg(&attrs);
 
 	if (attrs.rta_table >= V_rt_numfibs) {
 		NLMSG_REPORT_ERR_MSG(npt, "invalid fib");