mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
proc: Usable inode numbers for the namespace file descriptors.
Assign a unique proc inode to each namespace, and use that
inode number to ensure we only allocate at most one proc
inode for every namespace in proc.
A single proc inode per namespace allows userspace to test
to see if two processes are in the same namespace.
This has been a long requested feature and only blocked because
a naive implementation would put the id in a global space and
would ultimately require having a namespace for the names of
namespaces, making migration and certain virtualization tricks
impossible.
We still don't have per superblock inode numbers for proc, which
appears necessary for application unaware checkpoint/restart and
migrations (if the application is using namespace file descriptors)
but that is now allowd by the design if it becomes important.
I have preallocated the ipc and uts initial proc inode numbers so
their structures can be statically initialized.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
(cherry picked from commit 98f842e675
)
This commit is contained in:
parent
dadd033bd1
commit
60e6a983e5
18 changed files with 118 additions and 12 deletions
|
@ -4,6 +4,7 @@
|
||||||
|
|
||||||
struct mnt_namespace {
|
struct mnt_namespace {
|
||||||
atomic_t count;
|
atomic_t count;
|
||||||
|
unsigned int proc_inum;
|
||||||
struct mount * root;
|
struct mount * root;
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
struct user_namespace *user_ns;
|
struct user_namespace *user_ns;
|
||||||
|
|
|
@ -2244,6 +2244,7 @@ dput_out:
|
||||||
|
|
||||||
static void free_mnt_ns(struct mnt_namespace *ns)
|
static void free_mnt_ns(struct mnt_namespace *ns)
|
||||||
{
|
{
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
put_user_ns(ns->user_ns);
|
put_user_ns(ns->user_ns);
|
||||||
kfree(ns);
|
kfree(ns);
|
||||||
}
|
}
|
||||||
|
@ -2260,10 +2261,16 @@ static atomic64_t mnt_ns_seq = ATOMIC64_INIT(1);
|
||||||
static struct mnt_namespace *alloc_mnt_ns(struct user_namespace *user_ns)
|
static struct mnt_namespace *alloc_mnt_ns(struct user_namespace *user_ns)
|
||||||
{
|
{
|
||||||
struct mnt_namespace *new_ns;
|
struct mnt_namespace *new_ns;
|
||||||
|
int ret;
|
||||||
|
|
||||||
new_ns = kmalloc(sizeof(struct mnt_namespace), GFP_KERNEL);
|
new_ns = kmalloc(sizeof(struct mnt_namespace), GFP_KERNEL);
|
||||||
if (!new_ns)
|
if (!new_ns)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
ret = proc_alloc_inum(&new_ns->proc_inum);
|
||||||
|
if (ret) {
|
||||||
|
kfree(new_ns);
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
}
|
||||||
new_ns->seq = atomic64_add_return(1, &mnt_ns_seq);
|
new_ns->seq = atomic64_add_return(1, &mnt_ns_seq);
|
||||||
atomic_set(&new_ns->count, 1);
|
atomic_set(&new_ns->count, 1);
|
||||||
new_ns->root = NULL;
|
new_ns->root = NULL;
|
||||||
|
@ -2742,10 +2749,17 @@ static int mntns_install(struct nsproxy *nsproxy, void *ns)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int mntns_inum(void *ns)
|
||||||
|
{
|
||||||
|
struct mnt_namespace *mnt_ns = ns;
|
||||||
|
return mnt_ns->proc_inum;
|
||||||
|
}
|
||||||
|
|
||||||
const struct proc_ns_operations mntns_operations = {
|
const struct proc_ns_operations mntns_operations = {
|
||||||
.name = "mnt",
|
.name = "mnt",
|
||||||
.type = CLONE_NEWNS,
|
.type = CLONE_NEWNS,
|
||||||
.get = mntns_get,
|
.get = mntns_get,
|
||||||
.put = mntns_put,
|
.put = mntns_put,
|
||||||
.install = mntns_install,
|
.install = mntns_install,
|
||||||
|
.inum = mntns_inum,
|
||||||
};
|
};
|
||||||
|
|
|
@ -78,7 +78,7 @@ static struct dentry *proc_ns_get_dentry(struct super_block *sb,
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
inode = new_inode(sb);
|
inode = iget_locked(sb, ns_ops->inum(ns));
|
||||||
if (!inode) {
|
if (!inode) {
|
||||||
dput(dentry);
|
dput(dentry);
|
||||||
ns_ops->put(ns);
|
ns_ops->put(ns);
|
||||||
|
@ -86,13 +86,17 @@ static struct dentry *proc_ns_get_dentry(struct super_block *sb,
|
||||||
}
|
}
|
||||||
|
|
||||||
ei = PROC_I(inode);
|
ei = PROC_I(inode);
|
||||||
inode->i_ino = get_next_ino();
|
if (inode->i_state & I_NEW) {
|
||||||
inode->i_mtime = inode->i_atime = inode->i_ctime = CURRENT_TIME;
|
inode->i_mtime = inode->i_atime = inode->i_ctime = CURRENT_TIME;
|
||||||
inode->i_op = &ns_inode_operations;
|
inode->i_op = &ns_inode_operations;
|
||||||
inode->i_mode = S_IFREG | S_IRUGO;
|
inode->i_mode = S_IFREG | S_IRUGO;
|
||||||
inode->i_fop = &ns_file_operations;
|
inode->i_fop = &ns_file_operations;
|
||||||
ei->ns_ops = ns_ops;
|
ei->ns_ops = ns_ops;
|
||||||
ei->ns = ns;
|
ei->ns = ns;
|
||||||
|
unlock_new_inode(inode);
|
||||||
|
} else {
|
||||||
|
ns_ops->put(ns);
|
||||||
|
}
|
||||||
|
|
||||||
d_set_d_op(dentry, &ns_dentry_operations);
|
d_set_d_op(dentry, &ns_dentry_operations);
|
||||||
result = d_instantiate_unique(dentry, inode);
|
result = d_instantiate_unique(dentry, inode);
|
||||||
|
@ -158,12 +162,12 @@ static int proc_ns_readlink(struct dentry *dentry, char __user *buffer, int bufl
|
||||||
if (!ns)
|
if (!ns)
|
||||||
goto out_put_task;
|
goto out_put_task;
|
||||||
|
|
||||||
snprintf(name, sizeof(name), "%s", ns_ops->name);
|
snprintf(name, sizeof(name), "%s:[%u]", ns_ops->name, ns_ops->inum(ns));
|
||||||
len = strlen(name);
|
len = strlen(name);
|
||||||
|
|
||||||
if (len > buflen)
|
if (len > buflen)
|
||||||
len = buflen;
|
len = buflen;
|
||||||
if (copy_to_user(buffer, ns_ops->name, len))
|
if (copy_to_user(buffer, name, len))
|
||||||
len = -EFAULT;
|
len = -EFAULT;
|
||||||
|
|
||||||
ns_ops->put(ns);
|
ns_ops->put(ns);
|
||||||
|
|
|
@ -65,6 +65,8 @@ struct ipc_namespace {
|
||||||
|
|
||||||
/* user_ns which owns the ipc ns */
|
/* user_ns which owns the ipc ns */
|
||||||
struct user_namespace *user_ns;
|
struct user_namespace *user_ns;
|
||||||
|
|
||||||
|
unsigned int proc_inum;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct ipc_namespace init_ipc_ns;
|
extern struct ipc_namespace init_ipc_ns;
|
||||||
|
|
|
@ -34,6 +34,7 @@ struct pid_namespace {
|
||||||
gid_t pid_gid;
|
gid_t pid_gid;
|
||||||
int hide_pid;
|
int hide_pid;
|
||||||
int reboot; /* group exit code if this pidns was rebooted */
|
int reboot; /* group exit code if this pidns was rebooted */
|
||||||
|
unsigned int proc_inum;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct pid_namespace init_pid_ns;
|
extern struct pid_namespace init_pid_ns;
|
||||||
|
|
|
@ -28,7 +28,11 @@ struct mm_struct;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
PROC_ROOT_INO = 1,
|
PROC_ROOT_INO = 1,
|
||||||
|
PROC_IPC_INIT_INO = 0xEFFFFFFFU,
|
||||||
|
PROC_UTS_INIT_INO = 0xEFFFFFFEU,
|
||||||
|
PROC_USER_INIT_INO = 0xEFFFFFFDU,
|
||||||
|
PROC_PID_INIT_INO = 0xEFFFFFFCU,
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -263,6 +267,7 @@ struct proc_ns_operations {
|
||||||
void *(*get)(struct task_struct *task);
|
void *(*get)(struct task_struct *task);
|
||||||
void (*put)(void *ns);
|
void (*put)(void *ns);
|
||||||
int (*install)(struct nsproxy *nsproxy, void *ns);
|
int (*install)(struct nsproxy *nsproxy, void *ns);
|
||||||
|
unsigned int (*inum)(void *ns);
|
||||||
};
|
};
|
||||||
extern const struct proc_ns_operations netns_operations;
|
extern const struct proc_ns_operations netns_operations;
|
||||||
extern const struct proc_ns_operations utsns_operations;
|
extern const struct proc_ns_operations utsns_operations;
|
||||||
|
|
|
@ -14,6 +14,7 @@ struct user_namespace {
|
||||||
struct hlist_head uidhash_table[UIDHASH_SZ];
|
struct hlist_head uidhash_table[UIDHASH_SZ];
|
||||||
struct user_struct *creator;
|
struct user_struct *creator;
|
||||||
struct work_struct destroyer;
|
struct work_struct destroyer;
|
||||||
|
unsigned int proc_inum;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct user_namespace init_user_ns;
|
extern struct user_namespace init_user_ns;
|
||||||
|
|
|
@ -52,6 +52,7 @@ struct uts_namespace {
|
||||||
struct kref kref;
|
struct kref kref;
|
||||||
struct new_utsname name;
|
struct new_utsname name;
|
||||||
struct user_namespace *user_ns;
|
struct user_namespace *user_ns;
|
||||||
|
unsigned int proc_inum;
|
||||||
};
|
};
|
||||||
extern struct uts_namespace init_uts_ns;
|
extern struct uts_namespace init_uts_ns;
|
||||||
|
|
||||||
|
|
|
@ -52,6 +52,8 @@ struct net {
|
||||||
struct list_head cleanup_list; /* namespaces on death row */
|
struct list_head cleanup_list; /* namespaces on death row */
|
||||||
struct list_head exit_list; /* Use only net_mutex */
|
struct list_head exit_list; /* Use only net_mutex */
|
||||||
|
|
||||||
|
unsigned int proc_inum;
|
||||||
|
|
||||||
struct proc_dir_entry *proc_net;
|
struct proc_dir_entry *proc_net;
|
||||||
struct proc_dir_entry *proc_net_stat;
|
struct proc_dir_entry *proc_net_stat;
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
#include <linux/utsname.h>
|
#include <linux/utsname.h>
|
||||||
#include <generated/utsrelease.h>
|
#include <generated/utsrelease.h>
|
||||||
#include <linux/version.h>
|
#include <linux/version.h>
|
||||||
|
#include <linux/proc_fs.h>
|
||||||
|
|
||||||
#ifndef CONFIG_KALLSYMS
|
#ifndef CONFIG_KALLSYMS
|
||||||
#define version(a) Version_ ## a
|
#define version(a) Version_ ## a
|
||||||
|
@ -34,6 +35,7 @@ struct uts_namespace init_uts_ns = {
|
||||||
.domainname = UTS_DOMAINNAME,
|
.domainname = UTS_DOMAINNAME,
|
||||||
},
|
},
|
||||||
.user_ns = &init_user_ns,
|
.user_ns = &init_user_ns,
|
||||||
|
.proc_inum = PROC_UTS_INIT_INO,
|
||||||
};
|
};
|
||||||
EXPORT_SYMBOL_GPL(init_uts_ns);
|
EXPORT_SYMBOL_GPL(init_uts_ns);
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
#include <linux/msg.h>
|
#include <linux/msg.h>
|
||||||
#include <linux/ipc_namespace.h>
|
#include <linux/ipc_namespace.h>
|
||||||
#include <linux/utsname.h>
|
#include <linux/utsname.h>
|
||||||
|
#include <linux/proc_fs.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
|
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
@ -30,6 +31,7 @@ DEFINE_SPINLOCK(mq_lock);
|
||||||
struct ipc_namespace init_ipc_ns = {
|
struct ipc_namespace init_ipc_ns = {
|
||||||
.count = ATOMIC_INIT(1),
|
.count = ATOMIC_INIT(1),
|
||||||
.user_ns = &init_user_ns,
|
.user_ns = &init_user_ns,
|
||||||
|
.proc_inum = PROC_IPC_INIT_INO,
|
||||||
};
|
};
|
||||||
|
|
||||||
atomic_t nr_ipc_ns = ATOMIC_INIT(1);
|
atomic_t nr_ipc_ns = ATOMIC_INIT(1);
|
||||||
|
|
|
@ -26,9 +26,16 @@ static struct ipc_namespace *create_ipc_ns(struct task_struct *tsk,
|
||||||
if (ns == NULL)
|
if (ns == NULL)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
err = proc_alloc_inum(&ns->proc_inum);
|
||||||
|
if (err) {
|
||||||
|
kfree(ns);
|
||||||
|
return ERR_PTR(err);
|
||||||
|
}
|
||||||
|
|
||||||
atomic_set(&ns->count, 1);
|
atomic_set(&ns->count, 1);
|
||||||
err = mq_init_ns(ns);
|
err = mq_init_ns(ns);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
kfree(ns);
|
kfree(ns);
|
||||||
return ERR_PTR(err);
|
return ERR_PTR(err);
|
||||||
}
|
}
|
||||||
|
@ -113,6 +120,7 @@ static void free_ipc_ns(struct ipc_namespace *ns)
|
||||||
*/
|
*/
|
||||||
ipcns_notify(IPCNS_REMOVED);
|
ipcns_notify(IPCNS_REMOVED);
|
||||||
put_user_ns(ns->user_ns);
|
put_user_ns(ns->user_ns);
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
kfree(ns);
|
kfree(ns);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -170,10 +178,18 @@ static int ipcns_install(struct nsproxy *nsproxy, void *ns)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int ipcns_inum(void *vp)
|
||||||
|
{
|
||||||
|
struct ipc_namespace *ns = vp;
|
||||||
|
|
||||||
|
return ns->proc_inum;
|
||||||
|
}
|
||||||
|
|
||||||
const struct proc_ns_operations ipcns_operations = {
|
const struct proc_ns_operations ipcns_operations = {
|
||||||
.name = "ipc",
|
.name = "ipc",
|
||||||
.type = CLONE_NEWIPC,
|
.type = CLONE_NEWIPC,
|
||||||
.get = ipcns_get,
|
.get = ipcns_get,
|
||||||
.put = ipcns_put,
|
.put = ipcns_put,
|
||||||
.install = ipcns_install,
|
.install = ipcns_install,
|
||||||
|
.inum = ipcns_inum,
|
||||||
};
|
};
|
||||||
|
|
|
@ -78,6 +78,7 @@ struct pid_namespace init_pid_ns = {
|
||||||
.last_pid = 0,
|
.last_pid = 0,
|
||||||
.level = 0,
|
.level = 0,
|
||||||
.child_reaper = &init_task,
|
.child_reaper = &init_task,
|
||||||
|
.proc_inum = PROC_PID_INIT_INO,
|
||||||
};
|
};
|
||||||
EXPORT_SYMBOL_GPL(init_pid_ns);
|
EXPORT_SYMBOL_GPL(init_pid_ns);
|
||||||
|
|
||||||
|
|
|
@ -88,6 +88,10 @@ static struct pid_namespace *create_pid_namespace(struct pid_namespace *parent_p
|
||||||
if (ns->pid_cachep == NULL)
|
if (ns->pid_cachep == NULL)
|
||||||
goto out_free_map;
|
goto out_free_map;
|
||||||
|
|
||||||
|
err = proc_alloc_inum(&ns->proc_inum);
|
||||||
|
if (err)
|
||||||
|
goto out_free_map;
|
||||||
|
|
||||||
kref_init(&ns->kref);
|
kref_init(&ns->kref);
|
||||||
ns->level = level;
|
ns->level = level;
|
||||||
ns->parent = get_pid_ns(parent_pid_ns);
|
ns->parent = get_pid_ns(parent_pid_ns);
|
||||||
|
@ -118,6 +122,7 @@ static void destroy_pid_namespace(struct pid_namespace *ns)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
for (i = 0; i < PIDMAP_ENTRIES; i++)
|
for (i = 0; i < PIDMAP_ENTRIES; i++)
|
||||||
kfree(ns->pidmap[i].page);
|
kfree(ns->pidmap[i].page);
|
||||||
kmem_cache_free(pid_ns_cachep, ns);
|
kmem_cache_free(pid_ns_cachep, ns);
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/export.h>
|
#include <linux/export.h>
|
||||||
#include <linux/user_namespace.h>
|
#include <linux/user_namespace.h>
|
||||||
|
#include <linux/proc_fs.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* userns count is 1 for root user, 1 for init_uts_ns,
|
* userns count is 1 for root user, 1 for init_uts_ns,
|
||||||
|
@ -26,6 +27,7 @@ struct user_namespace init_user_ns = {
|
||||||
.refcount = ATOMIC_INIT(3),
|
.refcount = ATOMIC_INIT(3),
|
||||||
},
|
},
|
||||||
.creator = &root_user,
|
.creator = &root_user,
|
||||||
|
.proc_inum = PROC_USER_INIT_INO,
|
||||||
};
|
};
|
||||||
EXPORT_SYMBOL_GPL(init_user_ns);
|
EXPORT_SYMBOL_GPL(init_user_ns);
|
||||||
|
|
||||||
|
|
|
@ -27,11 +27,18 @@ int create_user_ns(struct cred *new)
|
||||||
struct user_namespace *ns;
|
struct user_namespace *ns;
|
||||||
struct user_struct *root_user;
|
struct user_struct *root_user;
|
||||||
int n;
|
int n;
|
||||||
|
int ret;
|
||||||
|
|
||||||
ns = kmem_cache_alloc(user_ns_cachep, GFP_KERNEL);
|
ns = kmem_cache_alloc(user_ns_cachep, GFP_KERNEL);
|
||||||
if (!ns)
|
if (!ns)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
ret = proc_alloc_inum(&ns->proc_inum);
|
||||||
|
if (ret) {
|
||||||
|
kmem_cache_free(user_ns_cachep, ns);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
kref_init(&ns->kref);
|
kref_init(&ns->kref);
|
||||||
|
|
||||||
for (n = 0; n < UIDHASH_SZ; ++n)
|
for (n = 0; n < UIDHASH_SZ; ++n)
|
||||||
|
@ -73,6 +80,7 @@ static void free_user_ns_work(struct work_struct *work)
|
||||||
struct user_namespace *ns =
|
struct user_namespace *ns =
|
||||||
container_of(work, struct user_namespace, destroyer);
|
container_of(work, struct user_namespace, destroyer);
|
||||||
free_uid(ns->creator);
|
free_uid(ns->creator);
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
kmem_cache_free(user_ns_cachep, ns);
|
kmem_cache_free(user_ns_cachep, ns);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -36,11 +36,18 @@ static struct uts_namespace *clone_uts_ns(struct task_struct *tsk,
|
||||||
struct uts_namespace *old_ns)
|
struct uts_namespace *old_ns)
|
||||||
{
|
{
|
||||||
struct uts_namespace *ns;
|
struct uts_namespace *ns;
|
||||||
|
int err;
|
||||||
|
|
||||||
ns = create_uts_ns();
|
ns = create_uts_ns();
|
||||||
if (!ns)
|
if (!ns)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
err = proc_alloc_inum(&ns->proc_inum);
|
||||||
|
if (err) {
|
||||||
|
kfree(ns);
|
||||||
|
return ERR_PTR(err);
|
||||||
|
}
|
||||||
|
|
||||||
down_read(&uts_sem);
|
down_read(&uts_sem);
|
||||||
memcpy(&ns->name, &old_ns->name, sizeof(ns->name));
|
memcpy(&ns->name, &old_ns->name, sizeof(ns->name));
|
||||||
ns->user_ns = get_user_ns(task_cred_xxx(tsk, user)->user_ns);
|
ns->user_ns = get_user_ns(task_cred_xxx(tsk, user)->user_ns);
|
||||||
|
@ -78,6 +85,7 @@ void free_uts_ns(struct kref *kref)
|
||||||
|
|
||||||
ns = container_of(kref, struct uts_namespace, kref);
|
ns = container_of(kref, struct uts_namespace, kref);
|
||||||
put_user_ns(ns->user_ns);
|
put_user_ns(ns->user_ns);
|
||||||
|
proc_free_inum(ns->proc_inum);
|
||||||
kfree(ns);
|
kfree(ns);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -110,11 +118,18 @@ static int utsns_install(struct nsproxy *nsproxy, void *ns)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int utsns_inum(void *vp)
|
||||||
|
{
|
||||||
|
struct uts_namespace *ns = vp;
|
||||||
|
|
||||||
|
return ns->proc_inum;
|
||||||
|
}
|
||||||
|
|
||||||
const struct proc_ns_operations utsns_operations = {
|
const struct proc_ns_operations utsns_operations = {
|
||||||
.name = "uts",
|
.name = "uts",
|
||||||
.type = CLONE_NEWUTS,
|
.type = CLONE_NEWUTS,
|
||||||
.get = utsns_get,
|
.get = utsns_get,
|
||||||
.put = utsns_put,
|
.put = utsns_put,
|
||||||
.install = utsns_install,
|
.install = utsns_install,
|
||||||
|
.inum = utsns_inum,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -376,6 +376,21 @@ struct net *get_net_ns_by_pid(pid_t pid)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(get_net_ns_by_pid);
|
EXPORT_SYMBOL_GPL(get_net_ns_by_pid);
|
||||||
|
|
||||||
|
static __net_init int net_ns_net_init(struct net *net)
|
||||||
|
{
|
||||||
|
return proc_alloc_inum(&net->proc_inum);
|
||||||
|
}
|
||||||
|
|
||||||
|
static __net_exit void net_ns_net_exit(struct net *net)
|
||||||
|
{
|
||||||
|
proc_free_inum(net->proc_inum);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct pernet_operations __net_initdata net_ns_ops = {
|
||||||
|
.init = net_ns_net_init,
|
||||||
|
.exit = net_ns_net_exit,
|
||||||
|
};
|
||||||
|
|
||||||
static int __init net_ns_init(void)
|
static int __init net_ns_init(void)
|
||||||
{
|
{
|
||||||
struct net_generic *ng;
|
struct net_generic *ng;
|
||||||
|
@ -407,6 +422,8 @@ static int __init net_ns_init(void)
|
||||||
|
|
||||||
mutex_unlock(&net_mutex);
|
mutex_unlock(&net_mutex);
|
||||||
|
|
||||||
|
register_pernet_subsys(&net_ns_ops);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -630,11 +647,18 @@ static int netns_install(struct nsproxy *nsproxy, void *ns)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int netns_inum(void *ns)
|
||||||
|
{
|
||||||
|
struct net *net = ns;
|
||||||
|
return net->proc_inum;
|
||||||
|
}
|
||||||
|
|
||||||
const struct proc_ns_operations netns_operations = {
|
const struct proc_ns_operations netns_operations = {
|
||||||
.name = "net",
|
.name = "net",
|
||||||
.type = CLONE_NEWNET,
|
.type = CLONE_NEWNET,
|
||||||
.get = netns_get,
|
.get = netns_get,
|
||||||
.put = netns_put,
|
.put = netns_put,
|
||||||
.install = netns_install,
|
.install = netns_install,
|
||||||
|
.inum = netns_inum,
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in a new issue