aboutsummaryrefslogtreecommitdiffstats
path: root/ipc
diff options
context:
space:
mode:
authorDavidlohr Bueso <dave@stgolabs.net>2018-04-10 19:35:30 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2018-04-11 13:28:37 -0400
commit23c8cec8cf679b10997a512abb1e86f0cedc42ba (patch)
tree6c4e7044aef117bf202b06210fbae71fa0229889 /ipc
parenta280d6dc77eb6002f269d58cd47c7c7e69b617b6 (diff)
ipc/msg: introduce msgctl(MSG_STAT_ANY)
There is a permission discrepancy when consulting msq ipc object metadata between /proc/sysvipc/msg (0444) and the MSG_STAT shmctl command. The later does permission checks for the object vs S_IRUGO. As such there can be cases where EACCESS is returned via syscall but the info is displayed anyways in the procfs files. While this might have security implications via info leaking (albeit no writing to the msq metadata), this behavior goes way back and showing all the objects regardless of the permissions was most likely an overlook - so we are stuck with it. Furthermore, modifying either the syscall or the procfs file can cause userspace programs to break (ie ipcs). Some applications require getting the procfs info (without root privileges) and can be rather slow in comparison with a syscall -- up to 500x in some reported cases for shm. This patch introduces a new MSG_STAT_ANY command such that the msq ipc object permissions are ignored, and only audited instead. In addition, I've left the lsm security hook checks in place, as if some policy can block the call, then the user has no other choice than just parsing the procfs file. Link: http://lkml.kernel.org/r/20180215162458.10059-4-dave@stgolabs.net Signed-off-by: Davidlohr Bueso <dbueso@suse.de> Reported-by: Robert Kettler <robert.kettler@outlook.com> Cc: Eric W. Biederman <ebiederm@xmission.com> Cc: Kees Cook <keescook@chromium.org> Cc: Manfred Spraul <manfred@colorfullife.com> Cc: Michael Kerrisk <mtk.manpages@gmail.com> Cc: Michal Hocko <mhocko@kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'ipc')
-rw-r--r--ipc/msg.c17
1 files changed, 12 insertions, 5 deletions
diff --git a/ipc/msg.c b/ipc/msg.c
index 114a21189613..56fd1c73eedc 100644
--- a/ipc/msg.c
+++ b/ipc/msg.c
@@ -497,14 +497,14 @@ static int msgctl_stat(struct ipc_namespace *ns, int msqid,
497 memset(p, 0, sizeof(*p)); 497 memset(p, 0, sizeof(*p));
498 498
499 rcu_read_lock(); 499 rcu_read_lock();
500 if (cmd == MSG_STAT) { 500 if (cmd == MSG_STAT || cmd == MSG_STAT_ANY) {
501 msq = msq_obtain_object(ns, msqid); 501 msq = msq_obtain_object(ns, msqid);
502 if (IS_ERR(msq)) { 502 if (IS_ERR(msq)) {
503 err = PTR_ERR(msq); 503 err = PTR_ERR(msq);
504 goto out_unlock; 504 goto out_unlock;
505 } 505 }
506 id = msq->q_perm.id; 506 id = msq->q_perm.id;
507 } else { 507 } else { /* IPC_STAT */
508 msq = msq_obtain_object_check(ns, msqid); 508 msq = msq_obtain_object_check(ns, msqid);
509 if (IS_ERR(msq)) { 509 if (IS_ERR(msq)) {
510 err = PTR_ERR(msq); 510 err = PTR_ERR(msq);
@@ -512,9 +512,14 @@ static int msgctl_stat(struct ipc_namespace *ns, int msqid,
512 } 512 }
513 } 513 }
514 514
515 err = -EACCES; 515 /* see comment for SHM_STAT_ANY */
516 if (ipcperms(ns, &msq->q_perm, S_IRUGO)) 516 if (cmd == MSG_STAT_ANY)
517 goto out_unlock; 517 audit_ipc_obj(&msq->q_perm);
518 else {
519 err = -EACCES;
520 if (ipcperms(ns, &msq->q_perm, S_IRUGO))
521 goto out_unlock;
522 }
518 523
519 err = security_msg_queue_msgctl(&msq->q_perm, cmd); 524 err = security_msg_queue_msgctl(&msq->q_perm, cmd);
520 if (err) 525 if (err)
@@ -572,6 +577,7 @@ long ksys_msgctl(int msqid, int cmd, struct msqid_ds __user *buf)
572 return err; 577 return err;
573 } 578 }
574 case MSG_STAT: /* msqid is an index rather than a msg queue id */ 579 case MSG_STAT: /* msqid is an index rather than a msg queue id */
580 case MSG_STAT_ANY:
575 case IPC_STAT: 581 case IPC_STAT:
576 err = msgctl_stat(ns, msqid, cmd, &msqid64); 582 err = msgctl_stat(ns, msqid, cmd, &msqid64);
577 if (err < 0) 583 if (err < 0)
@@ -690,6 +696,7 @@ long compat_ksys_msgctl(int msqid, int cmd, void __user *uptr)
690 } 696 }
691 case IPC_STAT: 697 case IPC_STAT:
692 case MSG_STAT: 698 case MSG_STAT:
699 case MSG_STAT_ANY:
693 err = msgctl_stat(ns, msqid, cmd, &msqid64); 700 err = msgctl_stat(ns, msqid, cmd, &msqid64);
694 if (err < 0) 701 if (err < 0)
695 return err; 702 return err;