xt_DNETMAP: support for Linux 3.10

This commit is contained in:
Jan Engelhardt
2013-06-07 15:12:21 +02:00
parent b5a2f9aa14
commit b2cd0ab65b
4 changed files with 57 additions and 27 deletions

View File

@@ -63,7 +63,7 @@ if test -n "$kbuilddir"; then
echo "WARNING: Version detection did not succeed. Continue at own luck."; echo "WARNING: Version detection did not succeed. Continue at own luck.";
else else
echo "$kmajor.$kminor.$kmicro.$kstable in $kbuilddir"; echo "$kmajor.$kminor.$kmicro.$kstable in $kbuilddir";
if test "$kmajor" -gt 3 -o "$kmajor" -eq 3 -a "$kminor" -gt 9; then if test "$kmajor" -gt 3 -o "$kmajor" -eq 3 -a "$kminor" -gt 10; then
echo "WARNING: That kernel version is not officially supported yet. Continue at own luck."; echo "WARNING: That kernel version is not officially supported yet. Continue at own luck.";
elif test "$kmajor" -eq 3 -a "$kminor" -ge 7; then elif test "$kmajor" -eq 3 -a "$kminor" -ge 7; then
:; :;

View File

@@ -1,6 +1,8 @@
HEAD HEAD
==== ====
Enhancements:
- Support for Linux 3.10
Fixes: Fixes:
- xt_DNETMAP, xt_condition, xt_quota2: resolve compile error when - xt_DNETMAP, xt_condition, xt_quota2: resolve compile error when
CONFIG_UIDGID_STRICT_TYPE_CHECKS=y CONFIG_UIDGID_STRICT_TYPE_CHECKS=y

View File

@@ -55,4 +55,31 @@
#define xt_request_find_match xtnu_request_find_match #define xt_request_find_match xtnu_request_find_match
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 9, 0)
static inline struct inode *file_inode(struct file *f)
{
return f->f_path.dentry->d_inode;
}
#endif
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 10, 0)
static inline void
proc_set_user(struct proc_dir_entry *de, kuid_t uid, kgid_t gid)
{
de->uid = uid;
de->gid = gid;
}
static inline void *PDE_DATA(struct inode *inode)
{
return PDE(inode)->data;
}
static inline void proc_remove(struct proc_dir_entry *de)
{
if (de != NULL)
remove_proc_entry(de->name, de->parent);
}
#endif
#endif /* _XTABLES_COMPAT_H */ #endif /* _XTABLES_COMPAT_H */

View File

