cregit-Linux how code gets into the kernel

Release 4.14 net/netfilter/x_tables.c

Directory: net/netfilter
/*
 * x_tables core - Backend for {ip,ip6,arp}_tables
 *
 * Copyright (C) 2006-2006 Harald Welte <laforge@netfilter.org>
 * Copyright (C) 2006-2012 Patrick McHardy <kaber@trash.net>
 *
 * Based on existing ip_tables code which is
 *   Copyright (C) 1999 Paul `Rusty' Russell & Michael J. Neuling
 *   Copyright (C) 2000-2005 Netfilter Core Team <coreteam@netfilter.org>
 *
 * 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.
 *
 */

#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/socket.h>
#include <linux/net.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/string.h>
#include <linux/vmalloc.h>
#include <linux/mutex.h>
#include <linux/mm.h>
#include <linux/slab.h>
#include <linux/audit.h>
#include <linux/user_namespace.h>
#include <net/net_namespace.h>

#include <linux/netfilter/x_tables.h>
#include <linux/netfilter_arp.h>
#include <linux/netfilter_ipv4/ip_tables.h>
#include <linux/netfilter_ipv6/ip6_tables.h>
#include <linux/netfilter_arp/arp_tables.h>

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Harald Welte <laforge@netfilter.org>");
MODULE_DESCRIPTION("{ip,ip6,arp,eb}_tables backend module");


#define SMP_ALIGN(x) (((x) + SMP_CACHE_BYTES-1) & ~(SMP_CACHE_BYTES-1))

#define XT_PCPU_BLOCK_SIZE 4096


struct compat_delta {
	
unsigned int offset; /* offset in kernel */
	
int delta; /* delta in 32bit user land */
};


struct xt_af {
	
struct mutex mutex;
	
struct list_head match;
	
struct list_head target;
#ifdef CONFIG_COMPAT
	
struct mutex compat_mutex;
	
struct compat_delta *compat_tab;
	
unsigned int number; /* number of slots in compat_tab[] */
	
unsigned int cur; /* number of used slots in compat_tab[] */
#endif
};


static struct xt_af *xt;


static const char *const xt_prefix[NFPROTO_NUMPROTO] = {
	[NFPROTO_UNSPEC] = "x",
	[NFPROTO_IPV4]   = "ip",
	[NFPROTO_ARP]    = "arp",
	[NFPROTO_BRIDGE] = "eb",
	[NFPROTO_IPV6]   = "ip6",
};

/* Registration hooks for targets. */

