struct mfc6_cache *c, struct rtmsg *rtm);
 static void mr6_netlink_event(struct mr6_table *mrt, struct mfc6_cache *mfc,
                              int cmd);
+static void mrt6msg_netlink_event(struct mr6_table *mrt, struct sk_buff *pkt);
 static int ip6mr_rtm_dumproute(struct sk_buff *skb,
                               struct netlink_callback *cb);
 static void mroute_clean_tables(struct mr6_table *mrt, bool all);
 }
 
 /*
- *     Bounce a cache query up to pim6sd. We could use netlink for this but pim6sd
- *     expects the following bizarre scheme.
+ *     Bounce a cache query up to pim6sd and netlink.
  *
  *     Called under mrt_lock.
  */
                return -EINVAL;
        }
 
+       mrt6msg_netlink_event(mrt, skb);
+
        /*
         *      Deliver to user space multicast routing algorithms
         */
                rtnl_set_sk_err(net, RTNLGRP_IPV6_MROUTE, err);
 }
 
+static size_t mrt6msg_netlink_msgsize(size_t payloadlen)
+{
+       size_t len =
+               NLMSG_ALIGN(sizeof(struct rtgenmsg))
+               + nla_total_size(1)     /* IP6MRA_CREPORT_MSGTYPE */
+               + nla_total_size(4)     /* IP6MRA_CREPORT_MIF_ID */
+                                       /* IP6MRA_CREPORT_SRC_ADDR */
+               + nla_total_size(sizeof(struct in6_addr))
+                                       /* IP6MRA_CREPORT_DST_ADDR */
+               + nla_total_size(sizeof(struct in6_addr))
+                                       /* IP6MRA_CREPORT_PKT */
+               + nla_total_size(payloadlen)
+               ;
+
+       return len;
+}
+
+static void mrt6msg_netlink_event(struct mr6_table *mrt, struct sk_buff *pkt)
+{
+       struct net *net = read_pnet(&mrt->net);
+       struct nlmsghdr *nlh;
+       struct rtgenmsg *rtgenm;
+       struct mrt6msg *msg;
+       struct sk_buff *skb;
+       struct nlattr *nla;
+       int payloadlen;
+
+       payloadlen = pkt->len - sizeof(struct mrt6msg);
+       msg = (struct mrt6msg *)skb_transport_header(pkt);
+
+       skb = nlmsg_new(mrt6msg_netlink_msgsize(payloadlen), GFP_ATOMIC);
+       if (!skb)
+               goto errout;
+
+       nlh = nlmsg_put(skb, 0, 0, RTM_NEWCACHEREPORT,
+                       sizeof(struct rtgenmsg), 0);
+       if (!nlh)
+               goto errout;
+       rtgenm = nlmsg_data(nlh);
+       rtgenm->rtgen_family = RTNL_FAMILY_IP6MR;
+       if (nla_put_u8(skb, IP6MRA_CREPORT_MSGTYPE, msg->im6_msgtype) ||
+           nla_put_u32(skb, IP6MRA_CREPORT_MIF_ID, msg->im6_mif) ||
+           nla_put_in6_addr(skb, IP6MRA_CREPORT_SRC_ADDR,
+                            &msg->im6_src) ||
+           nla_put_in6_addr(skb, IP6MRA_CREPORT_DST_ADDR,
+                            &msg->im6_dst))
+               goto nla_put_failure;
+
+       nla = nla_reserve(skb, IP6MRA_CREPORT_PKT, payloadlen);
+       if (!nla || skb_copy_bits(pkt, sizeof(struct mrt6msg),
+                                 nla_data(nla), payloadlen))
+               goto nla_put_failure;
+
+       nlmsg_end(skb, nlh);
+
+       rtnl_notify(skb, net, 0, RTNLGRP_IPV6_MROUTE_R, NULL, GFP_ATOMIC);
+       return;
+
+nla_put_failure:
+       nlmsg_cancel(skb, nlh);
+errout:
+       kfree_skb(skb);
+       rtnl_set_sk_err(net, RTNLGRP_IPV6_MROUTE_R, -ENOBUFS);
+}
+
 static int ip6mr_rtm_dumproute(struct sk_buff *skb, struct netlink_callback *cb)
 {
        struct net *net = sock_net(skb->sk);