@@ -26,6 +26,7 @@
#include <linux/netfilter.h> #include <linux/netfilter.h>
#include <linux/netfilter_ipv4.h> #include <linux/netfilter_ipv4.h>
#include <linux/netfilter/x_tables.h> #include <linux/netfilter/x_tables.h>
#include <linux/seq_file.h>
#include <linux/uidgid.h> #include <linux/uidgid.h>
#include <linux/version.h> #include <linux/version.h>
#include <net/net_namespace.h> #include <net/net_namespace.h>
@@ -113,13 +114,9 @@ static DEFINE_SPINLOCK(dnetmap_lock);
static DEFINE_MUTEX(dnetmap_mutex); static DEFINE_MUTEX(dnetmap_mutex);
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
static const struct file_operations dnetmap_tg_fops; static const struct file_operations dnetmap_tg_fops, dnetmap_stat_proc_fops;
#endif #endif
static int dnetmap_stat_proc_read(char __user *buffer, char **start,
off_t offset, int length, int *eof,
void *data);
static inline unsigned int dnetmap_entry_hash(const __be32 addr) static inline unsigned int dnetmap_entry_hash(const __be32 addr)
{ {
return ntohl(addr) & (hash_size - 1); return ntohl(addr) & (hash_size - 1);
@@ -329,21 +326,20 @@ static int dnetmap_tg_check(const struct xt_tgchk_param *par)
ret = -ENOMEM; ret = -ENOMEM;
goto out; goto out;
} }
pde_data->uid = make_kuid(&init_user_ns, proc_uid); proc_set_user(pde_data, make_kuid(&init_user_ns, proc_uid),
pde_data->gid = make_kgid(&init_user_ns, proc_gid); make_kgid(&init_user_ns, proc_gid));
/* statistics */ /* statistics */
pde_stat = create_proc_entry(p->proc_str_stat, proc_perms, pde_stat = proc_create_data(p->proc_str_stat, proc_perms,
dnetmap_net->xt_dnetmap); dnetmap_net->xt_dnetmap,
&dnetmap_stat_proc_fops, p);
if (pde_stat == NULL) { if (pde_stat == NULL) {
kfree(p); kfree(p);
ret = -ENOMEM; ret = -ENOMEM;
goto out; goto out;
} }
pde_stat->data = p; proc_set_user(pde_stat, make_kuid(&init_user_ns, proc_uid),
pde_stat->read_proc = dnetmap_stat_proc_read; make_kgid(&init_user_ns, proc_gid));
pde_stat->uid = make_kuid(&init_user_ns, proc_uid);
pde_stat->gid = make_kgid(&init_user_ns, proc_gid);
#endif #endif
spin_lock_bh(&dnetmap_lock); spin_lock_bh(&dnetmap_lock);
@@ -594,22 +590,20 @@ static const struct seq_operations dnetmap_seq_ops = {
static int dnetmap_seq_open(struct inode *inode, struct file *file) static int dnetmap_seq_open(struct inode *inode, struct file *file)
{ {
struct proc_dir_entry *pde = PDE(inode);
struct dnetmap_iter_state *st; struct dnetmap_iter_state *st;
st = __seq_open_private(file, &dnetmap_seq_ops, sizeof(*st)); st = __seq_open_private(file, &dnetmap_seq_ops, sizeof(*st));
if (st == NULL) if (st == NULL)
return -ENOMEM; return -ENOMEM;
st->p = pde->data; st->p = PDE_DATA(inode);
return 0; return 0;
} }
static ssize_t static ssize_t
dnetmap_tg_proc_write(struct file *file, const char __user *input,size_t size, loff_t *loff) dnetmap_tg_proc_write(struct file *file, const char __user *input,size_t size, loff_t *loff)
{ {
const struct proc_dir_entry *pde = PDE(file->f_path.dentry->d_inode); struct dnetmap_prefix *p = PDE_DATA(file_inode(file));
struct dnetmap_prefix *p = pde->data;
struct dnetmap_entry *e; struct dnetmap_entry *e;
char buf[sizeof("+192.168.100.100:200.200.200.200")]; char buf[sizeof("+192.168.100.100:200.200.200.200")];
const char *c = buf; const char *c = buf;
@@ -784,11 +778,9 @@ static const struct file_operations dnetmap_tg_fops = {
}; };
/* for statistics */ /* for statistics */
static int dnetmap_stat_proc_read(char __user *buffer, char **start, static int dnetmap_stat_proc_show(struct seq_file *m, void *data)
off_t offset, int length, int *eof,
void *data)
{ {
const struct dnetmap_prefix *p = data; const struct dnetmap_prefix *p = m->private;
struct dnetmap_entry *e; struct dnetmap_entry *e;
unsigned int used, used_static, all; unsigned int used, used_static, all;
long int ttl, sum_ttl; long int ttl, sum_ttl;
@@ -814,16 +806,25 @@ static int dnetmap_stat_proc_read(char __user *buffer, char **start,
} }
sum_ttl = used > 0 ? sum_ttl / (used * HZ) : 0; sum_ttl = used > 0 ? sum_ttl / (used * HZ) : 0;
sprintf(buffer, "%u %u %u %ld %s\n", used, used_static, all, sum_ttl,(p->flags & XT_DNETMAP_PERSISTENT ? "persistent" : "")); seq_printf(m, "%u %u %u %ld %s\n", used, used_static, all, sum_ttl,(p->flags & XT_DNETMAP_PERSISTENT ? "persistent" : ""));
if (length >= strlen(buffer))
*eof = true;
spin_unlock_bh(&dnetmap_lock); spin_unlock_bh(&dnetmap_lock);
return strlen(buffer); return 0;
} }
static int dnetmap_stat_proc_open(struct inode *inode, struct file *file)
{
return single_open(file, dnetmap_stat_proc_show, PDE_DATA(inode));
}
static const struct file_operations dnetmap_stat_proc_fops = {
.open = dnetmap_stat_proc_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release,
};
static int __net_init dnetmap_proc_net_init(struct net *net) static int __net_init dnetmap_proc_net_init(struct net *net)
{ {
struct dnetmap_net *dnetmap_net = dnetmap_pernet(net); struct dnetmap_net *dnetmap_net = dnetmap_pernet(net);