Skip to content

Commit d178bc3

Browse files
hallyngregkh
authored andcommitted
user namespace: usb: make usb urbs user namespace aware (v2)
Add to the dev_state and alloc_async structures the user namespace corresponding to the uid and euid. Pass these to kill_pid_info_as_uid(), which can then implement a proper, user-namespace-aware uid check. Changelog: Sep 20: Per Oleg's suggestion: Instead of caching and passing user namespace, uid, and euid each separately, pass a struct cred. Sep 26: Address Alan Stern's comments: don't define a struct cred at usbdev_open(), and take and put a cred at async_completed() to ensure it lasts for the duration of kill_pid_info_as_cred(). Signed-off-by: Serge Hallyn <serge.hallyn@canonical.com> Cc: Oleg Nesterov <oleg@redhat.com> Cc: "Eric W. Biederman" <ebiederm@xmission.com> Cc: Tejun Heo <tj@kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
1 parent edb2b25 commit d178bc3

File tree

3 files changed

+31
-26
lines changed

3 files changed

+31
-26
lines changed

drivers/usb/core/devio.c

+13-17
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
#include <linux/cdev.h>
4747
#include <linux/notifier.h>
4848
#include <linux/security.h>
49+
#include <linux/user_namespace.h>
4950
#include <asm/uaccess.h>
5051
#include <asm/byteorder.h>
5152
#include <linux/moduleparam.h>
@@ -68,7 +69,7 @@ struct dev_state {
6869
wait_queue_head_t wait; /* wake up if a request completed */
6970
unsigned int discsignr;
7071
struct pid *disc_pid;
71-
uid_t disc_uid, disc_euid;
72+
const struct cred *cred;
7273
void __user *disccontext;
7374
unsigned long ifclaimed;
7475
u32 secid;
@@ -79,7 +80,7 @@ struct async {
7980
struct list_head asynclist;
8081
struct dev_state *ps;
8182
struct pid *pid;
82-
uid_t uid, euid;
83+
const struct cred *cred;
8384
unsigned int signr;
8485
unsigned int ifnum;
8586
void __user *userbuffer;
@@ -248,6 +249,7 @@ static struct async *alloc_async(unsigned int numisoframes)
248249
static void free_async(struct async *as)
249250
{
250251
put_pid(as->pid);
252+
put_cred(as->cred);
251253
kfree(as->urb->transfer_buffer);
252254
kfree(as->urb->setup_packet);
253255
usb_free_urb(as->urb);
@@ -393,9 +395,8 @@ static void async_completed(struct urb *urb)
393395
struct dev_state *ps = as->ps;
394396
struct siginfo sinfo;
395397
struct pid *pid = NULL;
396-
uid_t uid = 0;
397-
uid_t euid = 0;
398398
u32 secid = 0;
399+
const struct cred *cred = NULL;
399400
int signr;
400401

401402
spin_lock(&ps->lock);
@@ -408,8 +409,7 @@ static void async_completed(struct urb *urb)
408409
sinfo.si_code = SI_ASYNCIO;
409410
sinfo.si_addr = as->userurb;
410411
pid = get_pid(as->pid);
411-
uid = as->uid;
412-
euid = as->euid;
412+
cred = get_cred(as->cred);
413413
secid = as->secid;
414414
}
415415
snoop(&urb->dev->dev, "urb complete\n");
@@ -423,9 +423,9 @@ static void async_completed(struct urb *urb)
423423
spin_unlock(&ps->lock);
424424

425425
if (signr) {
426-
kill_pid_info_as_uid(sinfo.si_signo, &sinfo, pid, uid,
427-
euid, secid);
426+
kill_pid_info_as_cred(sinfo.si_signo, &sinfo, pid, cred, secid);
428427
put_pid(pid);
428+
put_cred(cred);
429429
}
430430

431431
wake_up(&ps->wait);
@@ -672,7 +672,6 @@ static int usbdev_open(struct inode *inode, struct file *file)
672672
{
673673
struct usb_device *dev = NULL;
674674
struct dev_state *ps;
675-
const struct cred *cred = current_cred();
676675
int ret;
677676

678677
ret = -ENOMEM;
@@ -722,8 +721,7 @@ static int usbdev_open(struct inode *inode, struct file *file)
722721
init_waitqueue_head(&ps->wait);
723722
ps->discsignr = 0;
724723
ps->disc_pid = get_pid(task_pid(current));
725-
ps->disc_uid = cred->uid;
726-
ps->disc_euid = cred->euid;
724+
ps->cred = get_current_cred();
727725
ps->disccontext = NULL;
728726
ps->ifclaimed = 0;
729727
security_task_getsecid(current, &ps->secid);
@@ -765,6 +763,7 @@ static int usbdev_release(struct inode *inode, struct file *file)
765763
usb_unlock_device(dev);
766764
usb_put_dev(dev);
767765
put_pid(ps->disc_pid);
766+
put_cred(ps->cred);
768767

769768
as = async_getcompleted(ps);
770769
while (as) {
@@ -1065,7 +1064,6 @@ static int proc_do_submiturb(struct dev_state *ps, struct usbdevfs_urb *uurb,
10651064
struct usb_host_endpoint *ep;
10661065
struct async *as;
10671066
struct usb_ctrlrequest *dr = NULL;
1068-
const struct cred *cred = current_cred();
10691067
unsigned int u, totlen, isofrmlen;
10701068
int ret, ifnum = -1;
10711069
int is_in;
@@ -1279,8 +1277,7 @@ static int proc_do_submiturb(struct dev_state *ps, struct usbdevfs_urb *uurb,
12791277
as->signr = uurb->signr;
12801278
as->ifnum = ifnum;
12811279
as->pid = get_pid(task_pid(current));
1282-
as->uid = cred->uid;
1283-
as->euid = cred->euid;
1280+
as->cred = get_current_cred();
12841281
security_task_getsecid(current, &as->secid);
12851282
if (!is_in && uurb->buffer_length > 0) {
12861283
if (copy_from_user(as->urb->transfer_buffer, uurb->buffer,
@@ -1998,9 +1995,8 @@ static void usbdev_remove(struct usb_device *udev)
19981995
sinfo.si_errno = EPIPE;
19991996
sinfo.si_code = SI_ASYNCIO;
20001997
sinfo.si_addr = ps->disccontext;
2001-
kill_pid_info_as_uid(ps->discsignr, &sinfo,
2002-
ps->disc_pid, ps->disc_uid,
2003-
ps->disc_euid, ps->secid);
1998+
kill_pid_info_as_cred(ps->discsignr, &sinfo,
1999+
ps->disc_pid, ps->cred, ps->secid);
20042000
}
20052001
}
20062002
}

include/linux/sched.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -2166,7 +2166,8 @@ extern int force_sigsegv(int, struct task_struct *);
21662166
extern int force_sig_info(int, struct siginfo *, struct task_struct *);
21672167
extern int __kill_pgrp_info(int sig, struct siginfo *info, struct pid *pgrp);
21682168
extern int kill_pid_info(int sig, struct siginfo *info, struct pid *pid);
2169-
extern int kill_pid_info_as_uid(int, struct siginfo *, struct pid *, uid_t, uid_t, u32);
2169+
extern int kill_pid_info_as_cred(int, struct siginfo *, struct pid *,
2170+
const struct cred *, u32);
21702171
extern int kill_pgrp(struct pid *pid, int sig, int priv);
21712172
extern int kill_pid(struct pid *pid, int sig, int priv);
21722173
extern int kill_proc_info(int, struct siginfo *, pid_t);

kernel/signal.c

+16-8
Original file line numberDiff line numberDiff line change
@@ -1344,13 +1344,24 @@ int kill_proc_info(int sig, struct siginfo *info, pid_t pid)
13441344
return error;
13451345
}
13461346

1347+
static int kill_as_cred_perm(const struct cred *cred,
1348+
struct task_struct *target)
1349+
{
1350+
const struct cred *pcred = __task_cred(target);
1351+
if (cred->user_ns != pcred->user_ns)
1352+
return 0;
1353+
if (cred->euid != pcred->suid && cred->euid != pcred->uid &&
1354+
cred->uid != pcred->suid && cred->uid != pcred->uid)
1355+
return 0;
1356+
return 1;
1357+
}
1358+
13471359
/* like kill_pid_info(), but doesn't use uid/euid of "current" */
1348-
int kill_pid_info_as_uid(int sig, struct siginfo *info, struct pid *pid,
1349-
uid_t uid, uid_t euid, u32 secid)
1360+
int kill_pid_info_as_cred(int sig, struct siginfo *info, struct pid *pid,
1361+
const struct cred *cred, u32 secid)
13501362
{
13511363
int ret = -EINVAL;
13521364
struct task_struct *p;
1353-
const struct cred *pcred;
13541365
unsigned long flags;
13551366

13561367
if (!valid_signal(sig))
@@ -1362,10 +1373,7 @@ int kill_pid_info_as_uid(int sig, struct siginfo *info, struct pid *pid,
13621373
ret = -ESRCH;
13631374
goto out_unlock;
13641375
}
1365-
pcred = __task_cred(p);
1366-
if (si_fromuser(info) &&
1367-
euid != pcred->suid && euid != pcred->uid &&
1368-
uid != pcred->suid && uid != pcred->uid) {
1376+
if (si_fromuser(info) && !kill_as_cred_perm(cred, p)) {
13691377
ret = -EPERM;
13701378
goto out_unlock;
13711379
}
@@ -1384,7 +1392,7 @@ int kill_pid_info_as_uid(int sig, struct siginfo *info, struct pid *pid,
13841392
rcu_read_unlock();
13851393
return ret;
13861394
}
1387-
EXPORT_SYMBOL_GPL(kill_pid_info_as_uid);
1395+
EXPORT_SYMBOL_GPL(kill_pid_info_as_cred);
13881396

13891397
/*
13901398
* kill_something_info() interprets pid in interesting ways just like kill(2).

0 commit comments

Comments
 (0)