Home | History | Annotate | Line # | Download | only in libnpftest
npf_test_subr.c revision 1.13
      1  1.13     rmind /*	$NetBSD: npf_test_subr.c,v 1.13 2018/09/29 14:41:36 rmind Exp $	*/
      2   1.1     rmind 
      3   1.1     rmind /*
      4   1.1     rmind  * NPF initialisation and handler routines.
      5   1.1     rmind  *
      6   1.1     rmind  * Public Domain.
      7   1.1     rmind  */
      8   1.1     rmind 
      9  1.12  christos #ifdef _KERNEL
     10   1.1     rmind #include <sys/types.h>
     11   1.7     rmind #include <sys/cprng.h>
     12  1.13     rmind #include <sys/kmem.h>
     13   1.1     rmind #include <net/if.h>
     14   1.1     rmind #include <net/if_types.h>
     15  1.12  christos #endif
     16   1.1     rmind 
     17   1.1     rmind #include "npf_impl.h"
     18   1.1     rmind #include "npf_test.h"
     19   1.1     rmind 
     20   1.1     rmind /* State of the current stream. */
     21   1.1     rmind static npf_state_t	cstream_state;
     22   1.1     rmind static void *		cstream_ptr;
     23   1.1     rmind static bool		cstream_retval;
     24   1.1     rmind 
     25   1.9     rmind static long		(*_random_func)(void);
     26   1.9     rmind static int		(*_pton_func)(int, const char *, void *);
     27   1.9     rmind static const char *	(*_ntop_func)(int, const void *, char *, socklen_t);
     28   1.9     rmind 
     29   1.4     rmind static void		npf_state_sample(npf_state_t *, bool);
     30   1.4     rmind 
     31  1.13     rmind static void		load_npf_config_ifs(nvlist_t *, bool);
     32  1.13     rmind 
     33  1.12  christos #ifndef __NetBSD__
     34  1.12  christos /*
     35  1.12  christos  * Standalone NPF: we define the same struct ifnet members
     36  1.12  christos  * to reduce the npf_ifops_t implementation differences.
     37  1.12  christos  */
     38  1.12  christos struct ifnet {
     39  1.12  christos 	char		if_xname[32];
     40  1.12  christos 	void *		if_softc;
     41  1.12  christos 	TAILQ_ENTRY(ifnet) if_list;
     42  1.12  christos };
     43  1.12  christos #endif
     44  1.12  christos 
     45  1.12  christos static TAILQ_HEAD(, ifnet) npftest_ifnet_list =
     46  1.12  christos     TAILQ_HEAD_INITIALIZER(npftest_ifnet_list);
     47  1.12  christos 
     48  1.12  christos static const char *	npftest_ifop_getname(ifnet_t *);
     49  1.12  christos static void		npftest_ifop_flush(void *);
     50  1.12  christos static void *		npftest_ifop_getmeta(const ifnet_t *);
     51  1.12  christos static void		npftest_ifop_setmeta(ifnet_t *, void *);
     52  1.12  christos 
     53  1.12  christos static const npf_ifops_t npftest_ifops = {
     54  1.12  christos 	.getname	= npftest_ifop_getname,
     55  1.12  christos 	.lookup		= npf_test_getif,
     56  1.12  christos 	.flush		= npftest_ifop_flush,
     57  1.12  christos 	.getmeta	= npftest_ifop_getmeta,
     58  1.12  christos 	.setmeta	= npftest_ifop_setmeta,
     59  1.12  christos };
     60  1.12  christos 
     61   1.4     rmind void
     62   1.9     rmind npf_test_init(int (*pton_func)(int, const char *, void *),
     63   1.9     rmind     const char *(*ntop_func)(int, const void *, char *, socklen_t),
     64   1.9     rmind     long (*rndfunc)(void))
     65   1.4     rmind {
     66  1.12  christos 	npf_t *npf;
     67  1.12  christos 
     68  1.12  christos 	npf_sysinit(1);
     69  1.12  christos 	npf = npf_create(0, &npftest_mbufops, &npftest_ifops);
     70  1.12  christos 	npf_thread_register(npf);
     71  1.12  christos 	npf_setkernctx(npf);
     72  1.12  christos 
     73   1.4     rmind 	npf_state_setsampler(npf_state_sample);
     74   1.9     rmind 	_pton_func = pton_func;
     75   1.9     rmind 	_ntop_func = ntop_func;
     76   1.9     rmind 	_random_func = rndfunc;
     77   1.4     rmind }
     78   1.4     rmind 
     79  1.12  christos void
     80  1.12  christos npf_test_fini(void)
     81  1.12  christos {
     82  1.12  christos 	npf_t *npf = npf_getkernctx();
     83  1.12  christos 	npf_destroy(npf);
     84  1.12  christos 	npf_sysfini();
     85  1.12  christos }
     86  1.12  christos 
     87   1.1     rmind int
     88  1.13     rmind npf_test_load(const void *buf, size_t len, bool verbose)
     89   1.1     rmind {
     90  1.13     rmind 	nvlist_t *npf_dict;
     91  1.13     rmind 	npf_error_t error;
     92  1.13     rmind 
     93  1.13     rmind 	npf_dict = nvlist_unpack(buf, len, 0);
     94  1.13     rmind 	if (!npf_dict) {
     95  1.13     rmind 		printf("%s: could not unpack the nvlist\n", __func__);
     96  1.13     rmind 		return EINVAL;
     97  1.13     rmind 	}
     98  1.13     rmind 	load_npf_config_ifs(npf_dict, verbose);
     99  1.13     rmind 
    100  1.13     rmind 	// Note: npf_dict will be consumed by npf_load().
    101  1.13     rmind 	return npf_load(npf_getkernctx(), npf_dict, &error);
    102   1.1     rmind }
    103   1.1     rmind 
    104   1.6     rmind ifnet_t *
    105   1.6     rmind npf_test_addif(const char *ifname, bool reg, bool verbose)
    106   1.3     rmind {
    107  1.12  christos 	npf_t *npf = npf_getkernctx();
    108  1.13     rmind 	ifnet_t *ifp = kmem_zalloc(sizeof(*ifp), KM_SLEEP);
    109   1.3     rmind 
    110   1.3     rmind 	/*
    111   1.3     rmind 	 * This is a "fake" interface with explicitly set index.
    112   1.6     rmind 	 * Note: test modules may not setup pfil(9) hooks and if_attach()
    113   1.6     rmind 	 * may not trigger npf_ifmap_attach(), so we call it manually.
    114   1.3     rmind 	 */
    115   1.3     rmind 	strlcpy(ifp->if_xname, ifname, sizeof(ifp->if_xname));
    116  1.12  christos 	TAILQ_INSERT_TAIL(&npftest_ifnet_list, ifp, if_list);
    117   1.6     rmind 
    118  1.12  christos 	npf_ifmap_attach(npf, ifp);
    119   1.6     rmind 	if (reg) {
    120  1.12  christos 		npf_ifmap_register(npf, ifname);
    121   1.6     rmind 	}
    122   1.6     rmind 
    123   1.6     rmind 	if (verbose) {
    124   1.6     rmind 		printf("+ Interface %s\n", ifname);
    125   1.6     rmind 	}
    126   1.6     rmind 	return ifp;
    127   1.3     rmind }
    128   1.3     rmind 
    129  1.13     rmind static void
    130  1.13     rmind load_npf_config_ifs(nvlist_t *npf_dict, bool verbose)
    131  1.13     rmind {
    132  1.13     rmind 	const nvlist_t * const *iflist;
    133  1.13     rmind 	const nvlist_t *dbg_dict;
    134  1.13     rmind 	size_t nitems;
    135  1.13     rmind 
    136  1.13     rmind 	dbg_dict = dnvlist_get_nvlist(npf_dict, "debug", NULL);
    137  1.13     rmind 	if (!dbg_dict) {
    138  1.13     rmind 		return;
    139  1.13     rmind 	}
    140  1.13     rmind 	if (!nvlist_exists_nvlist_array(dbg_dict, "interfaces")) {
    141  1.13     rmind 		return;
    142  1.13     rmind 	}
    143  1.13     rmind 	iflist = nvlist_get_nvlist_array(dbg_dict, "interfaces", &nitems);
    144  1.13     rmind 	for (unsigned i = 0; i < nitems; i++) {
    145  1.13     rmind 		const nvlist_t *ifdict = iflist[i];
    146  1.13     rmind 		const char *ifname;
    147  1.13     rmind 
    148  1.13     rmind 		if ((ifname = nvlist_get_string(ifdict, "name")) != NULL) {
    149  1.13     rmind 			(void)npf_test_addif(ifname, true, verbose);
    150  1.13     rmind 		}
    151  1.13     rmind 	}
    152  1.13     rmind }
    153  1.13     rmind 
    154  1.12  christos static const char *
    155  1.12  christos npftest_ifop_getname(ifnet_t *ifp)
    156  1.12  christos {
    157  1.12  christos 	return ifp->if_xname;
    158  1.12  christos }
    159  1.12  christos 
    160   1.6     rmind ifnet_t *
    161   1.3     rmind npf_test_getif(const char *ifname)
    162   1.3     rmind {
    163  1.12  christos 	ifnet_t *ifp;
    164  1.12  christos 
    165  1.12  christos 	TAILQ_FOREACH(ifp, &npftest_ifnet_list, if_list) {
    166  1.12  christos 		if (!strcmp(ifp->if_xname, ifname))
    167  1.12  christos 			return ifp;
    168  1.12  christos 	}
    169  1.12  christos 	return NULL;
    170  1.12  christos }
    171  1.12  christos 
    172  1.12  christos static void
    173  1.12  christos npftest_ifop_flush(void *arg)
    174  1.12  christos {
    175  1.12  christos 	ifnet_t *ifp;
    176  1.12  christos 
    177  1.12  christos 	TAILQ_FOREACH(ifp, &npftest_ifnet_list, if_list)
    178  1.12  christos 		ifp->if_softc = arg;
    179  1.12  christos }
    180  1.12  christos 
    181  1.12  christos static void *
    182  1.12  christos npftest_ifop_getmeta(const ifnet_t *ifp)
    183  1.12  christos {
    184  1.12  christos 	return ifp->if_softc;
    185  1.12  christos }
    186  1.12  christos 
    187  1.12  christos static void
    188  1.12  christos npftest_ifop_setmeta(ifnet_t *ifp, void *arg)
    189  1.12  christos {
    190  1.12  christos 	ifp->if_softc = arg;
    191   1.3     rmind }
    192   1.3     rmind 
    193   1.1     rmind /*
    194   1.1     rmind  * State sampler - this routine is called from inside of NPF state engine.
    195   1.1     rmind  */
    196   1.4     rmind static void
    197   1.1     rmind npf_state_sample(npf_state_t *nst, bool retval)
    198   1.1     rmind {
    199   1.1     rmind 	/* Pointer will serve as an ID. */
    200   1.1     rmind 	cstream_ptr = nst;
    201   1.1     rmind 	memcpy(&cstream_state, nst, sizeof(npf_state_t));
    202   1.1     rmind 	cstream_retval = retval;
    203   1.1     rmind }
    204   1.1     rmind 
    205   1.1     rmind int
    206   1.6     rmind npf_test_statetrack(const void *data, size_t len, ifnet_t *ifp,
    207   1.1     rmind     bool forw, int64_t *result)
    208   1.1     rmind {
    209  1.12  christos 	npf_t *npf = npf_getkernctx();
    210   1.1     rmind 	struct mbuf *m;
    211   1.1     rmind 	int i = 0, error;
    212   1.1     rmind 
    213   1.1     rmind 	m = mbuf_getwithdata(data, len);
    214  1.12  christos 	error = npf_packet_handler(npf, &m, ifp, forw ? PFIL_OUT : PFIL_IN);
    215   1.1     rmind 	if (error) {
    216   1.1     rmind 		assert(m == NULL);
    217   1.1     rmind 		return error;
    218   1.1     rmind 	}
    219   1.1     rmind 	assert(m != NULL);
    220   1.1     rmind 	m_freem(m);
    221   1.1     rmind 
    222   1.1     rmind 	const int di = forw ? NPF_FLOW_FORW : NPF_FLOW_BACK;
    223   1.1     rmind 	npf_tcpstate_t *fstate = &cstream_state.nst_tcpst[di];
    224   1.1     rmind 	npf_tcpstate_t *tstate = &cstream_state.nst_tcpst[!di];
    225   1.1     rmind 
    226   1.1     rmind 	result[i++] = (intptr_t)cstream_ptr;
    227   1.1     rmind 	result[i++] = cstream_retval;
    228   1.1     rmind 	result[i++] = cstream_state.nst_state;
    229   1.1     rmind 
    230   1.1     rmind 	result[i++] = fstate->nst_end;
    231   1.1     rmind 	result[i++] = fstate->nst_maxend;
    232   1.1     rmind 	result[i++] = fstate->nst_maxwin;
    233   1.2     rmind 	result[i++] = fstate->nst_wscale;
    234   1.1     rmind 
    235   1.1     rmind 	result[i++] = tstate->nst_end;
    236   1.1     rmind 	result[i++] = tstate->nst_maxend;
    237   1.1     rmind 	result[i++] = tstate->nst_maxwin;
    238   1.2     rmind 	result[i++] = tstate->nst_wscale;
    239   1.1     rmind 
    240   1.1     rmind 	return 0;
    241   1.1     rmind }
    242   1.7     rmind 
    243   1.9     rmind int
    244   1.9     rmind npf_inet_pton(int af, const char *src, void *dst)
    245   1.9     rmind {
    246   1.9     rmind 	return _pton_func(af, src, dst);
    247   1.9     rmind }
    248   1.9     rmind 
    249   1.9     rmind const char *
    250   1.9     rmind npf_inet_ntop(int af, const void *src, char *dst, socklen_t size)
    251   1.9     rmind {
    252   1.9     rmind 	return _ntop_func(af, src, dst, size);
    253   1.9     rmind }
    254   1.9     rmind 
    255  1.12  christos #ifdef _KERNEL
    256   1.7     rmind /*
    257  1.11       tls  * Need to override cprng_fast32() -- we need deterministic PRNG.
    258   1.7     rmind  */
    259   1.7     rmind uint32_t
    260  1.11       tls cprng_fast32(void)
    261   1.7     rmind {
    262   1.9     rmind 	return (uint32_t)(_random_func ? _random_func() : random());
    263   1.7     rmind }
    264  1.12  christos #endif
    265