#include "opt_inet.h"
#include "opt_mrouting.h"
#define _PIM_VT 1
#include <sys/types.h>
#include <sys/param.h>
#include <sys/kernel.h>
#include <sys/stddef.h>
#include <sys/condvar.h>
#include <sys/eventhandler.h>
#include <sys/lock.h>
#include <sys/kthread.h>
#include <sys/ktr.h>
#include <sys/malloc.h>
#include <sys/mbuf.h>
#include <sys/module.h>
#include <sys/priv.h>
#include <sys/proc.h>
#include <sys/protosw.h>
#include <sys/signalvar.h>
#include <sys/socket.h>
#include <sys/socketvar.h>
#include <sys/sockio.h>
#include <sys/sx.h>
#include <sys/sysctl.h>
#include <sys/syslog.h>
#include <sys/systm.h>
#include <sys/taskqueue.h>
#include <sys/time.h>
#include <sys/counter.h>
#include <machine/atomic.h>
#include <net/if.h>
#include <net/if_var.h>
#include <net/if_private.h>
#include <net/if_types.h>
#include <net/netisr.h>
#include <net/route.h>
#include <net/vnet.h>
#include <netinet/in.h>
#include <netinet/igmp.h>
#include <netinet/in_systm.h>
#include <netinet/in_var.h>
#include <netinet/ip.h>
#include <netinet/ip_encap.h>
#include <netinet/ip_mroute.h>
#include <netinet/ip_var.h>
#include <netinet/ip_options.h>
#include <netinet/pim.h>
#include <netinet/pim_var.h>
#include <netinet/udp.h>
#include <machine/in_cksum.h>
#ifndef KTR_IPMF
#define KTR_IPMF KTR_INET
#endif
#define VIFI_INVALID ((vifi_t) -1)
static MALLOC_DEFINE(M_MRTABLE, "mroutetbl", "multicast forwarding cache");
static struct sx __exclusive_cache_line mrouter_teardown;
#define MRW_TEARDOWN_WLOCK() sx_xlock(&mrouter_teardown)
#define MRW_TEARDOWN_WUNLOCK() sx_xunlock(&mrouter_teardown)
#define MRW_TEARDOWN_LOCK_INIT() \
sx_init(&mrouter_teardown, "IPv4 multicast forwarding teardown")
#define MRW_TEARDOWN_LOCK_DESTROY() sx_destroy(&mrouter_teardown)
static struct rwlock mrouter_lock;
#define MRW_RLOCK() rw_rlock(&mrouter_lock)
#define MRW_WLOCK() rw_wlock(&mrouter_lock)
#define MRW_RUNLOCK() rw_runlock(&mrouter_lock)
#define MRW_WUNLOCK() rw_wunlock(&mrouter_lock)
#define MRW_UNLOCK() rw_unlock(&mrouter_lock)
#define MRW_LOCK_ASSERT() rw_assert(&mrouter_lock, RA_LOCKED)
#define MRW_WLOCK_ASSERT() rw_assert(&mrouter_lock, RA_WLOCKED)
#define MRW_LOCK_TRY_UPGRADE() rw_try_upgrade(&mrouter_lock)
#define MRW_WOWNED() rw_wowned(&mrouter_lock)
#define MRW_LOCK_INIT() \
rw_init(&mrouter_lock, "IPv4 multicast forwarding")
#define MRW_LOCK_DESTROY() rw_destroy(&mrouter_lock)
static int ip_mrouter_cnt;
static int ip_mrouter_unloading;
VNET_PCPUSTAT_DEFINE_STATIC(struct mrtstat, mrtstat);
VNET_PCPUSTAT_SYSINIT(mrtstat);
VNET_PCPUSTAT_SYSUNINIT(mrtstat);
SYSCTL_VNET_PCPUSTAT(_net_inet_ip, OID_AUTO, mrtstat, struct mrtstat,
mrtstat, "IPv4 Multicast Forwarding Statistics (struct mrtstat, "
"netinet/ip_mroute.h)");
struct mfctable {
struct socket *router;
LIST_HEAD(mfchashhdr, mfc) *mfchashtbl;
u_char *nexpire;
vifi_t numvifs;
struct vif viftable[MAXVIFS];
struct buf_ring *bw_upcalls;
struct mtx bw_upcalls_mtx;
struct ifnet *register_if;
vifi_t register_vif;
uint32_t api_config;
int pim_assert_enabled;
struct timeval pim_assert_interval;
};
VNET_DEFINE_STATIC(struct mfctable *, mfctables);
#define V_mfctables VNET(mfctables)
VNET_DEFINE_STATIC(uint32_t, nmfctables);
#define V_nmfctables VNET(nmfctables)
VNET_DEFINE_STATIC(u_long, mfchash);
#define V_mfchash VNET(mfchash)
#define MFCHASH(a, g) \
((((a).s_addr >> 20) ^ ((a).s_addr >> 10) ^ (a).s_addr ^ \
((g).s_addr >> 20) ^ ((g).s_addr >> 10) ^ (g).s_addr) & V_mfchash)
#define MFCHASHSIZE 256
static u_long mfchashsize = MFCHASHSIZE;
SYSCTL_ULONG(_net_inet_ip, OID_AUTO, mfchashsize, CTLFLAG_RDTUN,
&mfchashsize, 0, "IPv4 Multicast Forwarding Table hash size");
VNET_DEFINE_STATIC(struct taskqueue *, task_queue);
#define V_task_queue VNET(task_queue)
VNET_DEFINE_STATIC(struct task, task);
#define V_task VNET(task)
static eventhandler_tag if_detach_event_tag;
static eventhandler_tag rtnumfibs_change_tag;
VNET_DEFINE_STATIC(struct callout, expire_upcalls_ch);
#define V_expire_upcalls_ch VNET(expire_upcalls_ch)
#define EXPIRE_TIMEOUT (hz / 4)
#define UPCALL_EXPIRE 6
static MALLOC_DEFINE(M_BWMETER, "bwmeter", "multicast upcall bw meters");
VNET_DEFINE_STATIC(struct callout, bw_upcalls_ch);
#define V_bw_upcalls_ch VNET(bw_upcalls_ch)
#define BW_UPCALLS_PERIOD (hz)
VNET_PCPUSTAT_DEFINE_STATIC(struct pimstat, pimstat);
VNET_PCPUSTAT_SYSINIT(pimstat);
VNET_PCPUSTAT_SYSUNINIT(pimstat);
SYSCTL_NODE(_net_inet, IPPROTO_PIM, pim, CTLFLAG_RW | CTLFLAG_MPSAFE, 0,
"PIM");
SYSCTL_VNET_PCPUSTAT(_net_inet_pim, PIMCTL_STATS, stats, struct pimstat,
pimstat, "PIM Statistics (struct pimstat, netinet/pim_var.h)");
static u_long pim_squelch_wholepkt = 0;
SYSCTL_ULONG(_net_inet_pim, OID_AUTO, squelch_wholepkt, CTLFLAG_RWTUN,
&pim_squelch_wholepkt, 0,
"Disable IGMP_WHOLEPKT notifications if rendezvous point is unspecified");
static const struct encaptab *pim_encap_cookie;
static int pim_encapcheck(const struct mbuf *, int, int, void *);
static int pim_input(struct mbuf *, int, int, void *);
extern int in_mcast_loop;
static const struct encap_config ipv4_encap_cfg = {
.proto = IPPROTO_PIM,
.min_length = sizeof(struct ip) + PIM_MINLEN,
.exact_match = 8,
.check = pim_encapcheck,
.input = pim_input
};
struct pim_encap_pimhdr {
struct pim pim;
uint32_t flags;
};
#define PIM_ENCAP_TTL 64
static struct ip pim_encap_iphdr = {
#if BYTE_ORDER == LITTLE_ENDIAN
sizeof(struct ip) >> 2,
IPVERSION,
#else
IPVERSION,
sizeof(struct ip) >> 2,
#endif
0,
sizeof(struct ip),
0,
0,
PIM_ENCAP_TTL,
IPPROTO_PIM,
0,
};
static struct pim_encap_pimhdr pim_encap_pimhdr = {
{
PIM_MAKE_VT(PIM_VERSION, PIM_REGISTER),
0,
0,
},
0
};
static u_long X_ip_mcast_src(int, int);
static int X_ip_mforward(struct ip *, struct ifnet *, struct mbuf *,
struct ip_moptions *);
static void X_ip_mrouter_done(struct socket *);
static int X_ip_mrouter_get(struct socket *, struct sockopt *);
static int X_ip_mrouter_set(struct socket *, struct sockopt *);
static int X_legal_vif_num(int, int);
static int X_mrt_ioctl(u_long, caddr_t, int);
static int add_bw_upcall(struct mfctable *, struct bw_upcall *);
static int add_mfc(struct mfctable *, struct mfcctl2 *);
static int add_vif(struct mfctable *, int, struct vifctl *);
static void bw_meter_prepare_upcall(struct bw_meter *, struct timeval *);
static void bw_meter_geq_receive_packet(struct bw_meter *, int,
struct timeval *);
static void bw_upcalls_send(struct mfctable *);
static void bw_upcalls_send_all(void);
static int del_bw_upcall(struct mfctable *, struct bw_upcall *);
static int del_mfc(struct mfctable *, struct mfcctl2 *);
static int del_vif(struct mfctable *, vifi_t);
static int del_vif_locked(struct mfctable *, vifi_t, struct ifnet **,
struct ifnet **);
static void expire_bw_upcalls_send(void *);
static void expire_mfc(struct mfc *);
static void expire_upcalls(struct mfctable *);
static void expire_upcalls_all(void *);
static void free_bw_list(struct bw_meter *);
static int get_sg_cnt(struct mfctable *, struct sioc_sg_req *);
static int get_vif_cnt(struct mfctable *, struct sioc_vif_req *);
static void if_detached_event(void *, struct ifnet *);
static int ip_mdq(struct mfctable *, struct mbuf *, struct ifnet *,
struct mfc *, vifi_t);
static int ip_mrouter_init(struct socket *, int);
static __inline struct mfc *
mfc_find(const struct mfctable *mfct, const struct in_addr *,
const struct in_addr *);
static void phyint_send(struct ip *, struct vif *, struct mbuf *);
static struct mbuf *
pim_register_prepare(struct ip *, struct mbuf *);
static int pim_register_send(struct mfctable *, struct ip *, struct vif *,
struct mbuf *, struct mfc *);
static int pim_register_send_rp(struct mfctable *, struct ip *,
struct vif *, struct mbuf *, struct mfc *);
static int pim_register_send_upcall(struct mfctable *, struct ip *,
struct vif *, struct mbuf *, struct mfc *);
static void send_packet(struct vif *, struct mbuf *);
static int set_api_config(struct mfctable *, uint32_t *);
static int set_assert(struct mfctable *, int);
static int socket_send(struct socket *, struct mbuf *,
struct sockaddr_in *);
#define MRT_API_VERSION 0x0305
static const int mrt_api_version = MRT_API_VERSION;
static const uint32_t mrt_api_support = (MRT_MFC_FLAGS_DISABLE_WRONGVIF |
MRT_MFC_FLAGS_BORDER_VIF |
MRT_MFC_RP |
MRT_MFC_BW_UPCALL);
static struct mfc *
mfc_find(const struct mfctable *mfct, const struct in_addr *o,
const struct in_addr *g)
{
struct mfc *rt;
MRW_LOCK_ASSERT();
LIST_FOREACH(rt, &mfct->mfchashtbl[MFCHASH(*o, *g)], mfc_hash) {
if (in_hosteq(rt->mfc_origin, *o) &&
in_hosteq(rt->mfc_mcastgrp, *g) &&
buf_ring_empty(rt->mfc_stall_ring))
break;
}
return (rt);
}
static __inline struct mfc *
mfc_alloc(void)
{
struct mfc *rt;
rt = malloc(sizeof(*rt), M_MRTABLE, M_NOWAIT | M_ZERO);
if (rt == NULL)
return rt;
rt->mfc_stall_ring = buf_ring_alloc(MAX_UPQ, M_MRTABLE, M_NOWAIT, NULL);
if (rt->mfc_stall_ring == NULL) {
free(rt, M_MRTABLE);
return NULL;
}
return rt;
}
static struct mfctable *
somfctable(struct socket *so)
{
int fib;
fib = atomic_load_int(&so->so_fibnum);
KASSERT(fib >= 0 && fib < V_nmfctables,
("%s: so_fibnum %d out of range", __func__, fib));
return (&V_mfctables[fib]);
}
static int
X_ip_mrouter_set(struct socket *so, struct sockopt *sopt)
{
struct mfctable *mfct;
int error, optval;
vifi_t vifi;
struct vifctl vifc;
struct mfcctl2 mfc;
struct bw_upcall bw_upcall;
uint32_t i;
mfct = somfctable(so);
if (so != mfct->router && sopt->sopt_name != MRT_INIT)
return EPERM;
error = 0;
switch (sopt->sopt_name) {
case MRT_INIT:
error = sooptcopyin(sopt, &optval, sizeof optval, sizeof optval);
if (error)
break;
error = ip_mrouter_init(so, optval);
break;
case MRT_DONE:
ip_mrouter_done(so);
break;
case MRT_ADD_VIF:
error = sooptcopyin(sopt, &vifc, sizeof vifc, sizeof vifc);
if (error)
break;
error = add_vif(mfct, so->so_fibnum, &vifc);
break;
case MRT_DEL_VIF:
error = sooptcopyin(sopt, &vifi, sizeof vifi, sizeof vifi);
if (error)
break;
error = del_vif(mfct, vifi);
break;
case MRT_ADD_MFC:
case MRT_DEL_MFC:
if (sopt->sopt_name == MRT_ADD_MFC &&
(mfct->api_config & MRT_API_FLAGS_ALL) != 0) {
error = sooptcopyin(sopt, &mfc, sizeof(struct mfcctl2),
sizeof(struct mfcctl2));
} else {
error = sooptcopyin(sopt, &mfc, sizeof(struct mfcctl),
sizeof(struct mfcctl));
bzero((caddr_t)&mfc + sizeof(struct mfcctl),
sizeof(mfc) - sizeof(struct mfcctl));
}
if (error)
break;
if (sopt->sopt_name == MRT_ADD_MFC)
error = add_mfc(mfct, &mfc);
else
error = del_mfc(mfct, &mfc);
break;
case MRT_ASSERT:
error = sooptcopyin(sopt, &optval, sizeof optval, sizeof optval);
if (error)
break;
set_assert(mfct, optval);
break;
case MRT_API_CONFIG:
error = sooptcopyin(sopt, &i, sizeof i, sizeof i);
if (!error)
error = set_api_config(mfct, &i);
if (!error)
error = sooptcopyout(sopt, &i, sizeof i);
break;
case MRT_ADD_BW_UPCALL:
case MRT_DEL_BW_UPCALL:
error = sooptcopyin(sopt, &bw_upcall, sizeof bw_upcall,
sizeof bw_upcall);
if (error)
break;
if (sopt->sopt_name == MRT_ADD_BW_UPCALL)
error = add_bw_upcall(mfct, &bw_upcall);
else
error = del_bw_upcall(mfct, &bw_upcall);
break;
default:
error = EOPNOTSUPP;
break;
}
return error;
}
static int
X_ip_mrouter_get(struct socket *so, struct sockopt *sopt)
{
struct mfctable *mfct;
int error;
mfct = somfctable(so);
switch (sopt->sopt_name) {
case MRT_VERSION:
error = sooptcopyout(sopt, &mrt_api_version,
sizeof mrt_api_version);
break;
case MRT_ASSERT:
error = sooptcopyout(sopt, &mfct->pim_assert_enabled,
sizeof(mfct->pim_assert_enabled));
break;
case MRT_API_SUPPORT:
error = sooptcopyout(sopt, &mrt_api_support,
sizeof mrt_api_support);
break;
case MRT_API_CONFIG:
error = sooptcopyout(sopt, &mfct->api_config,
sizeof(mfct->api_config));
break;
default:
error = EOPNOTSUPP;
break;
}
return error;
}
static int
X_mrt_ioctl(u_long cmd, caddr_t data, int fibnum)
{
struct mfctable *mfct;
int error;
error = priv_check(curthread, PRIV_NETINET_MROUTE);
if (error)
return (error);
mfct = &V_mfctables[fibnum];
switch (cmd) {
case SIOCGETVIFCNT:
error = get_vif_cnt(mfct, (struct sioc_vif_req *)data);
break;
case SIOCGETSGCNT:
error = get_sg_cnt(mfct, (struct sioc_sg_req *)data);
break;
default:
error = EINVAL;
break;
}
return error;
}
static int
get_sg_cnt(struct mfctable *mfct, struct sioc_sg_req *req)
{
struct mfc *rt;
MRW_RLOCK();
rt = mfc_find(mfct, &req->src, &req->grp);
if (rt == NULL) {
MRW_RUNLOCK();
req->pktcnt = req->bytecnt = req->wrong_if = 0xffffffff;
return EADDRNOTAVAIL;
}
req->pktcnt = rt->mfc_pkt_cnt;
req->bytecnt = rt->mfc_byte_cnt;
req->wrong_if = rt->mfc_wrong_if;
MRW_RUNLOCK();
return 0;
}
static int
get_vif_cnt(struct mfctable *mfct, struct sioc_vif_req *req)
{
struct vif *vif;
vifi_t vifi;
vifi = req->vifi;
MRW_RLOCK();
if (vifi >= mfct->numvifs) {
MRW_RUNLOCK();
return EINVAL;
}
vif = &mfct->viftable[vifi];
mtx_lock(&vif->v_mtx);
req->icount = vif->v_pkt_in;
req->ocount = vif->v_pkt_out;
req->ibytes = vif->v_bytes_in;
req->obytes = vif->v_bytes_out;
mtx_unlock(&vif->v_mtx);
MRW_RUNLOCK();
return 0;
}
static int
detach_ifnet(struct mfctable *mfct, struct ifnet *ifp)
{
struct ifnet *free_ptr, *multi_leave;
int count;
count = 0;
restart:
for (vifi_t vifi = 0; vifi < mfct->numvifs; vifi++) {
if (mfct->viftable[vifi].v_ifp != ifp)
continue;
for (u_long i = 0; i < mfchashsize; i++) {
struct mfc *rt, *nrt;
LIST_FOREACH_SAFE(rt, &mfct->mfchashtbl[i], mfc_hash,
nrt) {
if (rt->mfc_parent == vifi) {
expire_mfc(rt);
}
}
}
del_vif_locked(mfct, vifi, &multi_leave, &free_ptr);
if (free_ptr != NULL)
count++;
if (multi_leave) {
MRW_WUNLOCK();
if_allmulti(multi_leave, 0);
MRW_WLOCK();
goto restart;
}
}
return (count);
}
static void
if_detached_event(void *arg __unused, struct ifnet *ifp)
{
int count;
MRW_WLOCK();
if (!V_ip_mrouting_enabled) {
MRW_WUNLOCK();
return;
}
count = 0;
for (int i = 0; i < V_nmfctables; i++)
count += detach_ifnet(&V_mfctables[i], ifp);
MRW_WUNLOCK();
for (int i = 0; i < count; i++)
if_free(ifp);
}
static void
ip_mrouter_upcall_thread(void *arg, int pending __unused)
{
CURVNET_SET((struct vnet *) arg);
MRW_WLOCK();
bw_upcalls_send_all();
MRW_WUNLOCK();
CURVNET_RESTORE();
}
static int
ip_mrouter_init(struct socket *so, int version)
{
struct mfctable *mfct;
CTR2(KTR_IPMF, "%s: so %p", __func__, so);
if (version != 1)
return ENOPROTOOPT;
mfct = somfctable(so);
MRW_TEARDOWN_WLOCK();
MRW_WLOCK();
if (ip_mrouter_unloading) {
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
return ENOPROTOOPT;
}
if (mfct->router != NULL) {
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
return EADDRINUSE;
}
mfct->mfchashtbl = hashinit_flags(mfchashsize, M_MRTABLE, &V_mfchash,
HASH_NOWAIT);
if (mfct->mfchashtbl == NULL) {
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
return (ENOMEM);
}
mtx_init(&mfct->bw_upcalls_mtx, "mroute upcall buf_ring mtx", NULL,
MTX_DEF);
mfct->bw_upcalls = buf_ring_alloc(BW_UPCALLS_MAX, M_MRTABLE, M_NOWAIT,
&mfct->bw_upcalls_mtx);
if (mfct->bw_upcalls == NULL) {
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
return (ENOMEM);
}
TASK_INIT(&V_task, 0, ip_mrouter_upcall_thread, curvnet);
taskqueue_cancel(V_task_queue, &V_task, NULL);
taskqueue_unblock(V_task_queue);
callout_reset(&V_expire_upcalls_ch, EXPIRE_TIMEOUT, expire_upcalls_all,
curvnet);
callout_reset(&V_bw_upcalls_ch, BW_UPCALLS_PERIOD, expire_bw_upcalls_send,
curvnet);
mfct->router = so;
mfct->pim_assert_interval.tv_sec = 3;
V_ip_mrouting_enabled = true;
atomic_add_int(&ip_mrouter_cnt, 1);
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
CTR1(KTR_IPMF, "%s: done", __func__);
return 0;
}
static void
X_ip_mrouter_done(struct socket *so)
{
struct mfctable *mfct;
struct ifnet **ifps;
int nifp;
u_long i;
vifi_t vifi;
struct bw_upcall *bu;
mfct = somfctable(so);
MRW_TEARDOWN_WLOCK();
if (so != mfct->router) {
MRW_TEARDOWN_WUNLOCK();
return;
}
mfct->router = NULL;
V_ip_mrouting_enabled = false;
atomic_subtract_int(&ip_mrouter_cnt, 1);
mfct->api_config = 0;
NET_EPOCH_WAIT();
taskqueue_block(V_task_queue);
while (taskqueue_cancel(V_task_queue, &V_task, NULL)) {
taskqueue_drain(V_task_queue, &V_task);
}
ifps = malloc(MAXVIFS * sizeof(*ifps), M_TEMP, M_WAITOK);
MRW_WLOCK();
taskqueue_cancel(V_task_queue, &V_task, NULL);
while ((bu = buf_ring_dequeue_mc(mfct->bw_upcalls)) != NULL) {
free(bu, M_MRTABLE);
}
buf_ring_free(mfct->bw_upcalls, M_MRTABLE);
mtx_destroy(&mfct->bw_upcalls_mtx);
KASSERT(mfct->numvifs <= MAXVIFS, ("More vifs than possible"));
for (vifi = 0, nifp = 0; vifi < mfct->numvifs; vifi++) {
struct vif *vif;
vif = &mfct->viftable[vifi];
if (!in_nullhost(vif->v_lcl_addr) &&
(vif->v_flags & (VIFF_TUNNEL | VIFF_REGISTER)) == 0) {
ifps[nifp++] = vif->v_ifp;
}
}
bzero(mfct->viftable, sizeof(*mfct->viftable) * MAXVIFS);
mfct->numvifs = 0;
mfct->pim_assert_enabled = 0;
for (i = 0; i < mfchashsize; i++) {
struct mfc *rt, *nrt;
LIST_FOREACH_SAFE(rt, &mfct->mfchashtbl[i], mfc_hash, nrt) {
expire_mfc(rt);
}
}
free(mfct->mfchashtbl, M_MRTABLE);
mfct->mfchashtbl = NULL;
bzero(mfct->nexpire, sizeof(mfct->nexpire[0]) * mfchashsize);
mfct->register_vif = VIFI_INVALID;
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
for (vifi = 0; vifi < nifp; vifi++)
if_allmulti(ifps[vifi], 0);
free(ifps, M_TEMP);
CTR1(KTR_IPMF, "%s: done", __func__);
}
static int
set_assert(struct mfctable *mfct, int i)
{
if ((i != 1) && (i != 0))
return EINVAL;
mfct->pim_assert_enabled = i;
return 0;
}
int
set_api_config(struct mfctable *mfct, uint32_t *apival)
{
u_long i;
if (mfct->numvifs > 0) {
*apival = 0;
return EPERM;
}
if (mfct->pim_assert_enabled) {
*apival = 0;
return EPERM;
}
MRW_RLOCK();
for (i = 0; i < mfchashsize; i++) {
if (LIST_FIRST(&mfct->mfchashtbl[i]) != NULL) {
MRW_RUNLOCK();
*apival = 0;
return EPERM;
}
}
MRW_RUNLOCK();
mfct->api_config = *apival & mrt_api_support;
*apival = mfct->api_config;
return 0;
}
static int
add_vif(struct mfctable *mfct, int fibnum, struct vifctl *vifcp)
{
struct vif *vifp;
struct sockaddr_in sin = {sizeof sin, AF_INET};
struct ifaddr *ifa;
struct ifnet *ifp;
int error;
if (vifcp->vifc_vifi >= MAXVIFS)
return EINVAL;
vifp = &mfct->viftable[vifcp->vifc_vifi];
if (vifcp->vifc_rate_limit != 0) {
log(LOG_ERR, "rate limiting is no longer supported\n");
return EINVAL;
}
if (in_nullhost(vifcp->vifc_lcl_addr))
return EADDRNOTAVAIL;
if (vifcp->vifc_flags & VIFF_REGISTER) {
ifp = NULL;
} else {
struct epoch_tracker et;
sin.sin_addr = vifcp->vifc_lcl_addr;
NET_EPOCH_ENTER(et);
ifa = ifa_ifwithaddr((struct sockaddr *)&sin);
if (ifa == NULL) {
NET_EPOCH_EXIT(et);
return EADDRNOTAVAIL;
}
ifp = ifa->ifa_ifp;
if (ifp->if_fib != fibnum) {
NET_EPOCH_EXIT(et);
return EADDRNOTAVAIL;
}
NET_EPOCH_EXIT(et);
}
if ((vifcp->vifc_flags & VIFF_TUNNEL) != 0) {
CTR1(KTR_IPMF, "%s: tunnels are no longer supported", __func__);
return EOPNOTSUPP;
} else if (vifcp->vifc_flags & VIFF_REGISTER) {
ifp = mfct->register_if = if_alloc(IFT_LOOP);
CTR2(KTR_IPMF, "%s: add register vif for ifp %p", __func__, ifp);
if (mfct->register_vif == VIFI_INVALID) {
if_initname(mfct->register_if, "register_vif", 0);
mfct->register_vif = vifcp->vifc_vifi;
}
} else {
if ((ifp->if_flags & IFF_MULTICAST) == 0)
return EOPNOTSUPP;
error = if_allmulti(ifp, 1);
if (error)
return error;
}
MRW_WLOCK();
if (!in_nullhost(vifp->v_lcl_addr)) {
if (ifp)
mfct->register_if = NULL;
MRW_WUNLOCK();
if (ifp)
if_free(ifp);
return EADDRINUSE;
}
vifp->v_flags = vifcp->vifc_flags;
vifp->v_threshold = vifcp->vifc_threshold;
vifp->v_lcl_addr = vifcp->vifc_lcl_addr;
vifp->v_rmt_addr = vifcp->vifc_rmt_addr;
vifp->v_ifp = ifp;
vifp->v_pkt_in = 0;
vifp->v_pkt_out = 0;
vifp->v_bytes_in = 0;
vifp->v_bytes_out = 0;
sprintf(vifp->v_mtx_name, "BM[%d] mtx", vifcp->vifc_vifi);
mtx_init(&vifp->v_mtx, vifp->v_mtx_name, NULL, MTX_DEF);
if (mfct->numvifs <= vifcp->vifc_vifi)
mfct->numvifs = vifcp->vifc_vifi + 1;
MRW_WUNLOCK();
CTR4(KTR_IPMF, "%s: add vif %d laddr 0x%08x thresh %x", __func__,
(int)vifcp->vifc_vifi, ntohl(vifcp->vifc_lcl_addr.s_addr),
(int)vifcp->vifc_threshold);
return 0;
}
static int
del_vif_locked(struct mfctable *mfct, vifi_t vifi,
struct ifnet **ifp_multi_leave, struct ifnet **ifp_free)
{
struct vif *vifp;
*ifp_free = NULL;
*ifp_multi_leave = NULL;
MRW_WLOCK_ASSERT();
if (vifi >= mfct->numvifs) {
return EINVAL;
}
vifp = &mfct->viftable[vifi];
if (in_nullhost(vifp->v_lcl_addr)) {
return EADDRNOTAVAIL;
}
if (!(vifp->v_flags & (VIFF_TUNNEL | VIFF_REGISTER)))
*ifp_multi_leave = vifp->v_ifp;
if (vifp->v_flags & VIFF_REGISTER) {
mfct->register_vif = VIFI_INVALID;
if (vifp->v_ifp) {
if (vifp->v_ifp == mfct->register_if)
mfct->register_if = NULL;
*ifp_free = vifp->v_ifp;
}
}
mtx_destroy(&vifp->v_mtx);
bzero((caddr_t)vifp, sizeof (*vifp));
CTR2(KTR_IPMF, "%s: delete vif %d", __func__, (int)vifi);
for (vifi = mfct->numvifs; vifi > 0; vifi--)
if (!in_nullhost(mfct->viftable[vifi - 1].v_lcl_addr))
break;
mfct->numvifs = vifi;
return 0;
}
static int
del_vif(struct mfctable *mfct, vifi_t vifi)
{
int cc;
struct ifnet *free_ptr, *multi_leave;
MRW_WLOCK();
cc = del_vif_locked(mfct, vifi, &multi_leave, &free_ptr);
MRW_WUNLOCK();
if (multi_leave)
if_allmulti(multi_leave, 0);
if (free_ptr) {
if_free(free_ptr);
}
return cc;
}
static void
update_mfc_params(struct mfctable *mfct, struct mfc *rt, struct mfcctl2 *mfccp)
{
int i;
rt->mfc_parent = mfccp->mfcc_parent;
for (i = 0; i < mfct->numvifs; i++) {
rt->mfc_ttls[i] = mfccp->mfcc_ttls[i];
rt->mfc_flags[i] = mfccp->mfcc_flags[i] & mfct->api_config &
MRT_MFC_FLAGS_ALL;
}
if (mfct->api_config & MRT_MFC_RP)
rt->mfc_rp = mfccp->mfcc_rp;
else
rt->mfc_rp.s_addr = INADDR_ANY;
}
static void
init_mfc_params(struct mfctable *mfct, struct mfc *rt, struct mfcctl2 *mfccp)
{
rt->mfc_origin = mfccp->mfcc_origin;
rt->mfc_mcastgrp = mfccp->mfcc_mcastgrp;
update_mfc_params(mfct, rt, mfccp);
rt->mfc_pkt_cnt = 0;
rt->mfc_byte_cnt = 0;
rt->mfc_wrong_if = 0;
timevalclear(&rt->mfc_last_assert);
}
static void
expire_mfc(struct mfc *rt)
{
struct rtdetq *rte;
MRW_WLOCK_ASSERT();
free_bw_list(rt->mfc_bw_meter_leq);
free_bw_list(rt->mfc_bw_meter_geq);
while (!buf_ring_empty(rt->mfc_stall_ring)) {
rte = buf_ring_dequeue_mc(rt->mfc_stall_ring);
if (rte) {
m_freem(rte->m);
free(rte, M_MRTABLE);
}
}
buf_ring_free(rt->mfc_stall_ring, M_MRTABLE);
LIST_REMOVE(rt, mfc_hash);
free(rt, M_MRTABLE);
}
static int
add_mfc(struct mfctable *mfct, struct mfcctl2 *mfccp)
{
struct mfc *rt;
struct rtdetq *rte;
u_long hash = 0;
u_short nstl;
struct epoch_tracker et;
MRW_WLOCK();
rt = mfc_find(mfct, &mfccp->mfcc_origin, &mfccp->mfcc_mcastgrp);
if (rt) {
CTR4(KTR_IPMF, "%s: update mfc orig 0x%08x group %lx parent %x",
__func__, ntohl(mfccp->mfcc_origin.s_addr),
(u_long)ntohl(mfccp->mfcc_mcastgrp.s_addr),
mfccp->mfcc_parent);
update_mfc_params(mfct, rt, mfccp);
MRW_WUNLOCK();
return (0);
}
nstl = 0;
hash = MFCHASH(mfccp->mfcc_origin, mfccp->mfcc_mcastgrp);
NET_EPOCH_ENTER(et);
LIST_FOREACH(rt, &mfct->mfchashtbl[hash], mfc_hash) {
if (in_hosteq(rt->mfc_origin, mfccp->mfcc_origin) &&
in_hosteq(rt->mfc_mcastgrp, mfccp->mfcc_mcastgrp) &&
!buf_ring_empty(rt->mfc_stall_ring)) {
CTR5(KTR_IPMF,
"%s: add mfc orig 0x%08x group %lx parent %x qh %p",
__func__, ntohl(mfccp->mfcc_origin.s_addr),
(u_long)ntohl(mfccp->mfcc_mcastgrp.s_addr),
mfccp->mfcc_parent,
rt->mfc_stall_ring);
if (nstl++)
CTR1(KTR_IPMF, "%s: multiple matches", __func__);
init_mfc_params(mfct, rt, mfccp);
rt->mfc_expire = 0;
mfct->nexpire[hash]--;
while (!buf_ring_empty(rt->mfc_stall_ring)) {
rte = buf_ring_dequeue_mc(rt->mfc_stall_ring);
if (rte->ifp != NULL)
ip_mdq(mfct, rte->m, rte->ifp, rt, -1);
m_freem(rte->m);
free(rte, M_MRTABLE);
}
}
}
NET_EPOCH_EXIT(et);
if (nstl == 0) {
CTR1(KTR_IPMF, "%s: adding mfc w/o upcall", __func__);
LIST_FOREACH(rt, &mfct->mfchashtbl[hash], mfc_hash) {
if (in_hosteq(rt->mfc_origin, mfccp->mfcc_origin) &&
in_hosteq(rt->mfc_mcastgrp, mfccp->mfcc_mcastgrp)) {
init_mfc_params(mfct, rt, mfccp);
if (rt->mfc_expire)
mfct->nexpire[hash]--;
rt->mfc_expire = 0;
break;
}
}
if (rt == NULL) {
rt = mfc_alloc();
if (rt == NULL) {
MRW_WUNLOCK();
return (ENOBUFS);
}
init_mfc_params(mfct, rt, mfccp);
rt->mfc_expire = 0;
rt->mfc_bw_meter_leq = NULL;
rt->mfc_bw_meter_geq = NULL;
LIST_INSERT_HEAD(&mfct->mfchashtbl[hash], rt, mfc_hash);
}
}
MRW_WUNLOCK();
return (0);
}
static int
del_mfc(struct mfctable *mfct, struct mfcctl2 *mfccp)
{
struct in_addr origin;
struct in_addr mcastgrp;
struct mfc *rt;
origin = mfccp->mfcc_origin;
mcastgrp = mfccp->mfcc_mcastgrp;
CTR3(KTR_IPMF, "%s: delete mfc orig 0x%08x group %lx", __func__,
ntohl(origin.s_addr), (u_long)ntohl(mcastgrp.s_addr));
MRW_WLOCK();
LIST_FOREACH(rt, &mfct->mfchashtbl[MFCHASH(origin, mcastgrp)],
mfc_hash) {
if (in_hosteq(rt->mfc_origin, origin) &&
in_hosteq(rt->mfc_mcastgrp, mcastgrp))
break;
}
if (rt == NULL) {
MRW_WUNLOCK();
return EADDRNOTAVAIL;
}
expire_mfc(rt);
MRW_WUNLOCK();
return (0);
}
static int
socket_send(struct socket *s, struct mbuf *mm, struct sockaddr_in *src)
{
if (s) {
SOCKBUF_LOCK(&s->so_rcv);
if (sbappendaddr_locked(&s->so_rcv, (struct sockaddr *)src, mm,
NULL) != 0) {
sorwakeup_locked(s);
return 0;
}
soroverflow_locked(s);
}
m_freem(mm);
return -1;
}
#define TUNNEL_LEN 12
static int
X_ip_mforward(struct ip *ip, struct ifnet *ifp, struct mbuf *m,
struct ip_moptions *imo)
{
struct mfc *rt;
struct mfctable *mfct;
int error;
vifi_t vifi;
struct mbuf *mb0;
struct rtdetq *rte;
u_long hash;
int hlen;
M_ASSERTMAPPED(m);
CTR3(KTR_IPMF, "ip_mforward: delete mfc orig 0x%08x group %lx ifp %p",
ntohl(ip->ip_src.s_addr), (u_long)ntohl(ip->ip_dst.s_addr), ifp);
if (ip->ip_hl < (sizeof(struct ip) + TUNNEL_LEN) >> 2 ||
((u_char *)(ip + 1))[1] != IPOPT_LSRR) {
} else {
return (1);
}
mfct = &V_mfctables[M_GETFIB(m)];
MRW_RLOCK();
if (__predict_false(mfct->router == NULL))
return (EADDRNOTAVAIL);
if (imo && ((vifi = imo->imo_multicast_vif) < mfct->numvifs)) {
if (ip->ip_ttl < MAXTTL)
ip->ip_ttl++;
error = ip_mdq(mfct, m, ifp, NULL, vifi);
MRW_RUNLOCK();
return error;
}
if (ip->ip_ttl <= 1 || IN_LOCAL_GROUP(ntohl(ip->ip_dst.s_addr))) {
MRW_RUNLOCK();
return 0;
}
mfc_find_retry:
MRTSTAT_INC(mrts_mfc_lookups);
rt = mfc_find(mfct, &ip->ip_src, &ip->ip_dst);
if (rt != NULL) {
error = ip_mdq(mfct, m, ifp, rt, -1);
MRW_UNLOCK();
return error;
}
if ((MRW_WOWNED() == 0) && (MRW_LOCK_TRY_UPGRADE() == 0)) {
MRW_RUNLOCK();
MRW_WLOCK();
goto mfc_find_retry;
}
hlen = ip->ip_hl << 2;
MRTSTAT_INC(mrts_mfc_misses);
MRTSTAT_INC(mrts_no_route);
CTR2(KTR_IPMF, "ip_mforward: no mfc for (0x%08x,%lx)",
ntohl(ip->ip_src.s_addr), (u_long)ntohl(ip->ip_dst.s_addr));
rte = malloc((sizeof *rte), M_MRTABLE, M_NOWAIT|M_ZERO);
if (rte == NULL) {
MRW_WUNLOCK();
return ENOBUFS;
}
mb0 = m_copypacket(m, M_NOWAIT);
if (mb0 && (!M_WRITABLE(mb0) || mb0->m_len < hlen))
mb0 = m_pullup(mb0, hlen);
if (mb0 == NULL) {
free(rte, M_MRTABLE);
MRW_WUNLOCK();
return ENOBUFS;
}
hash = MFCHASH(ip->ip_src, ip->ip_dst);
LIST_FOREACH(rt, &mfct->mfchashtbl[hash], mfc_hash) {
if (in_hosteq(ip->ip_src, rt->mfc_origin) &&
in_hosteq(ip->ip_dst, rt->mfc_mcastgrp) &&
!buf_ring_empty(rt->mfc_stall_ring))
break;
}
if (rt == NULL) {
int i;
struct igmpmsg *im;
struct sockaddr_in k_igmpsrc = { sizeof k_igmpsrc, AF_INET };
struct mbuf *mm;
for (vifi = 0; vifi < mfct->numvifs &&
mfct->viftable[vifi].v_ifp != ifp; vifi++)
;
if (vifi >= mfct->numvifs)
goto non_fatal;
rt = mfc_alloc();
if (rt == NULL)
goto fail;
mm = m_copym(mb0, 0, hlen, M_NOWAIT);
if (mm == NULL)
goto fail1;
im = mtod(mm, struct igmpmsg*);
im->im_msgtype = IGMPMSG_NOCACHE;
im->im_mbz = 0;
im->im_vif = vifi;
MRTSTAT_INC(mrts_upcalls);
k_igmpsrc.sin_addr = ip->ip_src;
if (socket_send(mfct->router, mm, &k_igmpsrc) < 0) {
CTR0(KTR_IPMF, "ip_mforward: socket queue full");
MRTSTAT_INC(mrts_upq_sockfull);
fail1: free(rt, M_MRTABLE);
fail: free(rte, M_MRTABLE);
m_freem(mb0);
MRW_WUNLOCK();
return ENOBUFS;
}
rt->mfc_origin.s_addr = ip->ip_src.s_addr;
rt->mfc_mcastgrp.s_addr = ip->ip_dst.s_addr;
rt->mfc_expire = UPCALL_EXPIRE;
mfct->nexpire[hash]++;
for (i = 0; i < mfct->numvifs; i++) {
rt->mfc_ttls[i] = 0;
rt->mfc_flags[i] = 0;
}
rt->mfc_parent = -1;
rt->mfc_rp.s_addr = INADDR_ANY;
rt->mfc_bw_meter_leq = NULL;
rt->mfc_bw_meter_geq = NULL;
rt->mfc_pkt_cnt = 0;
rt->mfc_byte_cnt = 0;
rt->mfc_wrong_if = 0;
timevalclear(&rt->mfc_last_assert);
buf_ring_enqueue(rt->mfc_stall_ring, rte);
LIST_INSERT_HEAD(&mfct->mfchashtbl[hash], rt, mfc_hash);
} else {
if (buf_ring_full(rt->mfc_stall_ring)) {
MRTSTAT_INC(mrts_upq_ovflw);
non_fatal: free(rte, M_MRTABLE);
m_freem(mb0);
MRW_WUNLOCK();
return (0);
}
buf_ring_enqueue(rt->mfc_stall_ring, rte);
}
rte->m = mb0;
rte->ifp = ifp;
MRW_WUNLOCK();
return 0;
}
static void
expire_upcalls(struct mfctable *mfct)
{
for (u_long i = 0; i < mfchashsize; i++) {
struct mfc *rt, *nrt;
if (mfct->nexpire[i] == 0)
continue;
LIST_FOREACH_SAFE(rt, &mfct->mfchashtbl[i], mfc_hash, nrt) {
if (buf_ring_empty(rt->mfc_stall_ring))
continue;
if (rt->mfc_expire == 0 || --rt->mfc_expire > 0)
continue;
MRTSTAT_INC(mrts_cache_cleanups);
CTR3(KTR_IPMF, "%s: expire (%lx, %lx)", __func__,
(u_long)ntohl(rt->mfc_origin.s_addr),
(u_long)ntohl(rt->mfc_mcastgrp.s_addr));
expire_mfc(rt);
}
}
}
static void
expire_upcalls_all(void *arg)
{
CURVNET_SET((struct vnet *)arg);
MRW_LOCK_ASSERT();
for (int i = 0; i < V_nmfctables; i++)
expire_upcalls(&V_mfctables[i]);
callout_reset(&V_expire_upcalls_ch, EXPIRE_TIMEOUT, expire_upcalls_all,
curvnet);
CURVNET_RESTORE();
}
static int
ip_mdq(struct mfctable *mfct, struct mbuf *m, struct ifnet *ifp, struct mfc *rt,
vifi_t xmt_vif)
{
struct ip *ip = mtod(m, struct ip *);
struct vif *vif;
vifi_t vifi;
int plen = ntohs(ip->ip_len);
M_ASSERTMAPPED(m);
MRW_LOCK_ASSERT();
NET_EPOCH_ASSERT();
if (xmt_vif < mfct->numvifs) {
if (mfct->viftable[xmt_vif].v_flags & VIFF_REGISTER)
pim_register_send(mfct, ip, &mfct->viftable[xmt_vif], m,
rt);
else
phyint_send(ip, &mfct->viftable[xmt_vif], m);
return 1;
}
vifi = rt->mfc_parent;
vif = &mfct->viftable[vifi];
if (vifi >= mfct->numvifs || vif->v_ifp != ifp) {
CTR4(KTR_IPMF, "%s: rx on wrong ifp %p (vifi %d, v_ifp %p)",
__func__, ifp, (int)vifi, vif->v_ifp);
MRTSTAT_INC(mrts_wrong_if);
++rt->mfc_wrong_if;
if (mfct->pim_assert_enabled && (vifi < mfct->numvifs) &&
vif->v_ifp != NULL) {
if (ifp == mfct->register_if)
PIMSTAT_INC(pims_rcv_registers_wrongiif);
for (vifi = 0; vifi < mfct->numvifs &&
mfct->viftable[vifi].v_ifp != ifp; vifi++)
;
if (vifi >= mfct->numvifs)
return 0;
if (rt->mfc_flags[vifi] & MRT_MFC_FLAGS_DISABLE_WRONGVIF)
return 0;
if (ratecheck(&rt->mfc_last_assert,
&mfct->pim_assert_interval)) {
struct sockaddr_in k_igmpsrc = { sizeof k_igmpsrc, AF_INET };
struct igmpmsg *im;
int hlen = ip->ip_hl << 2;
struct mbuf *mm = m_copym(m, 0, hlen, M_NOWAIT);
if (mm && (!M_WRITABLE(mm) || mm->m_len < hlen))
mm = m_pullup(mm, hlen);
if (mm == NULL)
return ENOBUFS;
im = mtod(mm, struct igmpmsg *);
im->im_msgtype = IGMPMSG_WRONGVIF;
im->im_mbz = 0;
im->im_vif = vifi;
MRTSTAT_INC(mrts_upcalls);
k_igmpsrc.sin_addr = im->im_src;
if (socket_send(mfct->router, mm,
&k_igmpsrc) < 0) {
CTR1(KTR_IPMF, "%s: socket queue full", __func__);
MRTSTAT_INC(mrts_upq_sockfull);
return ENOBUFS;
}
}
}
return 0;
}
mtx_lock(&vif->v_mtx);
if (in_hosteq(ip->ip_src, vif->v_lcl_addr)) {
vif->v_pkt_out++;
vif->v_bytes_out += plen;
} else {
vif->v_pkt_in++;
vif->v_bytes_in += plen;
}
mtx_unlock(&vif->v_mtx);
rt->mfc_pkt_cnt++;
rt->mfc_byte_cnt += plen;
for (vifi = 0; vifi < mfct->numvifs; vifi++)
if ((rt->mfc_ttls[vifi] > 0) && (ip->ip_ttl > rt->mfc_ttls[vifi])) {
vif = &mfct->viftable[vifi];
vif->v_pkt_out++;
vif->v_bytes_out += plen;
if (vif->v_flags & VIFF_REGISTER)
pim_register_send(mfct, ip, vif, m, rt);
else
phyint_send(ip, vif, m);
}
if ((rt->mfc_bw_meter_geq != NULL) || (rt->mfc_bw_meter_leq != NULL)) {
struct bw_meter *x;
struct timeval now;
microtime(&now);
for (x = rt->mfc_bw_meter_geq; x != NULL; x = x->bm_mfc_next)
bw_meter_geq_receive_packet(x, plen, &now);
for (x = rt->mfc_bw_meter_leq; x != NULL; x = x->bm_mfc_next) {
mtx_lock(&x->bm_mtx);
x->bm_measured.b_packets++;
x->bm_measured.b_bytes += plen;
mtx_unlock(&x->bm_mtx);
}
}
return 0;
}
static int
X_legal_vif_num(int fibnum, int vif)
{
struct mfctable *mfct;
int ret;
ret = 0;
if (vif < 0)
return (ret);
mfct = &V_mfctables[fibnum];
MRW_RLOCK();
if (vif < mfct->numvifs)
ret = 1;
MRW_RUNLOCK();
return (ret);
}
static u_long
X_ip_mcast_src(int fibnum, int vifi)
{
struct mfctable *mfct;
in_addr_t addr;
addr = INADDR_ANY;
if (vifi < 0)
return (addr);
mfct = &V_mfctables[fibnum];
MRW_RLOCK();
if (vifi < mfct->numvifs)
addr = mfct->viftable[vifi].v_lcl_addr.s_addr;
MRW_RUNLOCK();
return (addr);
}
static void
phyint_send(struct ip *ip, struct vif *vifp, struct mbuf *m)
{
struct mbuf *mb_copy;
int hlen = ip->ip_hl << 2;
MRW_LOCK_ASSERT();
M_ASSERTMAPPED(m);
mb_copy = m_copypacket(m, M_NOWAIT);
if (mb_copy && (!M_WRITABLE(mb_copy) || mb_copy->m_len < hlen))
mb_copy = m_pullup(mb_copy, hlen);
if (mb_copy == NULL)
return;
send_packet(vifp, mb_copy);
}
static void
send_packet(struct vif *vifp, struct mbuf *m)
{
struct ip_moptions imo;
int error __unused;
MRW_LOCK_ASSERT();
NET_EPOCH_ASSERT();
imo.imo_multicast_ifp = vifp->v_ifp;
imo.imo_multicast_ttl = mtod(m, struct ip *)->ip_ttl - 1;
imo.imo_multicast_loop = !!in_mcast_loop;
imo.imo_multicast_vif = -1;
STAILQ_INIT(&imo.imo_head);
error = ip_output(m, NULL, NULL, IP_FORWARDING, &imo, NULL);
}
static int
X_ip_rsvp_vif(struct socket *so __unused, struct sockopt *sopt __unused)
{
return (EOPNOTSUPP);
}
static void
X_ip_rsvp_force_done(struct socket *so __unused)
{
}
static int
X_rsvp_input(struct mbuf **mp, int *offp, int proto)
{
struct mbuf *m;
m = *mp;
*mp = NULL;
if (!V_rsvp_on)
m_freem(m);
return (IPPROTO_DONE);
}
#define BW_TIMEVALCMP(tvp, uvp, cmp) timevalcmp((tvp), (uvp), cmp)
#define BW_TIMEVALDECR(vvp, uvp) timevalsub((vvp), (uvp))
#define BW_TIMEVALADD(vvp, uvp) timevaladd((vvp), (uvp))
static uint32_t
compute_bw_meter_flags(struct bw_upcall *req)
{
uint32_t flags = 0;
if (req->bu_flags & BW_UPCALL_UNIT_PACKETS)
flags |= BW_METER_UNIT_PACKETS;
if (req->bu_flags & BW_UPCALL_UNIT_BYTES)
flags |= BW_METER_UNIT_BYTES;
if (req->bu_flags & BW_UPCALL_GEQ)
flags |= BW_METER_GEQ;
if (req->bu_flags & BW_UPCALL_LEQ)
flags |= BW_METER_LEQ;
return flags;
}
static void
expire_bw_meter_leq(void *arg)
{
struct bw_meter *x = arg;
struct timeval now;
CURVNET_SET((struct vnet *)x->arg);
MRW_LOCK_ASSERT();
microtime(&now);
if (((x->bm_flags & BW_METER_UNIT_PACKETS) &&
(x->bm_measured.b_packets <= x->bm_threshold.b_packets)) ||
((x->bm_flags & BW_METER_UNIT_BYTES) &&
(x->bm_measured.b_bytes <= x->bm_threshold.b_bytes))) {
bw_meter_prepare_upcall(x, &now);
}
taskqueue_enqueue(V_task_queue, &V_task);
x->bm_start_time = now;
mtx_lock(&x->bm_mtx);
x->bm_measured.b_bytes = 0;
x->bm_measured.b_packets = 0;
mtx_unlock(&x->bm_mtx);
callout_schedule(&x->bm_meter_callout, tvtohz(&x->bm_threshold.b_time));
CURVNET_RESTORE();
}
static int
add_bw_upcall(struct mfctable *mfct, struct bw_upcall *req)
{
struct mfc *mfc;
struct timeval delta = { BW_UPCALL_THRESHOLD_INTERVAL_MIN_SEC,
BW_UPCALL_THRESHOLD_INTERVAL_MIN_USEC };
struct timeval now;
struct bw_meter *x, **bwm_ptr;
uint32_t flags;
if (!(mfct->api_config & MRT_MFC_BW_UPCALL))
return EOPNOTSUPP;
if (!(req->bu_flags & (BW_UPCALL_UNIT_PACKETS | BW_UPCALL_UNIT_BYTES)))
return EINVAL;
if (!(req->bu_flags & (BW_UPCALL_GEQ | BW_UPCALL_LEQ)))
return EINVAL;
if ((req->bu_flags & (BW_UPCALL_GEQ | BW_UPCALL_LEQ)) == (BW_UPCALL_GEQ | BW_UPCALL_LEQ))
return EINVAL;
if (BW_TIMEVALCMP(&req->bu_threshold.b_time, &delta, <))
return EINVAL;
flags = compute_bw_meter_flags(req);
MRW_WLOCK();
mfc = mfc_find(mfct, &req->bu_src, &req->bu_dst);
if (mfc == NULL) {
MRW_WUNLOCK();
return EADDRNOTAVAIL;
}
if (req->bu_flags & BW_UPCALL_GEQ)
bwm_ptr = &mfc->mfc_bw_meter_geq;
else
bwm_ptr = &mfc->mfc_bw_meter_leq;
for (x = *bwm_ptr; x != NULL; x = x->bm_mfc_next) {
if ((BW_TIMEVALCMP(&x->bm_threshold.b_time,
&req->bu_threshold.b_time, ==))
&& (x->bm_threshold.b_packets
== req->bu_threshold.b_packets)
&& (x->bm_threshold.b_bytes
== req->bu_threshold.b_bytes)
&& (x->bm_flags & BW_METER_USER_FLAGS)
== flags) {
MRW_WUNLOCK();
return 0;
}
}
x = malloc(sizeof(*x), M_BWMETER, M_ZERO | M_NOWAIT);
if (x == NULL) {
MRW_WUNLOCK();
return ENOBUFS;
}
x->bm_threshold.b_time = req->bu_threshold.b_time;
microtime(&now);
x->bm_start_time = now;
x->bm_threshold.b_packets = req->bu_threshold.b_packets;
x->bm_threshold.b_bytes = req->bu_threshold.b_bytes;
x->bm_measured.b_packets = 0;
x->bm_measured.b_bytes = 0;
x->bm_flags = flags;
x->bm_time_next = NULL;
x->bm_mfc = mfc;
x->bm_mfctable = mfct;
x->arg = curvnet;
sprintf(x->bm_mtx_name, "BM mtx %p", x);
mtx_init(&x->bm_mtx, x->bm_mtx_name, NULL, MTX_DEF);
if (req->bu_flags & BW_UPCALL_LEQ) {
callout_init_rw(&x->bm_meter_callout, &mrouter_lock, CALLOUT_SHAREDLOCK);
callout_reset(&x->bm_meter_callout, tvtohz(&x->bm_threshold.b_time),
expire_bw_meter_leq, x);
}
x->bm_mfc_next = *bwm_ptr;
*bwm_ptr = x;
MRW_WUNLOCK();
return 0;
}
static void
free_bw_list(struct bw_meter *list)
{
while (list != NULL) {
struct bw_meter *x = list;
if (x->bm_flags & BW_METER_LEQ) {
callout_drain(&x->bm_meter_callout);
mtx_destroy(&x->bm_mtx);
}
list = list->bm_mfc_next;
free(x, M_BWMETER);
}
}
static int
del_bw_upcall(struct mfctable *mfct, struct bw_upcall *req)
{
struct mfc *mfc;
struct bw_meter *x, **bwm_ptr;
if (!(mfct->api_config & MRT_MFC_BW_UPCALL))
return EOPNOTSUPP;
MRW_WLOCK();
mfc = mfc_find(mfct, &req->bu_src, &req->bu_dst);
if (mfc == NULL) {
MRW_WUNLOCK();
return EADDRNOTAVAIL;
} else if (req->bu_flags & BW_UPCALL_DELETE_ALL) {
struct bw_meter *list;
list = mfc->mfc_bw_meter_leq;
mfc->mfc_bw_meter_leq = NULL;
free_bw_list(list);
list = mfc->mfc_bw_meter_geq;
mfc->mfc_bw_meter_geq = NULL;
free_bw_list(list);
MRW_WUNLOCK();
return 0;
} else {
struct bw_meter *prev;
uint32_t flags = 0;
flags = compute_bw_meter_flags(req);
if (req->bu_flags & BW_UPCALL_GEQ)
bwm_ptr = &mfc->mfc_bw_meter_geq;
else
bwm_ptr = &mfc->mfc_bw_meter_leq;
for (prev = NULL, x = *bwm_ptr; x != NULL;
prev = x, x = x->bm_mfc_next) {
if ((BW_TIMEVALCMP(&x->bm_threshold.b_time, &req->bu_threshold.b_time, ==)) &&
(x->bm_threshold.b_packets == req->bu_threshold.b_packets) &&
(x->bm_threshold.b_bytes == req->bu_threshold.b_bytes) &&
(x->bm_flags & BW_METER_USER_FLAGS) == flags)
break;
}
if (x != NULL) {
if (prev != NULL)
prev->bm_mfc_next = x->bm_mfc_next;
else
*bwm_ptr = x->bm_mfc_next;
if (req->bu_flags & BW_UPCALL_LEQ)
callout_stop(&x->bm_meter_callout);
MRW_WUNLOCK();
free(x, M_BWMETER);
return 0;
} else {
MRW_WUNLOCK();
return EINVAL;
}
}
__assert_unreachable();
}
static void
bw_meter_geq_receive_packet(struct bw_meter *x, int plen, struct timeval *nowp)
{
struct timeval delta;
MRW_LOCK_ASSERT();
delta = *nowp;
BW_TIMEVALDECR(&delta, &x->bm_start_time);
if (BW_TIMEVALCMP(&delta, &x->bm_threshold.b_time, >)) {
x->bm_start_time = *nowp;
x->bm_measured.b_packets = 0;
x->bm_measured.b_bytes = 0;
x->bm_flags &= ~BW_METER_UPCALL_DELIVERED;
}
x->bm_measured.b_packets++;
x->bm_measured.b_bytes += plen;
if (!(x->bm_flags & BW_METER_UPCALL_DELIVERED)) {
if (((x->bm_flags & BW_METER_UNIT_PACKETS) &&
(x->bm_measured.b_packets >= x->bm_threshold.b_packets)) ||
((x->bm_flags & BW_METER_UNIT_BYTES) &&
(x->bm_measured.b_bytes >= x->bm_threshold.b_bytes))) {
bw_meter_prepare_upcall(x, nowp);
x->bm_flags |= BW_METER_UPCALL_DELIVERED;
}
}
}
static void
bw_meter_prepare_upcall(struct bw_meter *x, struct timeval *nowp)
{
struct timeval delta;
struct bw_upcall *u;
MRW_LOCK_ASSERT();
delta = *nowp;
BW_TIMEVALDECR(&delta, &x->bm_start_time);
u = malloc(sizeof(struct bw_upcall), M_MRTABLE, M_NOWAIT | M_ZERO);
if (!u) {
log(LOG_WARNING, "bw_meter_prepare_upcall: cannot allocate entry\n");
return;
}
u->bu_src = x->bm_mfc->mfc_origin;
u->bu_dst = x->bm_mfc->mfc_mcastgrp;
u->bu_threshold.b_time = x->bm_threshold.b_time;
u->bu_threshold.b_packets = x->bm_threshold.b_packets;
u->bu_threshold.b_bytes = x->bm_threshold.b_bytes;
u->bu_measured.b_time = delta;
u->bu_measured.b_packets = x->bm_measured.b_packets;
u->bu_measured.b_bytes = x->bm_measured.b_bytes;
u->bu_flags = 0;
if (x->bm_flags & BW_METER_UNIT_PACKETS)
u->bu_flags |= BW_UPCALL_UNIT_PACKETS;
if (x->bm_flags & BW_METER_UNIT_BYTES)
u->bu_flags |= BW_UPCALL_UNIT_BYTES;
if (x->bm_flags & BW_METER_GEQ)
u->bu_flags |= BW_UPCALL_GEQ;
if (x->bm_flags & BW_METER_LEQ)
u->bu_flags |= BW_UPCALL_LEQ;
if (buf_ring_enqueue(x->bm_mfctable->bw_upcalls, u))
log(LOG_WARNING, "bw_meter_prepare_upcall: cannot enqueue upcall\n");
if (buf_ring_count(x->bm_mfctable->bw_upcalls) > (BW_UPCALLS_MAX / 2)) {
taskqueue_enqueue(V_task_queue, &V_task);
}
}
static void
bw_upcalls_send(struct mfctable *mfct)
{
struct mbuf *m;
int len = 0;
struct bw_upcall *bu;
struct sockaddr_in k_igmpsrc = { sizeof k_igmpsrc, AF_INET };
static struct igmpmsg igmpmsg = {
0,
0,
IGMPMSG_BW_UPCALL,
0,
0,
0,
{ 0 },
{ 0 }
};
MRW_LOCK_ASSERT();
if (buf_ring_empty(mfct->bw_upcalls))
return;
m = m_gethdr(M_NOWAIT, MT_DATA);
if (m == NULL) {
log(LOG_WARNING, "bw_upcalls_send: cannot allocate mbuf\n");
return;
}
m_copyback(m, 0, sizeof(struct igmpmsg), (caddr_t)&igmpmsg);
len += sizeof(struct igmpmsg);
while ((bu = buf_ring_dequeue_mc(mfct->bw_upcalls)) != NULL) {
m_copyback(m, len, sizeof(struct bw_upcall), (caddr_t)bu);
len += sizeof(struct bw_upcall);
free(bu, M_MRTABLE);
}
MRTSTAT_INC(mrts_upcalls);
if (socket_send(mfct->router, m, &k_igmpsrc) < 0) {
log(LOG_WARNING, "bw_upcalls_send: ip_mrouter socket queue full\n");
MRTSTAT_INC(mrts_upq_sockfull);
}
}
static void
bw_upcalls_send_all(void)
{
for (int i = 0; i < V_nmfctables; i++) {
struct mfctable *mfct;
mfct = &V_mfctables[i];
if (mfct->router != NULL)
bw_upcalls_send(mfct);
}
}
static void
expire_bw_upcalls_send(void *arg)
{
CURVNET_SET((struct vnet *) arg);
bw_upcalls_send_all();
callout_reset(&V_bw_upcalls_ch, BW_UPCALLS_PERIOD, expire_bw_upcalls_send,
curvnet);
CURVNET_RESTORE();
}
static int
pim_register_send(struct mfctable *mfct, struct ip *ip, struct vif *vifp,
struct mbuf *m, struct mfc *rt)
{
struct mbuf *mb_copy, *mm;
if (pim_squelch_wholepkt != 0 && (mfct->api_config & MRT_MFC_RP) &&
in_nullhost(rt->mfc_rp))
return 0;
mb_copy = pim_register_prepare(ip, m);
if (mb_copy == NULL)
return ENOBUFS;
for (mm = mb_copy; mm; mm = mb_copy) {
mb_copy = mm->m_nextpkt;
mm->m_nextpkt = 0;
mm = m_pullup(mm, sizeof(struct ip));
if (mm != NULL) {
ip = mtod(mm, struct ip *);
if ((mfct->api_config & MRT_MFC_RP) &&
!in_nullhost(rt->mfc_rp)) {
pim_register_send_rp(mfct, ip, vifp, mm, rt);
} else {
pim_register_send_upcall(mfct, ip, vifp, mm,
rt);
}
}
}
return 0;
}
static struct mbuf *
pim_register_prepare(struct ip *ip, struct mbuf *m)
{
struct mbuf *mb_copy = NULL;
int mtu;
if (m->m_pkthdr.csum_flags & CSUM_DELAY_DATA) {
in_delayed_cksum(m);
m->m_pkthdr.csum_flags &= ~CSUM_DELAY_DATA;
}
mb_copy = m_copypacket(m, M_NOWAIT);
if (mb_copy == NULL)
return NULL;
mb_copy = m_pullup(mb_copy, ip->ip_hl << 2);
if (mb_copy == NULL)
return NULL;
ip = mtod(mb_copy, struct ip *);
--ip->ip_ttl;
mtu = 0xffff - sizeof(pim_encap_iphdr) - sizeof(pim_encap_pimhdr);
if (ntohs(ip->ip_len) <= mtu) {
ip->ip_sum = 0;
ip->ip_sum = in_cksum(mb_copy, ip->ip_hl << 2);
} else {
mb_copy->m_pkthdr.csum_flags |= CSUM_IP;
if (ip_fragment(ip, &mb_copy, mtu, 0) != 0) {
m_freem(mb_copy);
return NULL;
}
}
return mb_copy;
}
static int
pim_register_send_upcall(struct mfctable *mfct, struct ip *ip, struct vif *vifp,
struct mbuf *mb_copy, struct mfc *rt)
{
struct mbuf *mb_first;
int len = ntohs(ip->ip_len);
struct igmpmsg *im;
struct sockaddr_in k_igmpsrc = { sizeof k_igmpsrc, AF_INET };
MRW_LOCK_ASSERT();
mb_first = m_gethdr(M_NOWAIT, MT_DATA);
if (mb_first == NULL) {
m_freem(mb_copy);
return ENOBUFS;
}
mb_first->m_data += max_linkhdr;
mb_first->m_pkthdr.len = len + sizeof(struct igmpmsg);
mb_first->m_len = sizeof(struct igmpmsg);
mb_first->m_next = mb_copy;
im = mtod(mb_first, struct igmpmsg *);
im->im_msgtype = IGMPMSG_WHOLEPKT;
im->im_mbz = 0;
im->im_vif = vifp - mfct->viftable;
im->im_src = ip->ip_src;
im->im_dst = ip->ip_dst;
k_igmpsrc.sin_addr = ip->ip_src;
MRTSTAT_INC(mrts_upcalls);
if (socket_send(mfct->router, mb_first, &k_igmpsrc) < 0) {
CTR1(KTR_IPMF, "%s: socket queue full", __func__);
MRTSTAT_INC(mrts_upq_sockfull);
return ENOBUFS;
}
PIMSTAT_INC(pims_snd_registers_msgs);
PIMSTAT_ADD(pims_snd_registers_bytes, len);
return 0;
}
static int
pim_register_send_rp(struct mfctable *mfct, struct ip *ip, struct vif *vifp,
struct mbuf *mb_copy, struct mfc *rt)
{
struct mbuf *mb_first;
struct ip *ip_outer;
struct pim_encap_pimhdr *pimhdr;
int len = ntohs(ip->ip_len);
vifi_t vifi = rt->mfc_parent;
MRW_LOCK_ASSERT();
if (vifi >= mfct->numvifs ||
in_nullhost(mfct->viftable[vifi].v_lcl_addr)) {
m_freem(mb_copy);
return EADDRNOTAVAIL;
}
mb_first = m_gethdr(M_NOWAIT, MT_DATA);
if (mb_first == NULL) {
m_freem(mb_copy);
return ENOBUFS;
}
mb_first->m_data += max_linkhdr;
mb_first->m_len = sizeof(pim_encap_iphdr) + sizeof(pim_encap_pimhdr);
mb_first->m_next = mb_copy;
mb_first->m_pkthdr.len = len + mb_first->m_len;
ip_outer = mtod(mb_first, struct ip *);
*ip_outer = pim_encap_iphdr;
ip_outer->ip_len = htons(len + sizeof(pim_encap_iphdr) +
sizeof(pim_encap_pimhdr));
ip_outer->ip_src = mfct->viftable[vifi].v_lcl_addr;
ip_outer->ip_dst = rt->mfc_rp;
ip_outer->ip_tos = ip->ip_tos;
if (ip->ip_off & htons(IP_DF))
ip_outer->ip_off |= htons(IP_DF);
ip_fillid(ip_outer, V_ip_random_id);
pimhdr = (struct pim_encap_pimhdr *)((caddr_t)ip_outer
+ sizeof(pim_encap_iphdr));
*pimhdr = pim_encap_pimhdr;
if (rt->mfc_flags[vifi] & MRT_MFC_FLAGS_BORDER_VIF & mfct->api_config)
pimhdr->flags |= htonl(PIM_BORDER_REGISTER);
mb_first->m_data += sizeof(pim_encap_iphdr);
pimhdr->pim.pim_cksum = in_cksum(mb_first, sizeof(pim_encap_pimhdr));
mb_first->m_data -= sizeof(pim_encap_iphdr);
send_packet(vifp, mb_first);
PIMSTAT_INC(pims_snd_registers_msgs);
PIMSTAT_ADD(pims_snd_registers_bytes, len);
return 0;
}
static int
pim_encapcheck(const struct mbuf *m __unused, int off __unused,
int proto __unused, void *arg __unused)
{
KASSERT(proto == IPPROTO_PIM, ("not for IPPROTO_PIM"));
return (8);
}
static int
pim_input(struct mbuf *m, int off, int proto, void *arg __unused)
{
struct mfctable *mfct;
struct ip *ip = mtod(m, struct ip *);
struct pim *pim;
int iphlen = off;
int minlen;
int datalen = ntohs(ip->ip_len) - iphlen;
int ip_tos;
mfct = &V_mfctables[M_GETFIB(m)];
PIMSTAT_INC(pims_rcv_total_msgs);
PIMSTAT_ADD(pims_rcv_total_bytes, datalen);
if (datalen < PIM_MINLEN) {
PIMSTAT_INC(pims_rcv_tooshort);
CTR3(KTR_IPMF, "%s: short packet (%d) from 0x%08x",
__func__, datalen, ntohl(ip->ip_src.s_addr));
m_freem(m);
return (IPPROTO_DONE);
}
minlen = iphlen + (datalen >= PIM_REG_MINLEN ? PIM_REG_MINLEN : PIM_MINLEN);
if (m->m_len < minlen && (m = m_pullup(m, minlen)) == NULL) {
CTR1(KTR_IPMF, "%s: m_pullup() failed", __func__);
return (IPPROTO_DONE);
}
ip = mtod(m, struct ip *);
ip_tos = ip->ip_tos;
m->m_data += iphlen;
m->m_len -= iphlen;
pim = mtod(m, struct pim *);
if (PIM_VT_T(pim->pim_vt) == PIM_REGISTER && in_cksum(m, PIM_MINLEN) == 0) {
} else if (in_cksum(m, datalen)) {
PIMSTAT_INC(pims_rcv_badsum);
CTR1(KTR_IPMF, "%s: invalid checksum", __func__);
m_freem(m);
return (IPPROTO_DONE);
}
if (PIM_VT_V(pim->pim_vt) < PIM_VERSION) {
PIMSTAT_INC(pims_rcv_badversion);
CTR3(KTR_IPMF, "%s: bad version %d expect %d", __func__,
(int)PIM_VT_V(pim->pim_vt), PIM_VERSION);
m_freem(m);
return (IPPROTO_DONE);
}
m->m_data -= iphlen;
m->m_len += iphlen;
if (PIM_VT_T(pim->pim_vt) == PIM_REGISTER) {
struct sockaddr_in dst = { sizeof(dst), AF_INET };
struct mbuf *mcp;
struct ip *encap_ip;
u_int32_t *reghdr;
struct ifnet *vifp;
MRW_RLOCK();
if (mfct->register_vif >= mfct->numvifs ||
mfct->register_vif == VIFI_INVALID) {
MRW_RUNLOCK();
CTR2(KTR_IPMF, "%s: register vif not set: %d", __func__,
(int)mfct->register_vif);
m_freem(m);
return (IPPROTO_DONE);
}
vifp = mfct->viftable[mfct->register_vif].v_ifp;
MRW_RUNLOCK();
if (datalen < PIM_REG_MINLEN) {
PIMSTAT_INC(pims_rcv_tooshort);
PIMSTAT_INC(pims_rcv_badregisters);
CTR1(KTR_IPMF, "%s: register packet size too small", __func__);
m_freem(m);
return (IPPROTO_DONE);
}
reghdr = (u_int32_t *)(pim + 1);
encap_ip = (struct ip *)(reghdr + 1);
CTR3(KTR_IPMF, "%s: register: encap ip src 0x%08x len %d",
__func__, ntohl(encap_ip->ip_src.s_addr),
ntohs(encap_ip->ip_len));
if (encap_ip->ip_v != IPVERSION) {
PIMSTAT_INC(pims_rcv_badregisters);
CTR1(KTR_IPMF, "%s: bad encap ip version", __func__);
m_freem(m);
return (IPPROTO_DONE);
}
if (!IN_MULTICAST(ntohl(encap_ip->ip_dst.s_addr))) {
PIMSTAT_INC(pims_rcv_badregisters);
CTR2(KTR_IPMF, "%s: bad encap ip dest 0x%08x", __func__,
ntohl(encap_ip->ip_dst.s_addr));
m_freem(m);
return (IPPROTO_DONE);
}
if ((ntohl(*reghdr) & PIM_NULL_REGISTER))
goto pim_input_to_daemon;
if (encap_ip->ip_tos != ip_tos) {
encap_ip->ip_tos = ip_tos;
m->m_data += (iphlen + PIM_MINLEN);
m->m_len -= (iphlen + PIM_MINLEN);
encap_ip->ip_sum = 0;
encap_ip->ip_sum = in_cksum(m, encap_ip->ip_hl << 2);
m->m_data -= (iphlen + PIM_MINLEN);
m->m_len += (iphlen + PIM_MINLEN);
}
mcp = m_copym(m, 0, iphlen + PIM_REG_MINLEN, M_NOWAIT);
if (mcp == NULL) {
CTR1(KTR_IPMF, "%s: m_copym() failed", __func__);
m_freem(m);
return (IPPROTO_DONE);
}
PIMSTAT_INC(pims_rcv_registers_msgs);
PIMSTAT_ADD(pims_rcv_registers_bytes, ntohs(encap_ip->ip_len));
m_adj(m, iphlen + PIM_MINLEN);
CTR4(KTR_IPMF,
"%s: forward decap'd REGISTER: src %lx dst %lx vif %d",
__func__,
(u_long)ntohl(encap_ip->ip_src.s_addr),
(u_long)ntohl(encap_ip->ip_dst.s_addr),
(int)mfct->register_vif);
if_simloop(vifp, m, dst.sin_family, 0);
m = mcp;
}
pim_input_to_daemon:
return (rip_input(&m, &off, proto));
}
static int
sysctl_mfctable(SYSCTL_HANDLER_ARGS)
{
struct mfctable *mfct;
struct mfc *rt;
int error, i;
if (req->newptr)
return (EPERM);
mfct = &V_mfctables[curthread->td_proc->p_fibnum];
if (mfct->mfchashtbl == NULL)
return (0);
error = sysctl_wire_old_buffer(req, 0);
if (error)
return (error);
MRW_RLOCK();
if (mfct->mfchashtbl == NULL)
goto out_locked;
for (i = 0; i < mfchashsize; i++) {
LIST_FOREACH(rt, &mfct->mfchashtbl[i], mfc_hash) {
error = SYSCTL_OUT(req, rt, sizeof(struct mfc));
if (error)
goto out_locked;
}
}
out_locked:
MRW_RUNLOCK();
return (error);
}
static SYSCTL_NODE(_net_inet_ip, OID_AUTO, mfctable,
CTLFLAG_RD | CTLFLAG_MPSAFE, sysctl_mfctable,
"IPv4 Multicast Forwarding Table "
"(struct *mfc[mfchashsize], netinet/ip_mroute.h)");
static int
sysctl_viflist(SYSCTL_HANDLER_ARGS)
{
struct mfctable *mfct;
int error, i;
if (req->newptr)
return (EPERM);
error = sysctl_wire_old_buffer(req, MROUTE_VIF_SYSCTL_LEN * MAXVIFS);
if (error)
return (error);
mfct = &V_mfctables[curthread->td_proc->p_fibnum];
MRW_RLOCK();
for (i = 0; i < MAXVIFS; i++) {
error = SYSCTL_OUT(req, &mfct->viftable[i],
MROUTE_VIF_SYSCTL_LEN);
if (error)
break;
}
MRW_RUNLOCK();
return (error);
}
SYSCTL_PROC(_net_inet_ip, OID_AUTO, viftable,
CTLTYPE_OPAQUE | CTLFLAG_VNET | CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, 0,
sysctl_viflist, "S,vif[MAXVIFS]",
"IPv4 Multicast Interfaces (struct vif[MAXVIFS], netinet/ip_mroute.h)");
static void
ip_mroute_rtnumfibs_change(void *arg __unused, uint32_t ntables)
{
struct mfctable *mfctables, *omfctables;
KASSERT(ntables >= V_nmfctables,
("%s: ntables %u nmfctables %u", __func__, ntables, V_nmfctables));
mfctables = mallocarray(ntables, sizeof(*mfctables), M_MRTABLE,
M_WAITOK | M_ZERO);
omfctables = V_mfctables;
for (int i = V_nmfctables; i < ntables; i++) {
struct mfctable *mfct;
mfct = &mfctables[i];
mfct->nexpire = malloc(mfchashsize, M_MRTABLE,
M_WAITOK | M_ZERO);
mfct->register_vif = VIFI_INVALID;
}
MRW_TEARDOWN_WLOCK();
MRW_WLOCK();
for (int i = 0; i < V_nmfctables; i++)
memcpy(&mfctables[i], &omfctables[i], sizeof(*mfctables));
atomic_store_rel_ptr((uintptr_t *)&V_mfctables, (uintptr_t)mfctables);
MRW_WUNLOCK();
MRW_TEARDOWN_WUNLOCK();
NET_EPOCH_WAIT();
V_nmfctables = ntables;
free(omfctables, M_MRTABLE);
}
static void
vnet_mroute_init(const void *unused __unused)
{
ip_mroute_rtnumfibs_change(NULL, V_rt_numfibs);
callout_init_rw(&V_expire_upcalls_ch, &mrouter_lock, 0);
callout_init_rw(&V_bw_upcalls_ch, &mrouter_lock, 0);
V_task_queue = taskqueue_create_fast("ip_mroute_tskq", M_NOWAIT,
taskqueue_thread_enqueue, &V_task_queue);
taskqueue_start_threads(&V_task_queue, 1, PI_NET, "ip_mroute_tskq task");
}
VNET_SYSINIT(vnet_mroute_init, SI_SUB_PROTO_MC, SI_ORDER_ANY, vnet_mroute_init,
NULL);
static void
vnet_mroute_uninit(const void *unused __unused)
{
taskqueue_free(V_task_queue);
for (int i = 0; i < V_rt_numfibs; i++) {
struct mfctable *mfct;
mfct = &V_mfctables[i];
free(mfct->nexpire, M_MRTABLE);
}
free(V_mfctables, M_MRTABLE);
callout_drain(&V_expire_upcalls_ch);
callout_drain(&V_bw_upcalls_ch);
}
VNET_SYSUNINIT(vnet_mroute_uninit, SI_SUB_PROTO_MC, SI_ORDER_MIDDLE,
vnet_mroute_uninit, NULL);
static int
ip_mroute_modevent(module_t mod, int type, void *unused)
{
switch (type) {
case MOD_LOAD:
MRW_TEARDOWN_LOCK_INIT();
MRW_LOCK_INIT();
if_detach_event_tag = EVENTHANDLER_REGISTER(
ifnet_departure_event, if_detached_event, NULL,
EVENTHANDLER_PRI_ANY);
rtnumfibs_change_tag = EVENTHANDLER_REGISTER(
rtnumfibs_change, ip_mroute_rtnumfibs_change,
NULL, EVENTHANDLER_PRI_ANY);
if (!powerof2(mfchashsize)) {
printf("WARNING: %s not a power of 2; using default\n",
"net.inet.ip.mfchashsize");
mfchashsize = MFCHASHSIZE;
}
pim_encap_cookie = ip_encap_attach(&ipv4_encap_cfg, NULL, M_WAITOK);
ip_mcast_src = X_ip_mcast_src;
ip_mforward = X_ip_mforward;
ip_mrouter_done = X_ip_mrouter_done;
ip_mrouter_get = X_ip_mrouter_get;
ip_mrouter_set = X_ip_mrouter_set;
ip_rsvp_force_done = X_ip_rsvp_force_done;
ip_rsvp_vif = X_ip_rsvp_vif;
legal_vif_num = X_legal_vif_num;
mrt_ioctl = X_mrt_ioctl;
rsvp_input_p = X_rsvp_input;
break;
case MOD_UNLOAD:
MRW_WLOCK();
if (ip_mrouter_cnt != 0) {
MRW_WUNLOCK();
return (EBUSY);
}
ip_mrouter_unloading = 1;
MRW_WUNLOCK();
EVENTHANDLER_DEREGISTER(rtnumfibs_change,
rtnumfibs_change_tag);
EVENTHANDLER_DEREGISTER(ifnet_departure_event,
if_detach_event_tag);
if (pim_encap_cookie) {
ip_encap_detach(pim_encap_cookie);
pim_encap_cookie = NULL;
}
ip_mcast_src = NULL;
ip_mforward = NULL;
ip_mrouter_done = NULL;
ip_mrouter_get = NULL;
ip_mrouter_set = NULL;
ip_rsvp_force_done = NULL;
ip_rsvp_vif = NULL;
legal_vif_num = NULL;
mrt_ioctl = NULL;
rsvp_input_p = NULL;
MRW_LOCK_DESTROY();
MRW_TEARDOWN_LOCK_DESTROY();
break;
default:
return EOPNOTSUPP;
}
return 0;
}
static moduledata_t ip_mroutemod = {
"ip_mroute",
ip_mroute_modevent,
0
};
DECLARE_MODULE(ip_mroute, ip_mroutemod, SI_SUB_PROTO_MC, SI_ORDER_MIDDLE);