blob: 4a9362f9325d60e5809db9e48ffb73b6ce2f756a [file] [log] [blame]
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -07001/*
2 * Copyright (C) 2004 IBM Corporation
3 *
4 * Author: Serge Hallyn <serue@us.ibm.com>
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation, version 2 of the
9 * License.
10 */
11
Paul Gortmaker9984de12011-05-23 14:51:41 -040012#include <linux/export.h>
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070013#include <linux/uts.h>
14#include <linux/utsname.h>
Cedric Le Goater467e9f42007-07-15 23:41:06 -070015#include <linux/err.h>
Robert P. J. Day1aeb2722008-04-29 00:59:25 -070016#include <linux/slab.h>
Serge E. Hallyn59607db2011-03-23 16:43:16 -070017#include <linux/user_namespace.h>
Eric W. Biederman34482e82010-03-07 18:43:27 -080018#include <linux/proc_fs.h>
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070019
Alexey Dobriyan4c2a7e72009-06-17 16:27:54 -070020static struct uts_namespace *create_uts_ns(void)
21{
22 struct uts_namespace *uts_ns;
23
24 uts_ns = kmalloc(sizeof(struct uts_namespace), GFP_KERNEL);
25 if (uts_ns)
26 kref_init(&uts_ns->kref);
27 return uts_ns;
28}
29
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070030/*
Serge E. Hallyn071df102006-10-02 02:18:17 -070031 * Clone a new ns copying an original utsname, setting refcount to 1
32 * @old_ns: namespace to clone
33 * Return NULL on error (failure to kmalloc), new ns otherwise
34 */
Serge E. Hallynbb96a6f2011-03-23 16:43:18 -070035static struct uts_namespace *clone_uts_ns(struct task_struct *tsk,
36 struct uts_namespace *old_ns)
Serge E. Hallyn071df102006-10-02 02:18:17 -070037{
38 struct uts_namespace *ns;
39
Alexey Dobriyan4c2a7e72009-06-17 16:27:54 -070040 ns = create_uts_ns();
Cedric Le Goater467e9f42007-07-15 23:41:06 -070041 if (!ns)
42 return ERR_PTR(-ENOMEM);
43
Alexey Dobriyanefc63c42007-09-18 22:46:27 -070044 down_read(&uts_sem);
Cedric Le Goater467e9f42007-07-15 23:41:06 -070045 memcpy(&ns->name, &old_ns->name, sizeof(ns->name));
Eric W. Biedermanc4a4d602011-11-16 23:15:31 -080046 ns->user_ns = get_user_ns(task_cred_xxx(tsk, user_ns));
Alexey Dobriyanefc63c42007-09-18 22:46:27 -070047 up_read(&uts_sem);
Serge E. Hallyn071df102006-10-02 02:18:17 -070048 return ns;
49}
50
51/*
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070052 * Copy task tsk's utsname namespace, or clone it if flags
53 * specifies CLONE_NEWUTS. In latter case, changes to the
54 * utsname of this process won't be seen by parent, and vice
55 * versa.
56 */
Serge E. Hallynbb96a6f2011-03-23 16:43:18 -070057struct uts_namespace *copy_utsname(unsigned long flags,
58 struct task_struct *tsk)
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070059{
Serge E. Hallynbb96a6f2011-03-23 16:43:18 -070060 struct uts_namespace *old_ns = tsk->nsproxy->uts_ns;
Serge E. Hallyn071df102006-10-02 02:18:17 -070061 struct uts_namespace *new_ns;
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070062
Badari Pulavartye3222c42007-05-08 00:25:21 -070063 BUG_ON(!old_ns);
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070064 get_uts_ns(old_ns);
65
Serge E. Hallyn071df102006-10-02 02:18:17 -070066 if (!(flags & CLONE_NEWUTS))
Badari Pulavartye3222c42007-05-08 00:25:21 -070067 return old_ns;
Serge E. Hallyn071df102006-10-02 02:18:17 -070068
Serge E. Hallynbb96a6f2011-03-23 16:43:18 -070069 new_ns = clone_uts_ns(tsk, old_ns);
Serge E. Hallyn071df102006-10-02 02:18:17 -070070
Serge E. Hallyn071df102006-10-02 02:18:17 -070071 put_uts_ns(old_ns);
Badari Pulavartye3222c42007-05-08 00:25:21 -070072 return new_ns;
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070073}
74
75void free_uts_ns(struct kref *kref)
76{
77 struct uts_namespace *ns;
78
79 ns = container_of(kref, struct uts_namespace, kref);
Serge E. Hallyn59607db2011-03-23 16:43:16 -070080 put_user_ns(ns->user_ns);
Serge E. Hallyn4865ecf2006-10-02 02:18:14 -070081 kfree(ns);
82}
Eric W. Biederman34482e82010-03-07 18:43:27 -080083
84static void *utsns_get(struct task_struct *task)
85{
86 struct uts_namespace *ns = NULL;
87 struct nsproxy *nsproxy;
88
89 rcu_read_lock();
90 nsproxy = task_nsproxy(task);
91 if (nsproxy) {
92 ns = nsproxy->uts_ns;
93 get_uts_ns(ns);
94 }
95 rcu_read_unlock();
96
97 return ns;
98}
99
100static void utsns_put(void *ns)
101{
102 put_uts_ns(ns);
103}
104
Eric W. Biederman142e1d12012-07-26 01:13:20 -0700105static int utsns_install(struct nsproxy *nsproxy, void *new)
Eric W. Biederman34482e82010-03-07 18:43:27 -0800106{
Eric W. Biederman142e1d12012-07-26 01:13:20 -0700107 struct uts_namespace *ns = new;
108
109 if (!ns_capable(ns->user_ns, CAP_SYS_ADMIN))
110 return -EPERM;
111
Eric W. Biederman34482e82010-03-07 18:43:27 -0800112 get_uts_ns(ns);
113 put_uts_ns(nsproxy->uts_ns);
114 nsproxy->uts_ns = ns;
115 return 0;
116}
117
118const struct proc_ns_operations utsns_operations = {
119 .name = "uts",
120 .type = CLONE_NEWUTS,
121 .get = utsns_get,
122 .put = utsns_put,
123 .install = utsns_install,
124};
125