From: Jozsef Kadlecsik Date: Fri, 31 May 2013 20:52:54 +0000 (+0200) Subject: ipset standalone package needs to ship em_ipset.c X-Git-Tag: v6.20~42 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=accfdb5cba698fdaa2c9aeef4a745d47fdd059e8;p=ipset ipset standalone package needs to ship em_ipset.c Due to the ipset kernel API changes, em_ipset.c needs to be provided in the ipset package, reported by Jan Engelhardt: WARNING: //lib/modules/3.7.10-jng15-default/kernel/net/sched/em_ipset.ko disagrees about version of symbol ip_set_test --- diff --git a/Makefile.am b/Makefile.am index c84a425..3fc9cb9 100644 --- a/Makefile.am +++ b/Makefile.am @@ -23,7 +23,7 @@ sparse-check: modules_sparse: if WITH_KMOD - ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net/netfilter \ + ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net \ V=$V W=1 C=2 CF="-D__CHECK_ENDIAN__ -Wsparse-all" \ KCFLAGS="-DPACKAGE_VERSION=$(PACKAGE_VERSION) -DCHECK_KCONFIG -Wextra" \ IP_SET_MAX=$(IP_SET_MAX) KDIR=$$PWD/kernel modules @@ -33,7 +33,7 @@ endif modules: if WITH_KMOD - ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net/netfilter V=$V \ + ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net V=$V \ KCFLAGS="-DPACKAGE_VERSION=$(PACKAGE_VERSION)" \ IP_SET_MAX=$(IP_SET_MAX) KDIR=$$PWD/kernel modules else @@ -59,7 +59,7 @@ export DEPMOD_WARNING modules_install: if WITH_KMOD - ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net/netfilter \ + ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net \ KDIR=$$PWD/kernel modules_install @modinfo ip_set_hash_ip | ${GREP} extra/ipset/ip_set_hash_ip >/dev/null || echo "$$DEPMOD_WARNING" else @@ -68,7 +68,7 @@ endif modules_clean: if WITH_KMOD - ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net/netfilter \ + ${MAKE} -C $(KBUILD_OUTPUT) M=$$PWD/kernel/net \ KDIR=$$PWD/kernel clean else @echo Skipping kernel modules due to --with-kmod=no diff --git a/kernel/net/Kbuild b/kernel/net/Kbuild new file mode 100644 index 0000000..d126970 --- /dev/null +++ b/kernel/net/Kbuild @@ -0,0 +1,9 @@ +NOSTDINC_FLAGS += -I$(KDIR)/include +EXTRA_CFLAGS := -DCONFIG_IP_SET_MAX=$(IP_SET_MAX) + +obj-m += netfilter/ +obj-m += sched/ + +# It's for me... +incdirs := $(M) $(KDIR)/include/linux/netfilter $(KDIR)/include/linux/netfilter/ipset +clean-files := $(foreach dir,$(incdirs),$(wildcard $(dir)/*~)) diff --git a/kernel/net/sched/Kbuild b/kernel/net/sched/Kbuild new file mode 100644 index 0000000..6d8e8d6 --- /dev/null +++ b/kernel/net/sched/Kbuild @@ -0,0 +1,4 @@ +NOSTDINC_FLAGS += -I$(KDIR)/include +EXTRA_CFLAGS := -DCONFIG_IP_SET_MAX=$(IP_SET_MAX) + +obj-$(CONFIG_NET_EMATCH_IPSET) += em_ipset.o diff --git a/kernel/net/sched/em_ipset.c b/kernel/net/sched/em_ipset.c new file mode 100644 index 0000000..938b7cb --- /dev/null +++ b/kernel/net/sched/em_ipset.c @@ -0,0 +1,135 @@ +/* + * net/sched/em_ipset.c ipset ematch + * + * Copyright (c) 2012 Florian Westphal + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * version 2 as published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +static int em_ipset_change(struct tcf_proto *tp, void *data, int data_len, + struct tcf_ematch *em) +{ + struct xt_set_info *set = data; + ip_set_id_t index; + + if (data_len != sizeof(*set)) + return -EINVAL; + + index = ip_set_nfnl_get_byindex(set->index); + if (index == IPSET_INVALID_ID) + return -ENOENT; + + em->datalen = sizeof(*set); + em->data = (unsigned long)kmemdup(data, em->datalen, GFP_KERNEL); + if (em->data) + return 0; + + ip_set_nfnl_put(index); + return -ENOMEM; +} + +static void em_ipset_destroy(struct tcf_proto *p, struct tcf_ematch *em) +{ + const struct xt_set_info *set = (const void *) em->data; + if (set) { + ip_set_nfnl_put(set->index); + kfree((void *) em->data); + } +} + +static int em_ipset_match(struct sk_buff *skb, struct tcf_ematch *em, + struct tcf_pkt_info *info) +{ + struct ip_set_adt_opt opt; + struct xt_action_param acpar; + const struct xt_set_info *set = (const void *) em->data; + struct net_device *dev, *indev = NULL; + int ret, network_offset; + + switch (skb->protocol) { + case htons(ETH_P_IP): + acpar.family = NFPROTO_IPV4; + if (!pskb_network_may_pull(skb, sizeof(struct iphdr))) + return 0; + acpar.thoff = ip_hdrlen(skb); + break; + case htons(ETH_P_IPV6): + acpar.family = NFPROTO_IPV6; + if (!pskb_network_may_pull(skb, sizeof(struct ipv6hdr))) + return 0; + /* doesn't call ipv6_find_hdr() because ipset doesn't use thoff, yet */ + acpar.thoff = sizeof(struct ipv6hdr); + break; + default: + return 0; + } + + acpar.hooknum = 0; + + opt.family = acpar.family; + opt.dim = set->dim; + opt.flags = set->flags; + opt.cmdflags = 0; + opt.ext.timeout = ~0u; + + network_offset = skb_network_offset(skb); + skb_pull(skb, network_offset); + + dev = skb->dev; + + rcu_read_lock(); + + if (dev && skb->skb_iif) + indev = dev_get_by_index_rcu(dev_net(dev), skb->skb_iif); + + acpar.in = indev ? indev : dev; + acpar.out = dev; + + ret = ip_set_test(set->index, skb, &acpar, &opt); + + rcu_read_unlock(); + + skb_push(skb, network_offset); + return ret; +} + +static struct tcf_ematch_ops em_ipset_ops = { + .kind = TCF_EM_IPSET, + .change = em_ipset_change, + .destroy = em_ipset_destroy, + .match = em_ipset_match, + .owner = THIS_MODULE, + .link = LIST_HEAD_INIT(em_ipset_ops.link) +}; + +static int __init init_em_ipset(void) +{ + return tcf_em_register(&em_ipset_ops); +} + +static void __exit exit_em_ipset(void) +{ + tcf_em_unregister(&em_ipset_ops); +} + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Florian Westphal "); +MODULE_DESCRIPTION("TC extended match for IP sets"); + +module_init(init_em_ipset); +module_exit(exit_em_ipset); + +MODULE_ALIAS_TCF_EMATCH(TCF_EM_IPSET);