int xt_register_target(struct xt_target *target) { u_int8_t af = target->family; mutex_lock(&xt[af].mutex); list_add(&target->list, &xt[af].target); mutex_unlock(&xt[af].mutex); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte4882.76%120.00%
Pablo Neira Ayuso813.79%240.00%
Jan Engelhardt11.72%120.00%
Ingo Molnar11.72%120.00%
Total58100.00%5100.00%

EXPORT_SYMBOL(xt_register_target);
void xt_unregister_target(struct xt_target *target) { u_int8_t af = target->family; mutex_lock(&xt[af].mutex); list_del(&target->list); mutex_unlock(&xt[af].mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte3574.47%120.00%
Pablo Neira Ayuso612.77%120.00%
Patrick McHardy36.38%120.00%
Ingo Molnar24.26%120.00%
Jan Engelhardt12.13%120.00%
Total47100.00%5100.00%

EXPORT_SYMBOL(xt_unregister_target);
int xt_register_targets(struct xt_target *target, unsigned int n) { unsigned int i; int err = 0; for (i = 0; i < n; i++) { err = xt_register_target(&target[i]); if (err) goto err; } return err; err: if (i > 0) xt_unregister_targets(target, i); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy7698.70%150.00%
Harald Welte11.30%150.00%
Total77100.00%2100.00%

EXPORT_SYMBOL(xt_register_targets);
void xt_unregister_targets(struct xt_target *target, unsigned int n) { while (n-- > 0) xt_unregister_target(&target[n]); }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy2376.67%150.00%
Changli Gao723.33%150.00%
Total30100.00%2100.00%

EXPORT_SYMBOL(xt_unregister_targets);
int xt_register_match(struct xt_match *match) { u_int8_t af = match->family; mutex_lock(&xt[af].mutex); list_add(&match->list, &xt[af].match); mutex_unlock(&xt[af].mutex); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte4882.76%120.00%
Pablo Neira Ayuso813.79%240.00%
Jan Engelhardt11.72%120.00%
Ingo Molnar11.72%120.00%
Total58100.00%5100.00%

EXPORT_SYMBOL(xt_register_match);
void xt_unregister_match(struct xt_match *match) { u_int8_t af = match->family; mutex_lock(&xt[af].mutex); list_del(&match->list); mutex_unlock(&xt[af].mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte3574.47%120.00%
Pablo Neira Ayuso612.77%120.00%
Patrick McHardy36.38%120.00%
Ingo Molnar24.26%120.00%
Jan Engelhardt12.13%120.00%
Total47100.00%5100.00%

EXPORT_SYMBOL(xt_unregister_match);
int xt_register_matches(struct xt_match *match, unsigned int n) { unsigned int i; int err = 0; for (i = 0; i < n; i++) { err = xt_register_match(&match[i]); if (err) goto err; } return err; err: if (i > 0) xt_unregister_matches(match, i); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy77100.00%1100.00%
Total77100.00%1100.00%

EXPORT_SYMBOL(xt_register_matches);
void xt_unregister_matches(struct xt_match *match, unsigned int n) { while (n-- > 0) xt_unregister_match(&match[n]); }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy2376.67%150.00%
Changli Gao723.33%150.00%
Total30100.00%2100.00%

EXPORT_SYMBOL(xt_unregister_matches); /* * These are weird, but module loading must not be done with mutex * held (since they will register), and we have to have a single * function to use. */ /* Find match, grabs ref. Returns ERR_PTR() on error. */
struct xt_match *xt_find_match(u8 af, const char *name, u8 revision) { struct xt_match *m; int err = -ENOENT; mutex_lock(&xt[af].mutex); list_for_each_entry(m, &xt[af].match, list) { if (strcmp(m->name, name) == 0) { if (m->revision == revision) { if (try_module_get(m->me)) { mutex_unlock(&xt[af].mutex); return m; } } else err = -EPROTOTYPE; /* Found something. */ } } mutex_unlock(&xt[af].mutex); if (af != NFPROTO_UNSPEC) /* Try searching again in the family-independent list */ return xt_find_match(NFPROTO_UNSPEC, name, revision); return ERR_PTR(err); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte11782.98%116.67%
Jan Engelhardt1812.77%233.33%
Ingo Molnar21.42%116.67%
Pablo Neira Ayuso21.42%116.67%
Patrick McHardy21.42%116.67%
Total141100.00%6100.00%

EXPORT_SYMBOL(xt_find_match);
struct xt_match * xt_request_find_match(uint8_t nfproto, const char *name, uint8_t revision) { struct xt_match *match; match = xt_find_match(nfproto, name, revision); if (IS_ERR(match)) { request_module("%st_%s", xt_prefix[nfproto], name); match = xt_find_match(nfproto, name, revision); } return match; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt4768.12%150.00%
Stephen Hemminger2231.88%150.00%
Total69100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_request_find_match); /* Find target, grabs ref. Returns ERR_PTR() on error. */
struct xt_target *xt_find_target(u8 af, const char *name, u8 revision) { struct xt_target *t; int err = -ENOENT; mutex_lock(&xt[af].mutex); list_for_each_entry(t, &xt[af].target, list) { if (strcmp(t->name, name) == 0) { if (t->revision == revision) { if (try_module_get(t->me)) { mutex_unlock(&xt[af].mutex); return t; } } else err = -EPROTOTYPE; /* Found something. */ } } mutex_unlock(&xt[af].mutex); if (af != NFPROTO_UNSPEC) /* Try searching again in the family-independent list */ return xt_find_target(NFPROTO_UNSPEC, name, revision); return ERR_PTR(err); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte11782.98%116.67%
Jan Engelhardt1812.77%233.33%
Patrick McHardy21.42%116.67%
Pablo Neira Ayuso21.42%116.67%
Ingo Molnar21.42%116.67%
Total141100.00%6100.00%

EXPORT_SYMBOL(xt_find_target);
struct xt_target *xt_request_find_target(u8 af, const char *name, u8 revision) { struct xt_target *target; target = xt_find_target(af, name, revision); if (IS_ERR(target)) { request_module("%st_%s", xt_prefix[af], name); target = xt_find_target(af, name, revision); } return target; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte4260.87%120.00%
Stephen Hemminger2231.88%120.00%
Jan Engelhardt34.35%240.00%
Patrick McHardy22.90%120.00%
Total69100.00%5100.00%

EXPORT_SYMBOL_GPL(xt_request_find_target);
static int xt_obj_to_user(u16 __user *psize, u16 size, void __user *pname, const char *name, u8 __user *prev, u8 rev) { if (put_user(size, psize)) return -EFAULT; if (copy_to_user(pname, name, strlen(name) + 1)) return -EFAULT; if (put_user(rev, prev)) return -EFAULT; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Willem de Bruijn81100.00%1100.00%
Total81100.00%1100.00%

#define XT_OBJ_TO_USER(U, K, TYPE, C_SIZE) \ xt_obj_to_user(&U->u.TYPE##_size, C_SIZE ? : K->u.TYPE##_size, \ U->u.user.name, K->u.kernel.TYPE->name, \ &U->u.user.revision, K->u.kernel.TYPE->revision)
int xt_data_to_user(void __user *dst, const void *src, int usersize, int size, int aligned_size) { usersize = usersize ? : size; if (copy_to_user(dst, src, usersize)) return -EFAULT; if (usersize != aligned_size && clear_user(dst + usersize, aligned_size - usersize)) return -EFAULT; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Willem de Bruijn70100.00%2100.00%
Total70100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_data_to_user); #define XT_DATA_TO_USER(U, K, TYPE) \ xt_data_to_user(U->data, K->data, \ K->u.kernel.TYPE->usersize, \ K->u.kernel.TYPE->TYPE##size, \ XT_ALIGN(K->u.kernel.TYPE->TYPE##size))
int xt_match_to_user(const struct xt_entry_match *m, struct xt_entry_match __user *u) { return XT_OBJ_TO_USER(u, m, match, 0) || XT_DATA_TO_USER(u, m, match); }

Contributors

PersonTokensPropCommitsCommitProp
Willem de Bruijn38100.00%1100.00%
Total38100.00%1100.00%

EXPORT_SYMBOL_GPL(xt_match_to_user);
int xt_target_to_user(const struct xt_entry_target *t, struct xt_entry_target __user *u) { return XT_OBJ_TO_USER(u, t, target, 0) || XT_DATA_TO_USER(u, t, target); }

Contributors

PersonTokensPropCommitsCommitProp
Willem de Bruijn38100.00%1100.00%
Total38100.00%1100.00%

EXPORT_SYMBOL_GPL(xt_target_to_user);
static int match_revfn(u8 af, const char *name, u8 revision, int *bestp) { const struct xt_match *m; int have_rev = 0; list_for_each_entry(m, &xt[af].match, list) { if (strcmp(m->name, name) == 0) { if (m->revision > *bestp) *bestp = m->revision; if (m->revision == revision) have_rev = 1; } } if (af != NFPROTO_UNSPEC && !have_rev) return match_revfn(NFPROTO_UNSPEC, name, revision, bestp); return have_rev; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte8678.90%125.00%
Patrick McHardy2119.27%125.00%
Jan Engelhardt21.83%250.00%
Total109100.00%4100.00%


static int target_revfn(u8 af, const char *name, u8 revision, int *bestp) { const struct xt_target *t; int have_rev = 0; list_for_each_entry(t, &xt[af].target, list) { if (strcmp(t->name, name) == 0) { if (t->revision > *bestp) *bestp = t->revision; if (t->revision == revision) have_rev = 1; } } if (af != NFPROTO_UNSPEC && !have_rev) return target_revfn(NFPROTO_UNSPEC, name, revision, bestp); return have_rev; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte8678.90%125.00%
Patrick McHardy2119.27%125.00%
Jan Engelhardt21.83%250.00%
Total109100.00%4100.00%

/* Returns true or false (if no such extension at all) */
int xt_find_revision(u8 af, const char *name, u8 revision, int target, int *err) { int have_rev, best = -1; mutex_lock(&xt[af].mutex); if (target == 1) have_rev = target_revfn(af, name, revision, &best); else have_rev = match_revfn(af, name, revision, &best); mutex_unlock(&xt[af].mutex); /* Nothing at all? Return 0 to try loading module. */ if (best == -1) { *err = -ENOENT; return 0; } *err = best; if (!have_rev) *err = -EPROTONOSUPPORT; return 1; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte12397.62%125.00%
Jan Engelhardt10.79%125.00%
Pablo Neira Ayuso10.79%125.00%
Ingo Molnar10.79%125.00%
Total126100.00%4100.00%

EXPORT_SYMBOL_GPL(xt_find_revision);
static char * textify_hooks(char *buf, size_t size, unsigned int mask, uint8_t nfproto) { static const char *const inetbr_names[] = { "PREROUTING", "INPUT", "FORWARD", "OUTPUT", "POSTROUTING", "BROUTING", }; static const char *const arp_names[] = { "INPUT", "FORWARD", "OUTPUT", }; const char *const *names; unsigned int i, max; char *p = buf; bool np = false; int res; names = (nfproto == NFPROTO_ARP) ? arp_names : inetbr_names; max = (nfproto == NFPROTO_ARP) ? ARRAY_SIZE(arp_names) : ARRAY_SIZE(inetbr_names); *p = '\0'; for (i = 0; i < max; ++i) { if (!(mask & (1 << i))) continue; res = snprintf(p, size, "%s%s", np ? "/" : "", names[i]); if (res > 0) { size -= res; p += res; } np = true; } return buf; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt195100.00%2100.00%
Total195100.00%2100.00%


int xt_check_match(struct xt_mtchk_param *par, unsigned int size, u_int8_t proto, bool inv_proto) { int ret; if (XT_ALIGN(par->match->matchsize) != size && par->match->matchsize != -1) { /* * ebt_among is exempt from centralized matchsize checking * because it uses a dynamic-size data set. */ pr_err("%s_tables: %s.%u match: invalid size " "%u (kernel) != (user) %u\n", xt_prefix[par->family], par->match->name, par->match->revision, XT_ALIGN(par->match->matchsize), size); return -EINVAL; } if (par->match->table != NULL && strcmp(par->match->table, par->table) != 0) { pr_err("%s_tables: %s match: only valid in %s table, not %s\n", xt_prefix[par->family], par->match->name, par->match->table, par->table); return -EINVAL; } if (par->match->hooks && (par->hook_mask & ~par->match->hooks) != 0) { char used[64], allow[64]; pr_err("%s_tables: %s match: used from hooks %s, but only " "valid from %s\n", xt_prefix[par->family], par->match->name, textify_hooks(used, sizeof(used), par->hook_mask, par->family), textify_hooks(allow, sizeof(allow), par->match->hooks, par->family)); return -EINVAL; } if (par->match->proto && (par->match->proto != proto || inv_proto)) { pr_err("%s_tables: %s match: only valid for protocol %u\n", xt_prefix[par->family], par->match->name, par->match->proto); return -EINVAL; } if (par->match->checkentry != NULL) { ret = par->match->checkentry(par); if (ret < 0) return ret; else if (ret > 0) /* Flag up potential errors. */ return -EIO; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy17350.73%19.09%
Jan Engelhardt16046.92%872.73%
Balazs Scheidler41.17%19.09%
Joe Perches41.17%19.09%
Total341100.00%11100.00%

EXPORT_SYMBOL_GPL(xt_check_match); /** xt_check_entry_match - check that matches end before start of target * * @match: beginning of xt_entry_match * @target: beginning of this rules target (alleged end of matches) * @alignment: alignment requirement of match structures * * Validates that all matches add up to the beginning of the target, * and that each match covers at least the base structure size. * * Return: 0 on success, negative errno on failure. */
static int xt_check_entry_match(const char *match, const char *target, const size_t alignment) { const struct xt_entry_match *pos; int length = target - match; if (length == 0) /* no matches */ return 0; pos = (struct xt_entry_match *)match; do { if ((unsigned long)pos % alignment) return -EINVAL; if (length < (int)sizeof(struct xt_entry_match)) return -EINVAL; if (pos->u.match_size < sizeof(struct xt_entry_match)) return -EINVAL; if (pos->u.match_size > length) return -EINVAL; length -= pos->u.match_size; pos = ((void *)((char *)(pos) + (pos)->u.match_size)); } while (length > 0); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal162100.00%1100.00%
Total162100.00%1100.00%

#ifdef CONFIG_COMPAT
int xt_compat_add_offset(u_int8_t af, unsigned int offset, int delta) { struct xt_af *xp = &xt[af]; if (!xp->compat_tab) { if (!xp->number) return -EINVAL; xp->compat_tab = vmalloc(sizeof(struct compat_delta) * xp->number); if (!xp->compat_tab) return -ENOMEM; xp->cur = 0; } if (xp->cur >= xp->number) return -EINVAL; if (xp->cur) delta += xp->compat_tab[xp->cur - 1].delta; xp->compat_tab[xp->cur].offset = offset; xp->compat_tab[xp->cur].delta = delta; xp->cur++; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Eric Dumazet8859.06%133.33%
Patrick McHardy6040.27%133.33%
Jan Engelhardt10.67%133.33%
Total149100.00%3100.00%

EXPORT_SYMBOL_GPL(xt_compat_add_offset);
void xt_compat_flush_offsets(u_int8_t af) { if (xt[af].compat_tab) { vfree(xt[af].compat_tab); xt[af].compat_tab = NULL; xt[af].number = 0; xt[af].cur = 0; } }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy3257.14%125.00%
Eric Dumazet2341.07%250.00%
Jan Engelhardt11.79%125.00%
Total56100.00%4100.00%

EXPORT_SYMBOL_GPL(xt_compat_flush_offsets);
int xt_compat_calc_jump(u_int8_t af, unsigned int offset) { struct compat_delta *tmp = xt[af].compat_tab; int mid, left = 0, right = xt[af].cur - 1; while (left <= right) { mid = (left + right) >> 1; if (offset > tmp[mid].offset) left = mid + 1; else if (offset < tmp[mid].offset) right = mid - 1; else return mid ? tmp[mid - 1].delta : 0; } return left ? tmp[left - 1].delta : 0; }

Contributors

PersonTokensPropCommitsCommitProp
Eric Dumazet8870.97%240.00%
Patrick McHardy3326.61%120.00%
Florian Westphal21.61%120.00%
Jan Engelhardt10.81%120.00%
Total124100.00%5100.00%

EXPORT_SYMBOL_GPL(xt_compat_calc_jump);
void xt_compat_init_offsets(u_int8_t af, unsigned int number) { xt[af].number = number; xt[af].cur = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Eric Dumazet30100.00%1100.00%
Total30100.00%1100.00%

EXPORT_SYMBOL(xt_compat_init_offsets);
int xt_compat_match_offset(const struct xt_match *match) { u_int16_t csize = match->compatsize ? : match->matchsize; return XT_ALIGN(match->matchsize) - COMPAT_XT_ALIGN(csize); }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy3186.11%133.33%
Dmitry Mishin411.11%133.33%
Jan Engelhardt12.78%133.33%
Total36100.00%3100.00%

EXPORT_SYMBOL_GPL(xt_compat_match_offset);
void xt_compat_match_from_user(struct xt_entry_match *m, void **dstptr, unsigned int *size) { const struct xt_match *match = m->u.kernel.match; struct compat_xt_entry_match *cm = (struct compat_xt_entry_match *)m; int pad, off = xt_compat_match_offset(match); u_int16_t msize = cm->u.user.match_size; char name[sizeof(m->u.user.name)]; m = *dstptr; memcpy(m, cm, sizeof(*cm)); if (match->compat_from_user) match->compat_from_user(m->data, cm->data); else memcpy(m->data, cm->data, msize - sizeof(*cm)); pad = XT_ALIGN(match->matchsize) - match->matchsize; if (pad > 0) memset(m->data + match->matchsize, 0, pad); msize += off; m->u.user.match_size = msize; strlcpy(name, match->name, sizeof(name)); module_put(match->me); strncpy(m->u.user.name, name, sizeof(m->u.user.name)); *size += off; *dstptr += msize; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy9940.91%233.33%
Dmitry Mishin8133.47%116.67%
Florian Westphal6125.21%233.33%
Jan Engelhardt10.41%116.67%
Total242100.00%6100.00%

EXPORT_SYMBOL_GPL(xt_compat_match_from_user); #define COMPAT_XT_DATA_TO_USER(U, K, TYPE, C_SIZE) \ xt_data_to_user(U->data, K->data, \ K->u.kernel.TYPE->usersize, \ C_SIZE, \ COMPAT_XT_ALIGN(C_SIZE))
int xt_compat_match_to_user(const struct xt_entry_match *m, void __user **dstptr, unsigned int *size) { const struct xt_match *match = m->u.kernel.match; struct compat_xt_entry_match __user *cm = *dstptr; int off = xt_compat_match_offset(match); u_int16_t msize = m->u.user.match_size - off; if (XT_OBJ_TO_USER(cm, m, match, msize)) return -EFAULT; if (match->compat_to_user) { if (match->compat_to_user((void __user *)cm->data, m->data)) return -EFAULT; } else { if (COMPAT_XT_DATA_TO_USER(cm, m, match, msize - sizeof(*cm))) return -EFAULT; } *size -= off; *dstptr += msize; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy14795.45%450.00%
Willem de Bruijn53.25%225.00%
Jan Engelhardt21.30%225.00%
Total154100.00%8100.00%

EXPORT_SYMBOL_GPL(xt_compat_match_to_user); /* non-compat version may have padding after verdict */ struct compat_xt_standard_target { struct compat_xt_entry_target t; compat_uint_t verdict; };
int xt_compat_check_entry_offsets(const void *base, const char *elems, unsigned int target_offset, unsigned int next_offset) { long size_of_base_struct = elems - (const char *)base; const struct compat_xt_entry_target *t; const char *e = base; if (target_offset < size_of_base_struct) return -EINVAL; if (target_offset + sizeof(*t) > next_offset) return -EINVAL; t = (void *)(e + target_offset); if (t->u.target_size < sizeof(*t)) return -EINVAL; if (target_offset + t->u.target_size > next_offset) return -EINVAL; if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0 && COMPAT_XT_ALIGN(target_offset + sizeof(struct compat_xt_standard_target)) != next_offset) return -EINVAL; /* compat_xt_entry match has less strict alignment requirements, * otherwise they are identical. In case of padding differences * we need to add compat version of xt_check_entry_match. */ BUILD_BUG_ON(sizeof(struct compat_xt_entry_match) != sizeof(struct xt_entry_match)); return xt_check_entry_match(elems, base + target_offset, __alignof__(struct compat_xt_entry_match)); }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal18599.46%583.33%
Masahiro Yamada10.54%116.67%
Total186100.00%6100.00%

EXPORT_SYMBOL(xt_compat_check_entry_offsets); #endif /* CONFIG_COMPAT */ /** * xt_check_entry_offsets - validate arp/ip/ip6t_entry * * @base: pointer to arp/ip/ip6t_entry * @elems: pointer to first xt_entry_match, i.e. ip(6)t_entry->elems * @target_offset: the arp/ip/ip6_t->target_offset * @next_offset: the arp/ip/ip6_t->next_offset * * validates that target_offset and next_offset are sane and that all * match sizes (if any) align with the target offset. * * This function does not validate the targets or matches themselves, it * only tests that all the offsets and sizes are correct, that all * match structures are aligned, and that the last structure ends where * the target structure begins. * * Also see xt_compat_check_entry_offsets for CONFIG_COMPAT version. * * The arp/ip/ip6t_entry structure @base must have passed following tests: * - it must point to a valid memory location * - base to base + next_offset must be accessible, i.e. not exceed allocated * length. * * A well-formed entry looks like this: * * ip(6)t_entry match [mtdata] match [mtdata] target [tgdata] ip(6)t_entry * e->elems[]-----' | | * matchsize | | * matchsize | | * | | * target_offset---------------------------------' | * next_offset---------------------------------------------------' * * elems[]: flexible array member at end of ip(6)/arpt_entry struct. * This is where matches (if any) and the target reside. * target_offset: beginning of target. * next_offset: start of the next rule; also: size of this rule. * Since targets have a minimum size, target_offset + minlen <= next_offset. * * Every match stores its size, sum of sizes must not exceed target_offset. * * Return: 0 on success, negative errno on failure. */
int xt_check_entry_offsets(const void *base, const char *elems, unsigned int target_offset, unsigned int next_offset) { long size_of_base_struct = elems - (const char *)base; const struct xt_entry_target *t; const char *e = base; /* target start is within the ip/ip6/arpt_entry struct */ if (target_offset < size_of_base_struct) return -EINVAL; if (target_offset + sizeof(*t) > next_offset) return -EINVAL; t = (void *)(e + target_offset); if (t->u.target_size < sizeof(*t)) return -EINVAL; if (target_offset + t->u.target_size > next_offset) return -EINVAL; if (strcmp(t->u.user.name, XT_STANDARD_TARGET) == 0 && XT_ALIGN(target_offset + sizeof(struct xt_standard_target)) != next_offset) return -EINVAL; return xt_check_entry_match(elems, base + target_offset, __alignof__(struct xt_entry_match)); }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal171100.00%6100.00%
Total171100.00%6100.00%

EXPORT_SYMBOL(xt_check_entry_offsets); /** * xt_alloc_entry_offsets - allocate array to store rule head offsets * * @size: number of entries * * Return: NULL or kmalloc'd or vmalloc'd array */
unsigned int *xt_alloc_entry_offsets(unsigned int size) { return kvmalloc_array(size, sizeof(unsigned int), GFP_KERNEL | __GFP_ZERO); }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal2488.89%150.00%
Michal Hocko311.11%150.00%
Total27100.00%2100.00%

EXPORT_SYMBOL(xt_alloc_entry_offsets); /** * xt_find_jump_offset - check if target is a valid jump offset * * @offsets: array containing all valid rule start offsets of a rule blob * @target: the jump target to search for * @size: entries in @offset */
bool xt_find_jump_offset(const unsigned int *offsets, unsigned int target, unsigned int size) { int m, low = 0, hi = size; while (hi > low) { m = (low + hi) / 2u; if (offsets[m] > target) hi = m; else if (offsets[m] < target) low = m + 1; else return true; } return false; }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal84100.00%1100.00%
Total84100.00%1100.00%

EXPORT_SYMBOL(xt_find_jump_offset);
int xt_check_target(struct xt_tgchk_param *par, unsigned int size, u_int8_t proto, bool inv_proto) { int ret; if (XT_ALIGN(par->target->targetsize) != size) { pr_err("%s_tables: %s.%u target: invalid size " "%u (kernel) != (user) %u\n", xt_prefix[par->family], par->target->name, par->target->revision, XT_ALIGN(par->target->targetsize), size); return -EINVAL; } if (par->target->table != NULL && strcmp(par->target->table, par->table) != 0) { pr_err("%s_tables: %s target: only valid in %s table, not %s\n", xt_prefix[par->family], par->target->name, par->target->table, par->table); return -EINVAL; } if (par->target->hooks && (par->hook_mask & ~par->target->hooks) != 0) { char used[64], allow[64]; pr_err("%s_tables: %s target: used from hooks %s, but only " "usable from %s\n", xt_prefix[par->family], par->target->name, textify_hooks(used, sizeof(used), par->hook_mask, par->family), textify_hooks(allow, sizeof(allow), par->target->hooks, par->family)); return -EINVAL; } if (par->target->proto && (par->target->proto != proto || inv_proto)) { pr_err("%s_tables: %s target: only valid for protocol %u\n", xt_prefix[par->family], par->target->name, par->target->proto); return -EINVAL; } if (par->target->checkentry != NULL) { ret = par->target->checkentry(par); if (ret < 0) return ret; else if (ret > 0) /* Flag up potential errors. */ return -EIO; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy17352.27%218.18%
Jan Engelhardt15045.32%763.64%
Balazs Scheidler41.21%19.09%
Joe Perches41.21%19.09%
Total331100.00%11100.00%

EXPORT_SYMBOL_GPL(xt_check_target); /** * xt_copy_counters_from_user - copy counters and metadata from userspace * * @user: src pointer to userspace memory * @len: alleged size of userspace memory * @info: where to store the xt_counters_info metadata * @compat: true if we setsockopt call is done by 32bit task on 64bit kernel * * Copies counter meta data from @user and stores it in @info. * * vmallocs memory to hold the counters, then copies the counter data * from @user to the new memory and returns a pointer to it. * * If @compat is true, @info gets converted automatically to the 64bit * representation. * * The metadata associated with the counters is stored in @info. * * Return: returns pointer that caller has to test via IS_ERR(). * If IS_ERR is false, caller has to vfree the pointer. */
void *xt_copy_counters_from_user(const void __user *user, unsigned int len, struct xt_counters_info *info, bool compat) { void *mem; u64 size; #ifdef CONFIG_COMPAT if (compat) { /* structures only differ in size due to alignment */ struct compat_xt_counters_info compat_tmp; if (len <= sizeof(compat_tmp)) return ERR_PTR(-EINVAL); len -= sizeof(compat_tmp); if (copy_from_user(&compat_tmp, user, sizeof(compat_tmp)) != 0) return ERR_PTR(-EFAULT); memcpy(info->name, compat_tmp.name, sizeof(info->name) - 1); info->num_counters = compat_tmp.num_counters; user += sizeof(compat_tmp); } else #endif { if (len <= sizeof(*info)) return ERR_PTR(-EINVAL); len -= sizeof(*info); if (copy_from_user(info, user, sizeof(*info)) != 0) return ERR_PTR(-EFAULT); user += sizeof(*info); } info->name[sizeof(info->name) - 1] = '\0'; size = sizeof(struct xt_counters); size *= info->num_counters; if (size != (u64)len) return ERR_PTR(-EINVAL); mem = vmalloc(len); if (!mem) return ERR_PTR(-ENOMEM); if (copy_from_user(mem, user, len) == 0) return mem; vfree(mem); return ERR_PTR(-EFAULT); }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal27095.74%150.00%
Eric Dumazet124.26%150.00%
Total282100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_copy_counters_from_user); #ifdef CONFIG_COMPAT
int xt_compat_target_offset(const struct xt_target *target) { u_int16_t csize = target->compatsize ? : target->targetsize; return XT_ALIGN(target->targetsize) - COMPAT_XT_ALIGN(csize); }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy3186.11%133.33%
Dmitry Mishin411.11%133.33%
Jan Engelhardt12.78%133.33%
Total36100.00%3100.00%

EXPORT_SYMBOL_GPL(xt_compat_target_offset);
void xt_compat_target_from_user(struct xt_entry_target *t, void **dstptr, unsigned int *size) { const struct xt_target *target = t->u.kernel.target; struct compat_xt_entry_target *ct = (struct compat_xt_entry_target *)t; int pad, off = xt_compat_target_offset(target); u_int16_t tsize = ct->u.user.target_size; char name[sizeof(t->u.user.name)]; t = *dstptr; memcpy(t, ct, sizeof(*ct)); if (target->compat_from_user) target->compat_from_user(t->data, ct->data); else memcpy(t->data, ct->data, tsize - sizeof(*ct)); pad = XT_ALIGN(target->targetsize) - target->targetsize; if (pad > 0) memset(t->data + target->targetsize, 0, pad); tsize += off; t->u.user.target_size = tsize; strlcpy(name, target->name, sizeof(name)); module_put(target->me); strncpy(t->u.user.name, name, sizeof(t->u.user.name)); *size += off; *dstptr += tsize; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy11647.93%240.00%
Dmitry Mishin6526.86%120.00%
Florian Westphal6024.79%120.00%
Jan Engelhardt10.41%120.00%
Total242100.00%5100.00%

EXPORT_SYMBOL_GPL(xt_compat_target_from_user);
int xt_compat_target_to_user(const struct xt_entry_target *t, void __user **dstptr, unsigned int *size) { const struct xt_target *target = t->u.kernel.target; struct compat_xt_entry_target __user *ct = *dstptr; int off = xt_compat_target_offset(target); u_int16_t tsize = t->u.user.target_size - off; if (XT_OBJ_TO_USER(ct, t, target, tsize)) return -EFAULT; if (target->compat_to_user) { if (target->compat_to_user((void __user *)ct->data, t->data)) return -EFAULT; } else { if (COMPAT_XT_DATA_TO_USER(ct, t, target, tsize - sizeof(*ct))) return -EFAULT; } *size -= off; *dstptr += tsize; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Patrick McHardy12178.57%337.50%
Dmitry Mishin2616.88%112.50%
Willem de Bruijn53.25%225.00%
Jan Engelhardt21.30%225.00%
Total154100.00%8100.00%

EXPORT_SYMBOL_GPL(xt_compat_target_to_user); #endif
struct xt_table_info *xt_alloc_table_info(unsigned int size) { struct xt_table_info *info = NULL; size_t sz = sizeof(*info) + size; if (sz < sizeof(*info)) return NULL; /* Pedantry: prevent them from hitting BUG() in vmalloc.c --RR */ if ((SMP_ALIGN(size) >> PAGE_SHIFT) + 2 > totalram_pages) return NULL; info = kvmalloc(sz, GFP_KERNEL); if (!info) return NULL; memset(info, 0, sizeof(*info)); info->size = size; return info; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte4545.00%116.67%
Eric Dumazet3838.00%116.67%
Florian Westphal1313.00%116.67%
Marcelo Ricardo Leitner22.00%116.67%
Michal Hocko11.00%116.67%
Jan Beulich11.00%116.67%
Total100100.00%6100.00%

EXPORT_SYMBOL(xt_alloc_table_info);
void xt_free_table_info(struct xt_table_info *info) { int cpu; if (info->jumpstack != NULL) { for_each_possible_cpu(cpu) kvfree(info->jumpstack[cpu]); kvfree(info->jumpstack); } kvfree(info); }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt2959.18%125.00%
Harald Welte1734.69%125.00%
Eric Dumazet36.12%250.00%
Total49100.00%4100.00%

EXPORT_SYMBOL(xt_free_table_info); /* Find table by name, grabs mutex & ref. Returns NULL on error. */
struct xt_table *xt_find_table_lock(struct net *net, u_int8_t af, const char *name) { struct xt_table *t, *found = NULL; mutex_lock(&xt[af].mutex); list_for_each_entry(t, &net->xt.tables[af], list) if (strcmp(t->name, name) == 0 && try_module_get(t->me)) return t; if (net == &init_net) goto out; /* Table doesn't exist in this netns, re-try init */ list_for_each_entry(t, &init_net.xt.tables[af], list) { if (strcmp(t->name, name)) continue; if (!try_module_get(t->me)) { mutex_unlock(&xt[af].mutex); return NULL; } mutex_unlock(&xt[af].mutex); if (t->table_init(net) != 0) { module_put(t->me); return NULL; } found = t; mutex_lock(&xt[af].mutex); break; } if (!found) goto out; /* and once again: */ list_for_each_entry(t, &net->xt.tables[af], list) if (strcmp(t->name, name) == 0) return t; module_put(found->me); out: mutex_unlock(&xt[af].mutex); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal14360.34%114.29%
Harald Welte7129.96%114.29%
Dan Carpenter135.49%114.29%
Alexey Dobriyan62.53%114.29%
Pablo Neira Ayuso20.84%114.29%
Jan Engelhardt10.42%114.29%
Ingo Molnar10.42%114.29%
Total237100.00%7100.00%

EXPORT_SYMBOL_GPL(xt_find_table_lock);
void xt_table_unlock(struct xt_table *table) { mutex_unlock(&xt[table->af].mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte2295.65%150.00%
Ingo Molnar14.35%150.00%
Total23100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_table_unlock); #ifdef CONFIG_COMPAT
void xt_compat_lock(u_int8_t af) { mutex_lock(&xt[af].compat_mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Dmitry Mishin1894.74%150.00%
Jan Engelhardt15.26%150.00%
Total19100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_compat_lock);
void xt_compat_unlock(u_int8_t af) { mutex_unlock(&xt[af].compat_mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Dmitry Mishin1894.74%150.00%
Jan Engelhardt15.26%150.00%
Total19100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_compat_unlock); #endif DEFINE_PER_CPU(seqcount_t, xt_recseq); EXPORT_PER_CPU_SYMBOL_GPL(xt_recseq); struct static_key xt_tee_enabled __read_mostly; EXPORT_SYMBOL_GPL(xt_tee_enabled);
static int xt_jumpstack_alloc(struct xt_table_info *i) { unsigned int size; int cpu; size = sizeof(void **) * nr_cpu_ids; if (size > PAGE_SIZE) i->jumpstack = kvzalloc(size, GFP_KERNEL); else i->jumpstack = kzalloc(size, GFP_KERNEL); if (i->jumpstack == NULL) return -ENOMEM; /* ruleset without jumps -- no stack needed */ if (i->stacksize == 0) return 0; /* Jumpstack needs to be able to record two full callchains, one * from the first rule set traversal, plus one table reentrancy * via -j TEE without clobbering the callchain that brought us to * TEE target. * * This is done by allocating two jumpstacks per cpu, on reentry * the upper half of the stack is used. * * see the jumpstack setup in ipt_do_table() for more details. */ size = sizeof(void *) * i->stacksize * 2u; for_each_possible_cpu(cpu) { i->jumpstack[cpu] = kvmalloc_node(size, GFP_KERNEL, cpu_to_node(cpu)); if (i->jumpstack[cpu] == NULL) /* * Freeing will be done later on by the callers. The * chain is: xt_replace_table -> __do_replace -> * do_replace -> xt_free_table_info. */ return -ENOMEM; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt12185.82%120.00%
Florian Westphal1510.64%240.00%
Michal Hocko42.84%120.00%
Joe Perches10.71%120.00%
Total141100.00%5100.00%


struct xt_table_info * xt_replace_table(struct xt_table *table, unsigned int num_counters, struct xt_table_info *newinfo, int *error) { struct xt_table_info *private; int ret; ret = xt_jumpstack_alloc(newinfo); if (ret < 0) { *error = ret; return NULL; } /* Do the substitution. */ local_bh_disable(); private = table->private; /* Check inside lock: is the old number correct? */ if (num_counters != private->number) { pr_debug("num_counters != table->private->number (%u/%u)\n", num_counters, private->number); local_bh_enable(); *error = -EAGAIN; return NULL; } newinfo->initial_entries = private->initial_entries; /* * Ensure contents of newinfo are visible before assigning to * private. */ smp_wmb(); table->private = newinfo; /* * Even though table entries have now been swapped, other CPU's * may still be using the old entries. This is okay, because * resynchronization happens because of the locking done * during the get_counters() routine. */ local_bh_enable(); #ifdef CONFIG_AUDIT if (audit_enabled) { audit_log(current->audit_context, GFP_KERNEL, AUDIT_NETFILTER_CFG, "table=%s family=%u entries=%u", table->name, table->af, private->number); } #endif return private; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte7848.45%111.11%
Thomas Graf3521.74%111.11%
Jan Engelhardt2716.77%333.33%
Stephen Hemminger106.21%222.22%
Will Deacon106.21%111.11%
Geliang Tang10.62%111.11%
Total161100.00%9100.00%

EXPORT_SYMBOL_GPL(xt_replace_table);
struct xt_table *xt_register_table(struct net *net, const struct xt_table *input_table, struct xt_table_info *bootstrap, struct xt_table_info *newinfo) { int ret; struct xt_table_info *private; struct xt_table *t, *table; /* Don't add one object to multiple lists. */ table = kmemdup(input_table, sizeof(struct xt_table), GFP_KERNEL); if (!table) { ret = -ENOMEM; goto out; } mutex_lock(&xt[table->af].mutex); /* Don't autoload: we'd eat our tail... */ list_for_each_entry(t, &net->xt.tables[table->af], list) { if (strcmp(t->name, table->name) == 0) { ret = -EEXIST; goto unlock; } } /* Simplifies replace_table code. */ table->private = bootstrap; if (!xt_replace_table(table, 0, newinfo, &ret)) goto unlock; private = table->private; pr_debug("table->private->number = %u\n", private->number); /* save number of initial entries */ private->initial_entries = private->number; list_add(&table->list, &net->xt.tables[table->af]); mutex_unlock(&xt[table->af].mutex); return table; unlock: mutex_unlock(&xt[table->af].mutex); kfree(table); out: return ERR_PTR(ret); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte13556.96%111.11%
Alexey Dobriyan6828.69%333.33%
Patrick McHardy2510.55%111.11%
Jan Engelhardt72.95%222.22%
Pablo Neira Ayuso10.42%111.11%
Ingo Molnar10.42%111.11%
Total237100.00%9100.00%

EXPORT_SYMBOL_GPL(xt_register_table);
void *xt_unregister_table(struct xt_table *table) { struct xt_table_info *private; mutex_lock(&xt[table->af].mutex); private = table->private; list_del(&table->list); mutex_unlock(&xt[table->af].mutex); kfree(table); return private; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte5585.94%125.00%
Alexey Dobriyan57.81%125.00%
Patrick McHardy23.12%125.00%
Ingo Molnar23.12%125.00%
Total64100.00%4100.00%

EXPORT_SYMBOL_GPL(xt_unregister_table); #ifdef CONFIG_PROC_FS struct xt_names_priv { struct seq_net_private p; u_int8_t af; };
static void *xt_table_seq_start(struct seq_file *seq, loff_t *pos) { struct xt_names_priv *priv = seq->private; struct net *net = seq_file_net(seq); u_int8_t af = priv->af; mutex_lock(&xt[af].mutex); return seq_list_start(&net->xt.tables[af], *pos); }

Contributors

PersonTokensPropCommitsCommitProp
Alexey Dobriyan3448.57%350.00%
Harald Welte3144.29%116.67%
Hideaki Yoshifuji / 吉藤英明45.71%116.67%
Jan Engelhardt11.43%116.67%
Total70100.00%6100.00%


static void *xt_table_seq_next(struct seq_file *seq, void *v, loff_t *pos) { struct xt_names_priv *priv = seq->private; struct net *net = seq_file_net(seq); u_int8_t af = priv->af; return seq_list_next(v, &net->xt.tables[af], pos); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte3859.38%120.00%
Alexey Dobriyan2132.81%240.00%
Hideaki Yoshifuji / 吉藤英明46.25%120.00%
Jan Engelhardt11.56%120.00%
Total64100.00%5100.00%


static void xt_table_seq_stop(struct seq_file *seq, void *v) { struct xt_names_priv *priv = seq->private; u_int8_t af = priv->af; mutex_unlock(&xt[af].mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte2969.05%125.00%
Alexey Dobriyan1228.57%250.00%
Jan Engelhardt12.38%125.00%
Total42100.00%4100.00%


static int xt_table_seq_show(struct seq_file *seq, void *v) { struct xt_table *table = list_entry(v, struct xt_table, list); if (*table->name) seq_printf(seq, "%s\n", table->name); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Alexey Dobriyan4894.12%133.33%
Steven Rostedt23.92%133.33%
Joe Perches11.96%133.33%
Total51100.00%3100.00%

static const struct seq_operations xt_table_seq_ops = { .start = xt_table_seq_start, .next = xt_table_seq_next, .stop = xt_table_seq_stop, .show = xt_table_seq_show, };
static int xt_table_open(struct inode *inode, struct file *file) { int ret; struct xt_names_priv *priv; ret = seq_open_net(inode, file, &xt_table_seq_ops, sizeof(struct xt_names_priv)); if (!ret) { priv = ((struct seq_file *)file->private_data)->private; priv->af = (unsigned long)PDE_DATA(inode); } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Alexey Dobriyan7998.75%266.67%
Al Viro11.25%133.33%
Total80100.00%3100.00%

static const struct file_operations xt_table_ops = { .owner = THIS_MODULE, .open = xt_table_open, .read = seq_read, .llseek = seq_lseek, .release = seq_release_net, }; /* * Traverse state for ip{,6}_{tables,matches} for helping crossing * the multi-AF mutexes. */ struct nf_mttg_trav { struct list_head *head, *curr; uint8_t class, nfproto; }; enum { MTTG_TRAV_INIT, MTTG_TRAV_NFP_UNSPEC, MTTG_TRAV_NFP_SPEC, MTTG_TRAV_DONE, };
static void *xt_mttg_seq_next(struct seq_file *seq, void *v, loff_t *ppos, bool is_target) { static const uint8_t next_class[] = { [MTTG_TRAV_NFP_UNSPEC] = MTTG_TRAV_NFP_SPEC, [MTTG_TRAV_NFP_SPEC] = MTTG_TRAV_DONE, }; struct nf_mttg_trav *trav = seq->private; switch (trav->class) { case MTTG_TRAV_INIT: trav->class = MTTG_TRAV_NFP_UNSPEC; mutex_lock(&xt[NFPROTO_UNSPEC].mutex); trav->head = trav->curr = is_target ? &xt[NFPROTO_UNSPEC].target : &xt[NFPROTO_UNSPEC].match; break; case MTTG_TRAV_NFP_UNSPEC: trav->curr = trav->curr->next; if (trav->curr != trav->head) break; mutex_unlock(&xt[NFPROTO_UNSPEC].mutex); mutex_lock(&xt[trav->nfproto].mutex); trav->head = trav->curr = is_target ? &xt[trav->nfproto].target : &xt[trav->nfproto].match; trav->class = next_class[trav->class]; break; case MTTG_TRAV_NFP_SPEC: trav->curr = trav->curr->next; if (trav->curr != trav->head) break; /* fallthru, _stop will unlock */ default: return NULL; } if (ppos != NULL) ++*ppos; return trav; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt19882.85%150.00%
Alexey Dobriyan4117.15%150.00%
Total239100.00%2100.00%


static void *xt_mttg_seq_start(struct seq_file *seq, loff_t *pos, bool is_target) { struct nf_mttg_trav *trav = seq->private; unsigned int j; trav->class = MTTG_TRAV_INIT; for (j = 0; j < *pos; ++j) if (xt_mttg_seq_next(seq, NULL, NULL, is_target) == NULL) return NULL; return trav; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt4460.27%150.00%
Alexey Dobriyan2939.73%150.00%
Total73100.00%2100.00%


static void xt_mttg_seq_stop(struct seq_file *seq, void *v) { struct nf_mttg_trav *trav = seq->private; switch (trav->class) { case MTTG_TRAV_NFP_UNSPEC: mutex_unlock(&xt[NFPROTO_UNSPEC].mutex); break; case MTTG_TRAV_NFP_SPEC: mutex_unlock(&xt[trav->nfproto].mutex); break; } }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt3351.56%150.00%
Alexey Dobriyan3148.44%150.00%
Total64100.00%2100.00%


static void *xt_match_seq_start(struct seq_file *seq, loff_t *pos) { return xt_mttg_seq_start(seq, pos, false); }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt26100.00%1100.00%
Total26100.00%1100.00%


static void *xt_match_seq_next(struct seq_file *seq, void *v, loff_t *ppos) { return xt_mttg_seq_next(seq, v, ppos, false); }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt3196.88%150.00%
Alexey Dobriyan13.12%150.00%
Total32100.00%2100.00%


static int xt_match_seq_show(struct seq_file *seq, void *v) { const struct nf_mttg_trav *trav = seq->private; const struct xt_match *match; switch (trav->class) { case MTTG_TRAV_NFP_UNSPEC: case MTTG_TRAV_NFP_SPEC: if (trav->curr == trav->head) return 0; match = list_entry(trav->curr, struct xt_match, list); if (*match->name) seq_printf(seq, "%s\n", match->name); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt4649.46%133.33%
Alexey Dobriyan4346.24%133.33%
Steven Rostedt44.30%133.33%
Total93100.00%3100.00%

static const struct seq_operations xt_match_seq_ops = { .start = xt_match_seq_start, .next = xt_match_seq_next, .stop = xt_mttg_seq_stop, .show = xt_match_seq_show, };
static int xt_match_open(struct inode *inode, struct file *file) { struct nf_mttg_trav *trav; trav = __seq_open_private(file, &xt_match_seq_ops, sizeof(*trav)); if (!trav) return -ENOMEM; trav->nfproto = (unsigned long)PDE_DATA(inode); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt3251.61%125.00%
Alexey Dobriyan2235.48%125.00%
Rob Jones711.29%125.00%
Al Viro11.61%125.00%
Total62100.00%4100.00%

static const struct file_operations xt_match_ops = { .owner = THIS_MODULE, .open = xt_match_open, .read = seq_read, .llseek = seq_lseek, .release = seq_release_private, };
static void *xt_target_seq_start(struct seq_file *seq, loff_t *pos) { return xt_mttg_seq_start(seq, pos, true); }

Contributors

PersonTokensPropCommitsCommitProp
Alexey Dobriyan2284.62%150.00%
Jan Engelhardt415.38%150.00%
Total26100.00%2100.00%


static void *xt_target_seq_next(struct seq_file *seq, void *v, loff_t *ppos) { return xt_mttg_seq_next(seq, v, ppos, true); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte1340.62%133.33%
Alexey Dobriyan1340.62%133.33%
Jan Engelhardt618.75%133.33%
Total32100.00%3100.00%


static int xt_target_seq_show(struct seq_file *seq, void *v) { const struct nf_mttg_trav *trav = seq->private; const struct xt_target *target; switch (trav->class) { case MTTG_TRAV_NFP_UNSPEC: case MTTG_TRAV_NFP_SPEC: if (trav->curr == trav->head) return 0; target = list_entry(trav->curr, struct xt_target, list); if (*target->name) seq_printf(seq, "%s\n", target->name); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt4144.09%125.00%
Alexey Dobriyan3133.33%125.00%
Harald Welte1718.28%125.00%
Steven Rostedt44.30%125.00%
Total93100.00%4100.00%

static const struct seq_operations xt_target_seq_ops = { .start = xt_target_seq_start, .next = xt_target_seq_next, .stop = xt_mttg_seq_stop, .show = xt_target_seq_show, };
static int xt_target_open(struct inode *inode, struct file *file) { struct nf_mttg_trav *trav; trav = __seq_open_private(file, &xt_target_seq_ops, sizeof(*trav)); if (!trav) return -ENOMEM; trav->nfproto = (unsigned long)PDE_DATA(inode); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt3251.61%120.00%
Harald Welte1829.03%120.00%
Rob Jones711.29%120.00%
Alexey Dobriyan46.45%120.00%
Al Viro11.61%120.00%
Total62100.00%5100.00%

static const struct file_operations xt_target_ops = { .owner = THIS_MODULE, .open = xt_target_open, .read = seq_read, .llseek = seq_lseek, .release = seq_release_private, }; #define FORMAT_TABLES "_tables_names" #define FORMAT_MATCHES "_tables_matches" #define FORMAT_TARGETS "_tables_targets" #endif /* CONFIG_PROC_FS */ /** * xt_hook_ops_alloc - set up hooks for a new table * @table: table with metadata needed to set up hooks * @fn: Hook function * * This function will create the nf_hook_ops that the x_table needs * to hand to xt_hook_link_net(). */
struct nf_hook_ops * xt_hook_ops_alloc(const struct xt_table *table, nf_hookfn *fn) { unsigned int hook_mask = table->valid_hooks; uint8_t i, num_hooks = hweight32(hook_mask); uint8_t hooknum; struct nf_hook_ops *ops; if (!num_hooks) return ERR_PTR(-EINVAL); ops = kcalloc(num_hooks, sizeof(*ops), GFP_KERNEL); if (ops == NULL) return ERR_PTR(-ENOMEM); for (i = 0, hooknum = 0; i < num_hooks && hook_mask != 0; hook_mask >>= 1, ++hooknum) { if (!(hook_mask & 1)) continue; ops[i].hook = fn; ops[i].pf = table->af; ops[i].hooknum = hooknum; ops[i].priority = table->priority; ++i; } return ops; }

Contributors

PersonTokensPropCommitsCommitProp
Jan Engelhardt15090.36%125.00%
Xiubo Li127.23%125.00%
Florian Westphal42.41%250.00%
Total166100.00%4100.00%

EXPORT_SYMBOL_GPL(xt_hook_ops_alloc);
int xt_proto_init(struct net *net, u_int8_t af) { #ifdef CONFIG_PROC_FS char buf[XT_FUNCTION_MAXNAMELEN]; struct proc_dir_entry *proc; kuid_t root_uid; kgid_t root_gid; #endif if (af >= ARRAY_SIZE(xt_prefix)) return -EINVAL; #ifdef CONFIG_PROC_FS root_uid = make_kuid(net->user_ns, 0); root_gid = make_kgid(net->user_ns, 0); strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_TABLES, sizeof(buf)); proc = proc_create_data(buf, 0440, net->proc_net, &xt_table_ops, (void *)(unsigned long)af); if (!proc) goto out; if (uid_valid(root_uid) && gid_valid(root_gid)) proc_set_user(proc, root_uid, root_gid); strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_MATCHES, sizeof(buf)); proc = proc_create_data(buf, 0440, net->proc_net, &xt_match_ops, (void *)(unsigned long)af); if (!proc) goto out_remove_tables; if (uid_valid(root_uid) && gid_valid(root_gid)) proc_set_user(proc, root_uid, root_gid); strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_TARGETS, sizeof(buf)); proc = proc_create_data(buf, 0440, net->proc_net, &xt_target_ops, (void *)(unsigned long)af); if (!proc) goto out_remove_matches; if (uid_valid(root_uid) && gid_valid(root_gid)) proc_set_user(proc, root_uid, root_gid); #endif return 0; #ifdef CONFIG_PROC_FS out_remove_matches: strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_MATCHES, sizeof(buf)); remove_proc_entry(buf, net->proc_net); out_remove_tables: strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_TABLES, sizeof(buf)); remove_proc_entry(buf, net->proc_net); out: return -1; #endif }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte24760.10%111.11%
Philip Whineray9122.14%111.11%
Denis V. Lunev4510.95%111.11%
Gao Feng102.43%111.11%
Alexey Dobriyan81.95%222.22%
Tobias Klauser51.22%111.11%
Jan Engelhardt51.22%222.22%
Total411100.00%9100.00%

EXPORT_SYMBOL_GPL(xt_proto_init);
void xt_proto_fini(struct net *net, u_int8_t af) { #ifdef CONFIG_PROC_FS char buf[XT_FUNCTION_MAXNAMELEN]; strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_TABLES, sizeof(buf)); remove_proc_entry(buf, net->proc_net); strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_TARGETS, sizeof(buf)); remove_proc_entry(buf, net->proc_net); strlcpy(buf, xt_prefix[af], sizeof(buf)); strlcat(buf, FORMAT_MATCHES, sizeof(buf)); remove_proc_entry(buf, net->proc_net); #endif /*CONFIG_PROC_FS*/ }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte10981.95%120.00%
Gao Feng1511.28%120.00%
Alexey Dobriyan53.76%120.00%
Tobias Klauser32.26%120.00%
Jan Engelhardt10.75%120.00%
Total133100.00%5100.00%

EXPORT_SYMBOL_GPL(xt_proto_fini); /** * xt_percpu_counter_alloc - allocate x_tables rule counter * * @state: pointer to xt_percpu allocation state * @counter: pointer to counter struct inside the ip(6)/arpt_entry struct * * On SMP, the packet counter [ ip(6)t_entry->counters.pcnt ] will then * contain the address of the real (percpu) counter. * * Rule evaluation needs to use xt_get_this_cpu_counter() helper * to fetch the real percpu counter. * * To speed up allocation and improve data locality, a 4kb block is * allocated. * * xt_percpu_counter_alloc_state contains the base address of the * allocated page and the current sub-offset. * * returns false on error. */
bool xt_percpu_counter_alloc(struct xt_percpu_counter_alloc_state *state, struct xt_counters *counter) { BUILD_BUG_ON(XT_PCPU_BLOCK_SIZE < (sizeof(*counter) * 2)); if (nr_cpu_ids <= 1) return true; if (!state->mem) { state->mem = __alloc_percpu(XT_PCPU_BLOCK_SIZE, XT_PCPU_BLOCK_SIZE); if (!state->mem) return false; } counter->pcnt = (__force unsigned long)(state->mem + state->off); state->off += sizeof(*counter); if (state->off > (XT_PCPU_BLOCK_SIZE - sizeof(*counter))) { state->mem = NULL; state->off = 0; } return true; }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal131100.00%2100.00%
Total131100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_percpu_counter_alloc);
void xt_percpu_counter_free(struct xt_counters *counters) { unsigned long pcnt = counters->pcnt; if (nr_cpu_ids > 1 && (pcnt & (XT_PCPU_BLOCK_SIZE - 1)) == 0) free_percpu((void __percpu *)pcnt); }

Contributors

PersonTokensPropCommitsCommitProp
Florian Westphal46100.00%2100.00%
Total46100.00%2100.00%

EXPORT_SYMBOL_GPL(xt_percpu_counter_free);
static int __net_init xt_net_init(struct net *net) { int i; for (i = 0; i < NFPROTO_NUMPROTO; i++) INIT_LIST_HEAD(&net->xt.tables[i]); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Alexey Dobriyan4397.73%150.00%
Jan Engelhardt12.27%150.00%
Total44100.00%2100.00%

static struct pernet_operations xt_net_ops = { .init = xt_net_init, };
static int __init xt_init(void) { unsigned int i; int rv; for_each_possible_cpu(i) { seqcount_init(&per_cpu(xt_recseq, i)); } xt = kmalloc(sizeof(struct xt_af) * NFPROTO_NUMPROTO, GFP_KERNEL); if (!xt) return -ENOMEM; for (i = 0; i < NFPROTO_NUMPROTO; i++) { mutex_init(&xt[i].mutex); #ifdef CONFIG_COMPAT mutex_init(&xt[i].compat_mutex); xt[i].compat_tab = NULL; #endif INIT_LIST_HEAD(&xt[i].target); INIT_LIST_HEAD(&xt[i].match); } rv = register_pernet_subsys(&xt_net_ops); if (rv < 0) kfree(xt); return rv; }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte8757.24%111.11%
Stephen Hemminger1711.18%111.11%
Alexey Dobriyan1711.18%111.11%
Dmitry Mishin1610.53%111.11%
Patrick McHardy85.26%111.11%
Eric Dumazet42.63%222.22%
Jan Engelhardt21.32%111.11%
Ingo Molnar10.66%111.11%
Total152100.00%9100.00%


static void __exit xt_fini(void) { unregister_pernet_subsys(&xt_net_ops); kfree(xt); }

Contributors

PersonTokensPropCommitsCommitProp
Harald Welte1470.00%150.00%
Alexey Dobriyan630.00%150.00%
Total20100.00%2100.00%

module_init(xt_init); module_exit(xt_fini);

Overall Contributors

PersonTokensPropCommitsCommitProp
Harald Welte200223.62%10.91%
Jan Engelhardt153318.08%2421.82%
Florian Westphal144217.01%2119.09%
Patrick McHardy141716.72%109.09%
Alexey Dobriyan7378.69%65.45%
Eric Dumazet3073.62%65.45%
Willem de Bruijn2923.44%43.64%
Dmitry Mishin2683.16%10.91%
Philip Whineray941.11%10.91%
Stephen Hemminger810.96%32.73%
Denis V. Lunev450.53%10.91%
Thomas Graf380.45%10.91%
Pablo Neira Ayuso360.42%21.82%
Gao Feng250.29%10.91%
Ingo Molnar210.25%10.91%
Rob Jones140.17%10.91%
Changli Gao140.17%10.91%
Dan Carpenter130.15%10.91%
Xiubo Li120.14%10.91%
Joe Perches100.12%32.73%
Steven Rostedt100.12%10.91%
Will Deacon100.12%10.91%
Hideaki Yoshifuji / 吉藤英明80.09%10.91%
Balazs Scheidler80.09%10.91%
Michal Hocko80.09%21.82%
Tobias Klauser80.09%10.91%
Al Viro60.07%21.82%
Eric W. Biedermann30.04%10.91%
Tejun Heo30.04%10.91%
Paul Gortmaker30.04%10.91%
Marcelo Ricardo Leitner20.02%10.91%
Geliang Tang10.01%10.91%
Philippe De Muyter10.01%10.91%
Jan Beulich10.01%10.91%
Pavel Emelyanov10.01%10.91%
Julia Lawall10.01%10.91%
Arjan van de Ven10.01%10.91%
Masahiro Yamada10.01%10.91%
Total8477100.00%110100.00%
Directory: net/netfilter
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.