Linux-2.6.12-rc2
Initial git repository build. I'm not bothering with the full history, even though we have it. We can create a separate "historical" git archive of that later if we want to, and in the meantime it's about 3.2GB when imported into git - space that would just make the early git days unnecessarily complicated, when we don't have a lot of good infrastructure for it. Let it rip!
This commit is contained in:
9
ipc/Makefile
Normal file
9
ipc/Makefile
Normal file
@@ -0,0 +1,9 @@
|
||||
#
|
||||
# Makefile for the linux ipc.
|
||||
#
|
||||
|
||||
obj-$(CONFIG_SYSVIPC_COMPAT) += compat.o
|
||||
obj-$(CONFIG_SYSVIPC) += util.o msgutil.o msg.o sem.o shm.o
|
||||
obj_mq-$(CONFIG_COMPAT) += compat_mq.o
|
||||
obj-$(CONFIG_POSIX_MQUEUE) += mqueue.o msgutil.o $(obj_mq-y)
|
||||
|
687
ipc/compat.c
Normal file
687
ipc/compat.c
Normal file
@@ -0,0 +1,687 @@
|
||||
/*
|
||||
* 32 bit compatibility code for System V IPC
|
||||
*
|
||||
* Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
|
||||
* Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu)
|
||||
* Copyright (C) 1999 Arun Sharma <arun.sharma@intel.com>
|
||||
* Copyright (C) 2000 VA Linux Co
|
||||
* Copyright (C) 2000 Don Dugger <n0ano@valinux.com>
|
||||
* Copyright (C) 2000 Hewlett-Packard Co.
|
||||
* Copyright (C) 2000 David Mosberger-Tang <davidm@hpl.hp.com>
|
||||
* Copyright (C) 2000 Gerhard Tonn (ton@de.ibm.com)
|
||||
* Copyright (C) 2000-2002 Andi Kleen, SuSE Labs (x86-64 port)
|
||||
* Copyright (C) 2000 Silicon Graphics, Inc.
|
||||
* Copyright (C) 2001 IBM
|
||||
* Copyright (C) 2004 IBM Deutschland Entwicklung GmbH, IBM Corporation
|
||||
* Copyright (C) 2004 Arnd Bergmann (arnd@arndb.de)
|
||||
*
|
||||
* This code is collected from the versions for sparc64, mips64, s390x, ia64,
|
||||
* ppc64 and x86_64, all of which are based on the original sparc64 version
|
||||
* by Jakub Jelinek.
|
||||
*
|
||||
*/
|
||||
#include <linux/compat.h>
|
||||
#include <linux/config.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/highuid.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/msg.h>
|
||||
#include <linux/shm.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/syscalls.h>
|
||||
|
||||
#include <asm/semaphore.h>
|
||||
#include <asm/uaccess.h>
|
||||
|
||||
#include "util.h"
|
||||
|
||||
struct compat_msgbuf {
|
||||
compat_long_t mtype;
|
||||
char mtext[1];
|
||||
};
|
||||
|
||||
struct compat_ipc_perm {
|
||||
key_t key;
|
||||
compat_uid_t uid;
|
||||
compat_gid_t gid;
|
||||
compat_uid_t cuid;
|
||||
compat_gid_t cgid;
|
||||
compat_mode_t mode;
|
||||
unsigned short seq;
|
||||
};
|
||||
|
||||
struct compat_semid_ds {
|
||||
struct compat_ipc_perm sem_perm;
|
||||
compat_time_t sem_otime;
|
||||
compat_time_t sem_ctime;
|
||||
compat_uptr_t sem_base;
|
||||
compat_uptr_t sem_pending;
|
||||
compat_uptr_t sem_pending_last;
|
||||
compat_uptr_t undo;
|
||||
unsigned short sem_nsems;
|
||||
};
|
||||
|
||||
struct compat_msqid_ds {
|
||||
struct compat_ipc_perm msg_perm;
|
||||
compat_uptr_t msg_first;
|
||||
compat_uptr_t msg_last;
|
||||
compat_time_t msg_stime;
|
||||
compat_time_t msg_rtime;
|
||||
compat_time_t msg_ctime;
|
||||
compat_ulong_t msg_lcbytes;
|
||||
compat_ulong_t msg_lqbytes;
|
||||
unsigned short msg_cbytes;
|
||||
unsigned short msg_qnum;
|
||||
unsigned short msg_qbytes;
|
||||
compat_ipc_pid_t msg_lspid;
|
||||
compat_ipc_pid_t msg_lrpid;
|
||||
};
|
||||
|
||||
struct compat_shmid_ds {
|
||||
struct compat_ipc_perm shm_perm;
|
||||
int shm_segsz;
|
||||
compat_time_t shm_atime;
|
||||
compat_time_t shm_dtime;
|
||||
compat_time_t shm_ctime;
|
||||
compat_ipc_pid_t shm_cpid;
|
||||
compat_ipc_pid_t shm_lpid;
|
||||
unsigned short shm_nattch;
|
||||
unsigned short shm_unused;
|
||||
compat_uptr_t shm_unused2;
|
||||
compat_uptr_t shm_unused3;
|
||||
};
|
||||
|
||||
struct compat_ipc_kludge {
|
||||
compat_uptr_t msgp;
|
||||
compat_long_t msgtyp;
|
||||
};
|
||||
|
||||
struct compat_shminfo64 {
|
||||
compat_ulong_t shmmax;
|
||||
compat_ulong_t shmmin;
|
||||
compat_ulong_t shmmni;
|
||||
compat_ulong_t shmseg;
|
||||
compat_ulong_t shmall;
|
||||
compat_ulong_t __unused1;
|
||||
compat_ulong_t __unused2;
|
||||
compat_ulong_t __unused3;
|
||||
compat_ulong_t __unused4;
|
||||
};
|
||||
|
||||
struct compat_shm_info {
|
||||
compat_int_t used_ids;
|
||||
compat_ulong_t shm_tot, shm_rss, shm_swp;
|
||||
compat_ulong_t swap_attempts, swap_successes;
|
||||
};
|
||||
|
||||
extern int sem_ctls[];
|
||||
#define sc_semopm (sem_ctls[2])
|
||||
#define MAXBUF (64*1024)
|
||||
|
||||
static inline int compat_ipc_parse_version(int *cmd)
|
||||
{
|
||||
int version = *cmd & IPC_64;
|
||||
|
||||
/* this is tricky: architectures that have support for the old
|
||||
* ipc structures in 64 bit binaries need to have IPC_64 set
|
||||
* in cmd, the others need to have it cleared */
|
||||
#ifndef ipc_parse_version
|
||||
*cmd |= IPC_64;
|
||||
#else
|
||||
*cmd &= ~IPC_64;
|
||||
#endif
|
||||
return version;
|
||||
}
|
||||
|
||||
static inline int __get_compat_ipc64_perm(struct ipc64_perm *p64,
|
||||
struct compat_ipc64_perm __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = __get_user(p64->uid, &up64->uid);
|
||||
err |= __get_user(p64->gid, &up64->gid);
|
||||
err |= __get_user(p64->mode, &up64->mode);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int __get_compat_ipc_perm(struct ipc64_perm *p,
|
||||
struct compat_ipc_perm __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = __get_user(p->uid, &up->uid);
|
||||
err |= __get_user(p->gid, &up->gid);
|
||||
err |= __get_user(p->mode, &up->mode);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int __put_compat_ipc64_perm(struct ipc64_perm *p64,
|
||||
struct compat_ipc64_perm __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
err = __put_user(p64->key, &up64->key);
|
||||
err |= __put_user(p64->uid, &up64->uid);
|
||||
err |= __put_user(p64->gid, &up64->gid);
|
||||
err |= __put_user(p64->cuid, &up64->cuid);
|
||||
err |= __put_user(p64->cgid, &up64->cgid);
|
||||
err |= __put_user(p64->mode, &up64->mode);
|
||||
err |= __put_user(p64->seq, &up64->seq);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int __put_compat_ipc_perm(struct ipc64_perm *p,
|
||||
struct compat_ipc_perm __user *up)
|
||||
{
|
||||
int err;
|
||||
compat_uid_t u;
|
||||
compat_gid_t g;
|
||||
|
||||
err = __put_user(p->key, &up->key);
|
||||
SET_UID(u, p->uid);
|
||||
err |= __put_user(u, &up->uid);
|
||||
SET_GID(g, p->gid);
|
||||
err |= __put_user(g, &up->gid);
|
||||
SET_UID(u, p->cuid);
|
||||
err |= __put_user(u, &up->cuid);
|
||||
SET_GID(g, p->cgid);
|
||||
err |= __put_user(g, &up->cgid);
|
||||
err |= __put_user(p->mode, &up->mode);
|
||||
err |= __put_user(p->seq, &up->seq);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int get_compat_semid64_ds(struct semid64_ds *s64,
|
||||
struct compat_semid64_ds __user *up64)
|
||||
{
|
||||
if (!access_ok (VERIFY_READ, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
return __get_compat_ipc64_perm(&s64->sem_perm, &up64->sem_perm);
|
||||
}
|
||||
|
||||
static inline int get_compat_semid_ds(struct semid64_ds *s,
|
||||
struct compat_semid_ds __user *up)
|
||||
{
|
||||
if (!access_ok (VERIFY_READ, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
return __get_compat_ipc_perm(&s->sem_perm, &up->sem_perm);
|
||||
}
|
||||
|
||||
static inline int put_compat_semid64_ds(struct semid64_ds *s64,
|
||||
struct compat_semid64_ds __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok (VERIFY_WRITE, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
err = __put_compat_ipc64_perm(&s64->sem_perm, &up64->sem_perm);
|
||||
err |= __put_user(s64->sem_otime, &up64->sem_otime);
|
||||
err |= __put_user(s64->sem_ctime, &up64->sem_ctime);
|
||||
err |= __put_user(s64->sem_nsems, &up64->sem_nsems);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_semid_ds(struct semid64_ds *s,
|
||||
struct compat_semid_ds __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok (VERIFY_WRITE, up, sizeof(*up)))
|
||||
err = -EFAULT;
|
||||
err = __put_compat_ipc_perm(&s->sem_perm, &up->sem_perm);
|
||||
err |= __put_user(s->sem_otime, &up->sem_otime);
|
||||
err |= __put_user(s->sem_ctime, &up->sem_ctime);
|
||||
err |= __put_user(s->sem_nsems, &up->sem_nsems);
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_semctl(int first, int second, int third, void __user *uptr)
|
||||
{
|
||||
union semun fourth;
|
||||
u32 pad;
|
||||
int err, err2;
|
||||
struct semid64_ds s64;
|
||||
struct semid64_ds __user *up64;
|
||||
int version = compat_ipc_parse_version(&third);
|
||||
|
||||
if (!uptr)
|
||||
return -EINVAL;
|
||||
if (get_user(pad, (u32 __user *) uptr))
|
||||
return -EFAULT;
|
||||
if ((third & (~IPC_64)) == SETVAL)
|
||||
fourth.val = (int) pad;
|
||||
else
|
||||
fourth.__pad = compat_ptr(pad);
|
||||
switch (third & (~IPC_64)) {
|
||||
case IPC_INFO:
|
||||
case IPC_RMID:
|
||||
case SEM_INFO:
|
||||
case GETVAL:
|
||||
case GETPID:
|
||||
case GETNCNT:
|
||||
case GETZCNT:
|
||||
case GETALL:
|
||||
case SETVAL:
|
||||
case SETALL:
|
||||
err = sys_semctl(first, second, third, fourth);
|
||||
break;
|
||||
|
||||
case IPC_STAT:
|
||||
case SEM_STAT:
|
||||
up64 = compat_alloc_user_space(sizeof(s64));
|
||||
fourth.__pad = up64;
|
||||
err = sys_semctl(first, second, third, fourth);
|
||||
if (err < 0)
|
||||
break;
|
||||
if (copy_from_user(&s64, up64, sizeof(s64)))
|
||||
err2 = -EFAULT;
|
||||
else if (version == IPC_64)
|
||||
err2 = put_compat_semid64_ds(&s64, compat_ptr(pad));
|
||||
else
|
||||
err2 = put_compat_semid_ds(&s64, compat_ptr(pad));
|
||||
if (err2)
|
||||
err = -EFAULT;
|
||||
break;
|
||||
|
||||
case IPC_SET:
|
||||
if (version == IPC_64) {
|
||||
err = get_compat_semid64_ds(&s64, compat_ptr(pad));
|
||||
} else {
|
||||
err = get_compat_semid_ds(&s64, compat_ptr(pad));
|
||||
}
|
||||
up64 = compat_alloc_user_space(sizeof(s64));
|
||||
if (copy_to_user(up64, &s64, sizeof(s64)))
|
||||
err = -EFAULT;
|
||||
if (err)
|
||||
break;
|
||||
|
||||
fourth.__pad = up64;
|
||||
err = sys_semctl(first, second, third, fourth);
|
||||
break;
|
||||
|
||||
default:
|
||||
err = -EINVAL;
|
||||
break;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_msgsnd(int first, int second, int third, void __user *uptr)
|
||||
{
|
||||
struct msgbuf __user *p;
|
||||
struct compat_msgbuf __user *up = uptr;
|
||||
long type;
|
||||
|
||||
if (first < 0)
|
||||
return -EINVAL;
|
||||
if (second < 0 || (second >= MAXBUF - sizeof(struct msgbuf)))
|
||||
return -EINVAL;
|
||||
|
||||
p = compat_alloc_user_space(second + sizeof(struct msgbuf));
|
||||
if (get_user(type, &up->mtype) ||
|
||||
put_user(type, &p->mtype) ||
|
||||
copy_in_user(p->mtext, up->mtext, second))
|
||||
return -EFAULT;
|
||||
|
||||
return sys_msgsnd(first, p, second, third);
|
||||
}
|
||||
|
||||
long compat_sys_msgrcv(int first, int second, int msgtyp, int third,
|
||||
int version, void __user *uptr)
|
||||
{
|
||||
struct msgbuf __user *p;
|
||||
struct compat_msgbuf __user *up;
|
||||
long type;
|
||||
int err;
|
||||
|
||||
if (first < 0)
|
||||
return -EINVAL;
|
||||
if (second < 0 || (second >= MAXBUF - sizeof(struct msgbuf)))
|
||||
return -EINVAL;
|
||||
|
||||
if (!version) {
|
||||
struct compat_ipc_kludge ipck;
|
||||
err = -EINVAL;
|
||||
if (!uptr)
|
||||
goto out;
|
||||
err = -EFAULT;
|
||||
if (copy_from_user (&ipck, uptr, sizeof(ipck)))
|
||||
goto out;
|
||||
uptr = compat_ptr(ipck.msgp);
|
||||
msgtyp = ipck.msgtyp;
|
||||
}
|
||||
p = compat_alloc_user_space(second + sizeof(struct msgbuf));
|
||||
err = sys_msgrcv(first, p, second, msgtyp, third);
|
||||
if (err < 0)
|
||||
goto out;
|
||||
up = uptr;
|
||||
if (get_user(type, &p->mtype) ||
|
||||
put_user(type, &up->mtype) ||
|
||||
copy_in_user(up->mtext, p->mtext, err))
|
||||
err = -EFAULT;
|
||||
out:
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int get_compat_msqid64(struct msqid64_ds *m64,
|
||||
struct compat_msqid64_ds __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_READ, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
err = __get_compat_ipc64_perm(&m64->msg_perm, &up64->msg_perm);
|
||||
err |= __get_user(m64->msg_qbytes, &up64->msg_qbytes);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int get_compat_msqid(struct msqid64_ds *m,
|
||||
struct compat_msqid_ds __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_READ, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
err = __get_compat_ipc_perm(&m->msg_perm, &up->msg_perm);
|
||||
err |= __get_user(m->msg_qbytes, &up->msg_qbytes);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_msqid64_ds(struct msqid64_ds *m64,
|
||||
struct compat_msqid64_ds __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
err = __put_compat_ipc64_perm(&m64->msg_perm, &up64->msg_perm);
|
||||
err |= __put_user(m64->msg_stime, &up64->msg_stime);
|
||||
err |= __put_user(m64->msg_rtime, &up64->msg_rtime);
|
||||
err |= __put_user(m64->msg_ctime, &up64->msg_ctime);
|
||||
err |= __put_user(m64->msg_cbytes, &up64->msg_cbytes);
|
||||
err |= __put_user(m64->msg_qnum, &up64->msg_qnum);
|
||||
err |= __put_user(m64->msg_qbytes, &up64->msg_qbytes);
|
||||
err |= __put_user(m64->msg_lspid, &up64->msg_lspid);
|
||||
err |= __put_user(m64->msg_lrpid, &up64->msg_lrpid);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_msqid_ds(struct msqid64_ds *m,
|
||||
struct compat_msqid_ds __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
err = __put_compat_ipc_perm(&m->msg_perm, &up->msg_perm);
|
||||
err |= __put_user(m->msg_stime, &up->msg_stime);
|
||||
err |= __put_user(m->msg_rtime, &up->msg_rtime);
|
||||
err |= __put_user(m->msg_ctime, &up->msg_ctime);
|
||||
err |= __put_user(m->msg_cbytes, &up->msg_cbytes);
|
||||
err |= __put_user(m->msg_qnum, &up->msg_qnum);
|
||||
err |= __put_user(m->msg_qbytes, &up->msg_qbytes);
|
||||
err |= __put_user(m->msg_lspid, &up->msg_lspid);
|
||||
err |= __put_user(m->msg_lrpid, &up->msg_lrpid);
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_msgctl(int first, int second, void __user *uptr)
|
||||
{
|
||||
int err, err2;
|
||||
struct msqid64_ds m64;
|
||||
int version = compat_ipc_parse_version(&second);
|
||||
void __user *p;
|
||||
|
||||
switch (second & (~IPC_64)) {
|
||||
case IPC_INFO:
|
||||
case IPC_RMID:
|
||||
case MSG_INFO:
|
||||
err = sys_msgctl(first, second, uptr);
|
||||
break;
|
||||
|
||||
case IPC_SET:
|
||||
if (version == IPC_64) {
|
||||
err = get_compat_msqid64(&m64, uptr);
|
||||
} else {
|
||||
err = get_compat_msqid(&m64, uptr);
|
||||
}
|
||||
if (err)
|
||||
break;
|
||||
p = compat_alloc_user_space(sizeof(m64));
|
||||
if (copy_to_user(p, &m64, sizeof(m64)))
|
||||
err = -EFAULT;
|
||||
else
|
||||
err = sys_msgctl(first, second, p);
|
||||
break;
|
||||
|
||||
case IPC_STAT:
|
||||
case MSG_STAT:
|
||||
p = compat_alloc_user_space(sizeof(m64));
|
||||
err = sys_msgctl(first, second, p);
|
||||
if (err < 0)
|
||||
break;
|
||||
if (copy_from_user(&m64, p, sizeof(m64)))
|
||||
err2 = -EFAULT;
|
||||
else if (version == IPC_64)
|
||||
err2 = put_compat_msqid64_ds(&m64, uptr);
|
||||
else
|
||||
err2 = put_compat_msqid_ds(&m64, uptr);
|
||||
if (err2)
|
||||
err = -EFAULT;
|
||||
break;
|
||||
|
||||
default:
|
||||
err = -EINVAL;
|
||||
break;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_shmat(int first, int second, compat_uptr_t third, int version,
|
||||
void __user *uptr)
|
||||
{
|
||||
int err;
|
||||
unsigned long raddr;
|
||||
compat_ulong_t __user *uaddr;
|
||||
|
||||
if (version == 1)
|
||||
return -EINVAL;
|
||||
err = do_shmat(first, uptr, second, &raddr);
|
||||
if (err < 0)
|
||||
return err;
|
||||
uaddr = compat_ptr(third);
|
||||
return put_user(raddr, uaddr);
|
||||
}
|
||||
|
||||
static inline int get_compat_shmid64_ds(struct shmid64_ds *s64,
|
||||
struct compat_shmid64_ds __user *up64)
|
||||
{
|
||||
if (!access_ok(VERIFY_READ, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
return __get_compat_ipc64_perm(&s64->shm_perm, &up64->shm_perm);
|
||||
}
|
||||
|
||||
static inline int get_compat_shmid_ds(struct shmid64_ds *s,
|
||||
struct compat_shmid_ds __user *up)
|
||||
{
|
||||
if (!access_ok(VERIFY_READ, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
return __get_compat_ipc_perm(&s->shm_perm, &up->shm_perm);
|
||||
}
|
||||
|
||||
static inline int put_compat_shmid64_ds(struct shmid64_ds *s64,
|
||||
struct compat_shmid64_ds __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
err = __put_compat_ipc64_perm(&s64->shm_perm, &up64->shm_perm);
|
||||
err |= __put_user(s64->shm_atime, &up64->shm_atime);
|
||||
err |= __put_user(s64->shm_dtime, &up64->shm_dtime);
|
||||
err |= __put_user(s64->shm_ctime, &up64->shm_ctime);
|
||||
err |= __put_user(s64->shm_segsz, &up64->shm_segsz);
|
||||
err |= __put_user(s64->shm_nattch, &up64->shm_nattch);
|
||||
err |= __put_user(s64->shm_cpid, &up64->shm_cpid);
|
||||
err |= __put_user(s64->shm_lpid, &up64->shm_lpid);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_shmid_ds(struct shmid64_ds *s,
|
||||
struct compat_shmid_ds __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
err = __put_compat_ipc_perm(&s->shm_perm, &up->shm_perm);
|
||||
err |= __put_user(s->shm_atime, &up->shm_atime);
|
||||
err |= __put_user(s->shm_dtime, &up->shm_dtime);
|
||||
err |= __put_user(s->shm_ctime, &up->shm_ctime);
|
||||
err |= __put_user(s->shm_segsz, &up->shm_segsz);
|
||||
err |= __put_user(s->shm_nattch, &up->shm_nattch);
|
||||
err |= __put_user(s->shm_cpid, &up->shm_cpid);
|
||||
err |= __put_user(s->shm_lpid, &up->shm_lpid);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_shminfo64(struct shminfo64 *smi,
|
||||
struct compat_shminfo64 __user *up64)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up64, sizeof(*up64)))
|
||||
return -EFAULT;
|
||||
err = __put_user(smi->shmmax, &up64->shmmax);
|
||||
err |= __put_user(smi->shmmin, &up64->shmmin);
|
||||
err |= __put_user(smi->shmmni, &up64->shmmni);
|
||||
err |= __put_user(smi->shmseg, &up64->shmseg);
|
||||
err |= __put_user(smi->shmall, &up64->shmall);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int put_compat_shminfo(struct shminfo64 *smi,
|
||||
struct shminfo __user *up)
|
||||
{
|
||||
int err;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, up, sizeof(*up)))
|
||||
return -EFAULT;
|
||||
err = __put_user(smi->shmmax, &up->shmmax);
|
||||
err |= __put_user(smi->shmmin, &up->shmmin);
|
||||
err |= __put_user(smi->shmmni, &up->shmmni);
|
||||
err |= __put_user(smi->shmseg, &up->shmseg);
|
||||
err |= __put_user(smi->shmall, &up->shmall);
|
||||
}
|
||||
|
||||
static inline int put_compat_shm_info(struct shm_info __user *ip,
|
||||
struct compat_shm_info __user *uip)
|
||||
{
|
||||
int err;
|
||||
struct shm_info si;
|
||||
|
||||
if (!access_ok(VERIFY_WRITE, uip, sizeof(*uip)) ||
|
||||
copy_from_user(&si, ip, sizeof(si)))
|
||||
return -EFAULT;
|
||||
err = __put_user(si.used_ids, &uip->used_ids);
|
||||
err |= __put_user(si.shm_tot, &uip->shm_tot);
|
||||
err |= __put_user(si.shm_rss, &uip->shm_rss);
|
||||
err |= __put_user(si.shm_swp, &uip->shm_swp);
|
||||
err |= __put_user(si.swap_attempts, &uip->swap_attempts);
|
||||
err |= __put_user(si.swap_successes, &uip->swap_successes);
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_shmctl(int first, int second, void __user *uptr)
|
||||
{
|
||||
void __user *p;
|
||||
struct shmid64_ds s64;
|
||||
struct shminfo64 smi;
|
||||
int err, err2;
|
||||
int version = compat_ipc_parse_version(&second);
|
||||
|
||||
switch (second & (~IPC_64)) {
|
||||
case IPC_RMID:
|
||||
case SHM_LOCK:
|
||||
case SHM_UNLOCK:
|
||||
err = sys_shmctl(first, second, uptr);
|
||||
break;
|
||||
|
||||
case IPC_INFO:
|
||||
p = compat_alloc_user_space(sizeof(smi));
|
||||
err = sys_shmctl(first, second, p);
|
||||
if (err < 0)
|
||||
break;
|
||||
if (copy_from_user(&smi, p, sizeof(smi)))
|
||||
err2 = -EFAULT;
|
||||
else if (version == IPC_64)
|
||||
err2 = put_compat_shminfo64(&smi, uptr);
|
||||
else
|
||||
err2 = put_compat_shminfo(&smi, uptr);
|
||||
if (err2)
|
||||
err = -EFAULT;
|
||||
break;
|
||||
|
||||
|
||||
case IPC_SET:
|
||||
if (version == IPC_64) {
|
||||
err = get_compat_shmid64_ds(&s64, uptr);
|
||||
} else {
|
||||
err = get_compat_shmid_ds(&s64, uptr);
|
||||
}
|
||||
if (err)
|
||||
break;
|
||||
p = compat_alloc_user_space(sizeof(s64));
|
||||
if (copy_to_user(p, &s64, sizeof(s64)))
|
||||
err = -EFAULT;
|
||||
else
|
||||
err = sys_shmctl(first, second, p);
|
||||
break;
|
||||
|
||||
case IPC_STAT:
|
||||
case SHM_STAT:
|
||||
p = compat_alloc_user_space(sizeof(s64));
|
||||
err = sys_shmctl(first, second, p);
|
||||
if (err < 0)
|
||||
break;
|
||||
if (copy_from_user(&s64, p, sizeof(s64)))
|
||||
err2 = -EFAULT;
|
||||
else if (version == IPC_64)
|
||||
err2 = put_compat_shmid64_ds(&s64, uptr);
|
||||
else
|
||||
err2 = put_compat_shmid_ds(&s64, uptr);
|
||||
if (err2)
|
||||
err = -EFAULT;
|
||||
break;
|
||||
|
||||
case SHM_INFO:
|
||||
p = compat_alloc_user_space(sizeof(struct shm_info));
|
||||
err = sys_shmctl(first, second, p);
|
||||
if (err < 0)
|
||||
break;
|
||||
err2 = put_compat_shm_info(p, uptr);
|
||||
if (err2)
|
||||
err = -EFAULT;
|
||||
break;
|
||||
|
||||
default:
|
||||
err = -EINVAL;
|
||||
break;
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
long compat_sys_semtimedop(int semid, struct sembuf __user *tsems,
|
||||
unsigned nsops, const struct compat_timespec __user *timeout)
|
||||
{
|
||||
struct timespec __user *ts64 = NULL;
|
||||
if (timeout) {
|
||||
struct timespec ts;
|
||||
ts64 = compat_alloc_user_space(sizeof(*ts64));
|
||||
if (get_compat_timespec(&ts, timeout))
|
||||
return -EFAULT;
|
||||
if (copy_to_user(ts64, &ts, sizeof(ts)))
|
||||
return -EFAULT;
|
||||
}
|
||||
return sys_semtimedop(semid, tsems, nsops, ts64);
|
||||
}
|
146
ipc/compat_mq.c
Normal file
146
ipc/compat_mq.c
Normal file
@@ -0,0 +1,146 @@
|
||||
/*
|
||||
* ipc/compat_mq.c
|
||||
* 32 bit emulation for POSIX message queue system calls
|
||||
*
|
||||
* Copyright (C) 2004 IBM Deutschland Entwicklung GmbH, IBM Corporation
|
||||
* Author: Arnd Bergmann <arnd@arndb.de>
|
||||
*/
|
||||
|
||||
#include <linux/compat.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/mqueue.h>
|
||||
#include <linux/syscalls.h>
|
||||
|
||||
#include <asm/uaccess.h>
|
||||
|
||||
struct compat_mq_attr {
|
||||
compat_long_t mq_flags; /* message queue flags */
|
||||
compat_long_t mq_maxmsg; /* maximum number of messages */
|
||||
compat_long_t mq_msgsize; /* maximum message size */
|
||||
compat_long_t mq_curmsgs; /* number of messages currently queued */
|
||||
compat_long_t __reserved[4]; /* ignored for input, zeroed for output */
|
||||
};
|
||||
|
||||
static inline int get_compat_mq_attr(struct mq_attr *attr,
|
||||
const struct compat_mq_attr __user *uattr)
|
||||
{
|
||||
if (!access_ok(VERIFY_READ, uattr, sizeof *uattr))
|
||||
return -EFAULT;
|
||||
|
||||
return __get_user(attr->mq_flags, &uattr->mq_flags)
|
||||
| __get_user(attr->mq_maxmsg, &uattr->mq_maxmsg)
|
||||
| __get_user(attr->mq_msgsize, &uattr->mq_msgsize)
|
||||
| __get_user(attr->mq_curmsgs, &uattr->mq_curmsgs);
|
||||
}
|
||||
|
||||
static inline int put_compat_mq_attr(const struct mq_attr *attr,
|
||||
struct compat_mq_attr __user *uattr)
|
||||
{
|
||||
if (clear_user(uattr, sizeof *uattr))
|
||||
return -EFAULT;
|
||||
|
||||
return __put_user(attr->mq_flags, &uattr->mq_flags)
|
||||
| __put_user(attr->mq_maxmsg, &uattr->mq_maxmsg)
|
||||
| __put_user(attr->mq_msgsize, &uattr->mq_msgsize)
|
||||
| __put_user(attr->mq_curmsgs, &uattr->mq_curmsgs);
|
||||
}
|
||||
|
||||
asmlinkage long compat_sys_mq_open(const char __user *u_name,
|
||||
int oflag, compat_mode_t mode,
|
||||
struct compat_mq_attr __user *u_attr)
|
||||
{
|
||||
void __user *p = NULL;
|
||||
if (u_attr && oflag & O_CREAT) {
|
||||
struct mq_attr attr;
|
||||
p = compat_alloc_user_space(sizeof(attr));
|
||||
if (get_compat_mq_attr(&attr, u_attr) ||
|
||||
copy_to_user(p, &attr, sizeof(attr)))
|
||||
return -EFAULT;
|
||||
}
|
||||
return sys_mq_open(u_name, oflag, mode, p);
|
||||
}
|
||||
|
||||
static int compat_prepare_timeout(struct timespec __user * *p,
|
||||
const struct compat_timespec __user *u)
|
||||
{
|
||||
struct timespec ts;
|
||||
if (!u) {
|
||||
*p = NULL;
|
||||
return 0;
|
||||
}
|
||||
*p = compat_alloc_user_space(sizeof(ts));
|
||||
if (get_compat_timespec(&ts, u) || copy_to_user(*p, &ts, sizeof(ts)))
|
||||
return -EFAULT;
|
||||
return 0;
|
||||
}
|
||||
|
||||
asmlinkage long compat_sys_mq_timedsend(mqd_t mqdes,
|
||||
const char __user *u_msg_ptr,
|
||||
size_t msg_len, unsigned int msg_prio,
|
||||
const struct compat_timespec __user *u_abs_timeout)
|
||||
{
|
||||
struct timespec __user *u_ts;
|
||||
|
||||
if (compat_prepare_timeout(&u_ts, u_abs_timeout))
|
||||
return -EFAULT;
|
||||
|
||||
return sys_mq_timedsend(mqdes, u_msg_ptr, msg_len,
|
||||
msg_prio, u_ts);
|
||||
}
|
||||
|
||||
asmlinkage ssize_t compat_sys_mq_timedreceive(mqd_t mqdes,
|
||||
char __user *u_msg_ptr,
|
||||
size_t msg_len, unsigned int __user *u_msg_prio,
|
||||
const struct compat_timespec __user *u_abs_timeout)
|
||||
{
|
||||
struct timespec __user *u_ts;
|
||||
if (compat_prepare_timeout(&u_ts, u_abs_timeout))
|
||||
return -EFAULT;
|
||||
|
||||
return sys_mq_timedreceive(mqdes, u_msg_ptr, msg_len,
|
||||
u_msg_prio, u_ts);
|
||||
}
|
||||
|
||||
asmlinkage long compat_sys_mq_notify(mqd_t mqdes,
|
||||
const struct compat_sigevent __user *u_notification)
|
||||
{
|
||||
struct sigevent __user *p = NULL;
|
||||
if (u_notification) {
|
||||
struct sigevent n;
|
||||
p = compat_alloc_user_space(sizeof(*p));
|
||||
if (get_compat_sigevent(&n, u_notification))
|
||||
return -EFAULT;
|
||||
if (n.sigev_notify == SIGEV_THREAD)
|
||||
n.sigev_value.sival_ptr = compat_ptr(n.sigev_value.sival_int);
|
||||
if (copy_to_user(p, &n, sizeof(*p)))
|
||||
return -EFAULT;
|
||||
}
|
||||
return sys_mq_notify(mqdes, p);
|
||||
}
|
||||
|
||||
asmlinkage long compat_sys_mq_getsetattr(mqd_t mqdes,
|
||||
const struct compat_mq_attr __user *u_mqstat,
|
||||
struct compat_mq_attr __user *u_omqstat)
|
||||
{
|
||||
struct mq_attr mqstat;
|
||||
struct mq_attr __user *p = compat_alloc_user_space(2 * sizeof(*p));
|
||||
long ret;
|
||||
|
||||
if (u_mqstat) {
|
||||
if (get_compat_mq_attr(&mqstat, u_mqstat) ||
|
||||
copy_to_user(p, &mqstat, sizeof(mqstat)))
|
||||
return -EFAULT;
|
||||
}
|
||||
ret = sys_mq_getsetattr(mqdes,
|
||||
u_mqstat ? p : NULL,
|
||||
u_omqstat ? p + 1 : NULL);
|
||||
if (ret)
|
||||
return ret;
|
||||
if (u_omqstat) {
|
||||
if (copy_from_user(&mqstat, p + 1, sizeof(mqstat)) ||
|
||||
put_compat_mq_attr(&mqstat, u_omqstat))
|
||||
return -EFAULT;
|
||||
}
|
||||
return 0;
|
||||
}
|
1252
ipc/mqueue.c
Normal file
1252
ipc/mqueue.c
Normal file
File diff suppressed because it is too large
Load Diff
862
ipc/msg.c
Normal file
862
ipc/msg.c
Normal file
@@ -0,0 +1,862 @@
|
||||
/*
|
||||
* linux/ipc/msg.c
|
||||
* Copyright (C) 1992 Krishna Balasubramanian
|
||||
*
|
||||
* Removed all the remaining kerneld mess
|
||||
* Catch the -EFAULT stuff properly
|
||||
* Use GFP_KERNEL for messages as in 1.2
|
||||
* Fixed up the unchecked user space derefs
|
||||
* Copyright (C) 1998 Alan Cox & Andi Kleen
|
||||
*
|
||||
* /proc/sysvipc/msg support (c) 1999 Dragos Acostachioaie <dragos@iname.com>
|
||||
*
|
||||
* mostly rewritten, threaded and wake-one semantics added
|
||||
* MSGMAX limit removed, sysctl's added
|
||||
* (c) 1999 Manfred Spraul <manfreds@colorfullife.com>
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/msg.h>
|
||||
#include <linux/spinlock.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/proc_fs.h>
|
||||
#include <linux/list.h>
|
||||
#include <linux/security.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <linux/audit.h>
|
||||
#include <asm/current.h>
|
||||
#include <asm/uaccess.h>
|
||||
#include "util.h"
|
||||
|
||||
/* sysctl: */
|
||||
int msg_ctlmax = MSGMAX;
|
||||
int msg_ctlmnb = MSGMNB;
|
||||
int msg_ctlmni = MSGMNI;
|
||||
|
||||
/* one msg_receiver structure for each sleeping receiver */
|
||||
struct msg_receiver {
|
||||
struct list_head r_list;
|
||||
struct task_struct* r_tsk;
|
||||
|
||||
int r_mode;
|
||||
long r_msgtype;
|
||||
long r_maxsize;
|
||||
|
||||
struct msg_msg* volatile r_msg;
|
||||
};
|
||||
|
||||
/* one msg_sender for each sleeping sender */
|
||||
struct msg_sender {
|
||||
struct list_head list;
|
||||
struct task_struct* tsk;
|
||||
};
|
||||
|
||||
#define SEARCH_ANY 1
|
||||
#define SEARCH_EQUAL 2
|
||||
#define SEARCH_NOTEQUAL 3
|
||||
#define SEARCH_LESSEQUAL 4
|
||||
|
||||
static atomic_t msg_bytes = ATOMIC_INIT(0);
|
||||
static atomic_t msg_hdrs = ATOMIC_INIT(0);
|
||||
|
||||
static struct ipc_ids msg_ids;
|
||||
|
||||
#define msg_lock(id) ((struct msg_queue*)ipc_lock(&msg_ids,id))
|
||||
#define msg_unlock(msq) ipc_unlock(&(msq)->q_perm)
|
||||
#define msg_rmid(id) ((struct msg_queue*)ipc_rmid(&msg_ids,id))
|
||||
#define msg_checkid(msq, msgid) \
|
||||
ipc_checkid(&msg_ids,&msq->q_perm,msgid)
|
||||
#define msg_buildid(id, seq) \
|
||||
ipc_buildid(&msg_ids, id, seq)
|
||||
|
||||
static void freeque (struct msg_queue *msq, int id);
|
||||
static int newque (key_t key, int msgflg);
|
||||
#ifdef CONFIG_PROC_FS
|
||||
static int sysvipc_msg_read_proc(char *buffer, char **start, off_t offset, int length, int *eof, void *data);
|
||||
#endif
|
||||
|
||||
void __init msg_init (void)
|
||||
{
|
||||
ipc_init_ids(&msg_ids,msg_ctlmni);
|
||||
|
||||
#ifdef CONFIG_PROC_FS
|
||||
create_proc_read_entry("sysvipc/msg", 0, NULL, sysvipc_msg_read_proc, NULL);
|
||||
#endif
|
||||
}
|
||||
|
||||
static int newque (key_t key, int msgflg)
|
||||
{
|
||||
int id;
|
||||
int retval;
|
||||
struct msg_queue *msq;
|
||||
|
||||
msq = ipc_rcu_alloc(sizeof(*msq));
|
||||
if (!msq)
|
||||
return -ENOMEM;
|
||||
|
||||
msq->q_perm.mode = (msgflg & S_IRWXUGO);
|
||||
msq->q_perm.key = key;
|
||||
|
||||
msq->q_perm.security = NULL;
|
||||
retval = security_msg_queue_alloc(msq);
|
||||
if (retval) {
|
||||
ipc_rcu_putref(msq);
|
||||
return retval;
|
||||
}
|
||||
|
||||
id = ipc_addid(&msg_ids, &msq->q_perm, msg_ctlmni);
|
||||
if(id == -1) {
|
||||
security_msg_queue_free(msq);
|
||||
ipc_rcu_putref(msq);
|
||||
return -ENOSPC;
|
||||
}
|
||||
|
||||
msq->q_stime = msq->q_rtime = 0;
|
||||
msq->q_ctime = get_seconds();
|
||||
msq->q_cbytes = msq->q_qnum = 0;
|
||||
msq->q_qbytes = msg_ctlmnb;
|
||||
msq->q_lspid = msq->q_lrpid = 0;
|
||||
INIT_LIST_HEAD(&msq->q_messages);
|
||||
INIT_LIST_HEAD(&msq->q_receivers);
|
||||
INIT_LIST_HEAD(&msq->q_senders);
|
||||
msg_unlock(msq);
|
||||
|
||||
return msg_buildid(id,msq->q_perm.seq);
|
||||
}
|
||||
|
||||
static inline void ss_add(struct msg_queue* msq, struct msg_sender* mss)
|
||||
{
|
||||
mss->tsk=current;
|
||||
current->state=TASK_INTERRUPTIBLE;
|
||||
list_add_tail(&mss->list,&msq->q_senders);
|
||||
}
|
||||
|
||||
static inline void ss_del(struct msg_sender* mss)
|
||||
{
|
||||
if(mss->list.next != NULL)
|
||||
list_del(&mss->list);
|
||||
}
|
||||
|
||||
static void ss_wakeup(struct list_head* h, int kill)
|
||||
{
|
||||
struct list_head *tmp;
|
||||
|
||||
tmp = h->next;
|
||||
while (tmp != h) {
|
||||
struct msg_sender* mss;
|
||||
|
||||
mss = list_entry(tmp,struct msg_sender,list);
|
||||
tmp = tmp->next;
|
||||
if(kill)
|
||||
mss->list.next=NULL;
|
||||
wake_up_process(mss->tsk);
|
||||
}
|
||||
}
|
||||
|
||||
static void expunge_all(struct msg_queue* msq, int res)
|
||||
{
|
||||
struct list_head *tmp;
|
||||
|
||||
tmp = msq->q_receivers.next;
|
||||
while (tmp != &msq->q_receivers) {
|
||||
struct msg_receiver* msr;
|
||||
|
||||
msr = list_entry(tmp,struct msg_receiver,r_list);
|
||||
tmp = tmp->next;
|
||||
msr->r_msg = NULL;
|
||||
wake_up_process(msr->r_tsk);
|
||||
smp_mb();
|
||||
msr->r_msg = ERR_PTR(res);
|
||||
}
|
||||
}
|
||||
/*
|
||||
* freeque() wakes up waiters on the sender and receiver waiting queue,
|
||||
* removes the message queue from message queue ID
|
||||
* array, and cleans up all the messages associated with this queue.
|
||||
*
|
||||
* msg_ids.sem and the spinlock for this message queue is hold
|
||||
* before freeque() is called. msg_ids.sem remains locked on exit.
|
||||
*/
|
||||
static void freeque (struct msg_queue *msq, int id)
|
||||
{
|
||||
struct list_head *tmp;
|
||||
|
||||
expunge_all(msq,-EIDRM);
|
||||
ss_wakeup(&msq->q_senders,1);
|
||||
msq = msg_rmid(id);
|
||||
msg_unlock(msq);
|
||||
|
||||
tmp = msq->q_messages.next;
|
||||
while(tmp != &msq->q_messages) {
|
||||
struct msg_msg* msg = list_entry(tmp,struct msg_msg,m_list);
|
||||
tmp = tmp->next;
|
||||
atomic_dec(&msg_hdrs);
|
||||
free_msg(msg);
|
||||
}
|
||||
atomic_sub(msq->q_cbytes, &msg_bytes);
|
||||
security_msg_queue_free(msq);
|
||||
ipc_rcu_putref(msq);
|
||||
}
|
||||
|
||||
asmlinkage long sys_msgget (key_t key, int msgflg)
|
||||
{
|
||||
int id, ret = -EPERM;
|
||||
struct msg_queue *msq;
|
||||
|
||||
down(&msg_ids.sem);
|
||||
if (key == IPC_PRIVATE)
|
||||
ret = newque(key, msgflg);
|
||||
else if ((id = ipc_findkey(&msg_ids, key)) == -1) { /* key not used */
|
||||
if (!(msgflg & IPC_CREAT))
|
||||
ret = -ENOENT;
|
||||
else
|
||||
ret = newque(key, msgflg);
|
||||
} else if (msgflg & IPC_CREAT && msgflg & IPC_EXCL) {
|
||||
ret = -EEXIST;
|
||||
} else {
|
||||
msq = msg_lock(id);
|
||||
if(msq==NULL)
|
||||
BUG();
|
||||
if (ipcperms(&msq->q_perm, msgflg))
|
||||
ret = -EACCES;
|
||||
else {
|
||||
int qid = msg_buildid(id, msq->q_perm.seq);
|
||||
ret = security_msg_queue_associate(msq, msgflg);
|
||||
if (!ret)
|
||||
ret = qid;
|
||||
}
|
||||
msg_unlock(msq);
|
||||
}
|
||||
up(&msg_ids.sem);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static inline unsigned long copy_msqid_to_user(void __user *buf, struct msqid64_ds *in, int version)
|
||||
{
|
||||
switch(version) {
|
||||
case IPC_64:
|
||||
return copy_to_user (buf, in, sizeof(*in));
|
||||
case IPC_OLD:
|
||||
{
|
||||
struct msqid_ds out;
|
||||
|
||||
memset(&out,0,sizeof(out));
|
||||
|
||||
ipc64_perm_to_ipc_perm(&in->msg_perm, &out.msg_perm);
|
||||
|
||||
out.msg_stime = in->msg_stime;
|
||||
out.msg_rtime = in->msg_rtime;
|
||||
out.msg_ctime = in->msg_ctime;
|
||||
|
||||
if(in->msg_cbytes > USHRT_MAX)
|
||||
out.msg_cbytes = USHRT_MAX;
|
||||
else
|
||||
out.msg_cbytes = in->msg_cbytes;
|
||||
out.msg_lcbytes = in->msg_cbytes;
|
||||
|
||||
if(in->msg_qnum > USHRT_MAX)
|
||||
out.msg_qnum = USHRT_MAX;
|
||||
else
|
||||
out.msg_qnum = in->msg_qnum;
|
||||
|
||||
if(in->msg_qbytes > USHRT_MAX)
|
||||
out.msg_qbytes = USHRT_MAX;
|
||||
else
|
||||
out.msg_qbytes = in->msg_qbytes;
|
||||
out.msg_lqbytes = in->msg_qbytes;
|
||||
|
||||
out.msg_lspid = in->msg_lspid;
|
||||
out.msg_lrpid = in->msg_lrpid;
|
||||
|
||||
return copy_to_user (buf, &out, sizeof(out));
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
struct msq_setbuf {
|
||||
unsigned long qbytes;
|
||||
uid_t uid;
|
||||
gid_t gid;
|
||||
mode_t mode;
|
||||
};
|
||||
|
||||
static inline unsigned long copy_msqid_from_user(struct msq_setbuf *out, void __user *buf, int version)
|
||||
{
|
||||
switch(version) {
|
||||
case IPC_64:
|
||||
{
|
||||
struct msqid64_ds tbuf;
|
||||
|
||||
if (copy_from_user (&tbuf, buf, sizeof (tbuf)))
|
||||
return -EFAULT;
|
||||
|
||||
out->qbytes = tbuf.msg_qbytes;
|
||||
out->uid = tbuf.msg_perm.uid;
|
||||
out->gid = tbuf.msg_perm.gid;
|
||||
out->mode = tbuf.msg_perm.mode;
|
||||
|
||||
return 0;
|
||||
}
|
||||
case IPC_OLD:
|
||||
{
|
||||
struct msqid_ds tbuf_old;
|
||||
|
||||
if (copy_from_user (&tbuf_old, buf, sizeof (tbuf_old)))
|
||||
return -EFAULT;
|
||||
|
||||
out->uid = tbuf_old.msg_perm.uid;
|
||||
out->gid = tbuf_old.msg_perm.gid;
|
||||
out->mode = tbuf_old.msg_perm.mode;
|
||||
|
||||
if(tbuf_old.msg_qbytes == 0)
|
||||
out->qbytes = tbuf_old.msg_lqbytes;
|
||||
else
|
||||
out->qbytes = tbuf_old.msg_qbytes;
|
||||
|
||||
return 0;
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
asmlinkage long sys_msgctl (int msqid, int cmd, struct msqid_ds __user *buf)
|
||||
{
|
||||
int err, version;
|
||||
struct msg_queue *msq;
|
||||
struct msq_setbuf setbuf;
|
||||
struct kern_ipc_perm *ipcp;
|
||||
|
||||
if (msqid < 0 || cmd < 0)
|
||||
return -EINVAL;
|
||||
|
||||
version = ipc_parse_version(&cmd);
|
||||
|
||||
switch (cmd) {
|
||||
case IPC_INFO:
|
||||
case MSG_INFO:
|
||||
{
|
||||
struct msginfo msginfo;
|
||||
int max_id;
|
||||
if (!buf)
|
||||
return -EFAULT;
|
||||
/* We must not return kernel stack data.
|
||||
* due to padding, it's not enough
|
||||
* to set all member fields.
|
||||
*/
|
||||
|
||||
err = security_msg_queue_msgctl(NULL, cmd);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
memset(&msginfo,0,sizeof(msginfo));
|
||||
msginfo.msgmni = msg_ctlmni;
|
||||
msginfo.msgmax = msg_ctlmax;
|
||||
msginfo.msgmnb = msg_ctlmnb;
|
||||
msginfo.msgssz = MSGSSZ;
|
||||
msginfo.msgseg = MSGSEG;
|
||||
down(&msg_ids.sem);
|
||||
if (cmd == MSG_INFO) {
|
||||
msginfo.msgpool = msg_ids.in_use;
|
||||
msginfo.msgmap = atomic_read(&msg_hdrs);
|
||||
msginfo.msgtql = atomic_read(&msg_bytes);
|
||||
} else {
|
||||
msginfo.msgmap = MSGMAP;
|
||||
msginfo.msgpool = MSGPOOL;
|
||||
msginfo.msgtql = MSGTQL;
|
||||
}
|
||||
max_id = msg_ids.max_id;
|
||||
up(&msg_ids.sem);
|
||||
if (copy_to_user (buf, &msginfo, sizeof(struct msginfo)))
|
||||
return -EFAULT;
|
||||
return (max_id < 0) ? 0: max_id;
|
||||
}
|
||||
case MSG_STAT:
|
||||
case IPC_STAT:
|
||||
{
|
||||
struct msqid64_ds tbuf;
|
||||
int success_return;
|
||||
if (!buf)
|
||||
return -EFAULT;
|
||||
if(cmd == MSG_STAT && msqid >= msg_ids.entries->size)
|
||||
return -EINVAL;
|
||||
|
||||
memset(&tbuf,0,sizeof(tbuf));
|
||||
|
||||
msq = msg_lock(msqid);
|
||||
if (msq == NULL)
|
||||
return -EINVAL;
|
||||
|
||||
if(cmd == MSG_STAT) {
|
||||
success_return = msg_buildid(msqid, msq->q_perm.seq);
|
||||
} else {
|
||||
err = -EIDRM;
|
||||
if (msg_checkid(msq,msqid))
|
||||
goto out_unlock;
|
||||
success_return = 0;
|
||||
}
|
||||
err = -EACCES;
|
||||
if (ipcperms (&msq->q_perm, S_IRUGO))
|
||||
goto out_unlock;
|
||||
|
||||
err = security_msg_queue_msgctl(msq, cmd);
|
||||
if (err)
|
||||
goto out_unlock;
|
||||
|
||||
kernel_to_ipc64_perm(&msq->q_perm, &tbuf.msg_perm);
|
||||
tbuf.msg_stime = msq->q_stime;
|
||||
tbuf.msg_rtime = msq->q_rtime;
|
||||
tbuf.msg_ctime = msq->q_ctime;
|
||||
tbuf.msg_cbytes = msq->q_cbytes;
|
||||
tbuf.msg_qnum = msq->q_qnum;
|
||||
tbuf.msg_qbytes = msq->q_qbytes;
|
||||
tbuf.msg_lspid = msq->q_lspid;
|
||||
tbuf.msg_lrpid = msq->q_lrpid;
|
||||
msg_unlock(msq);
|
||||
if (copy_msqid_to_user(buf, &tbuf, version))
|
||||
return -EFAULT;
|
||||
return success_return;
|
||||
}
|
||||
case IPC_SET:
|
||||
if (!buf)
|
||||
return -EFAULT;
|
||||
if (copy_msqid_from_user (&setbuf, buf, version))
|
||||
return -EFAULT;
|
||||
if ((err = audit_ipc_perms(setbuf.qbytes, setbuf.uid, setbuf.gid, setbuf.mode)))
|
||||
return err;
|
||||
break;
|
||||
case IPC_RMID:
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
down(&msg_ids.sem);
|
||||
msq = msg_lock(msqid);
|
||||
err=-EINVAL;
|
||||
if (msq == NULL)
|
||||
goto out_up;
|
||||
|
||||
err = -EIDRM;
|
||||
if (msg_checkid(msq,msqid))
|
||||
goto out_unlock_up;
|
||||
ipcp = &msq->q_perm;
|
||||
err = -EPERM;
|
||||
if (current->euid != ipcp->cuid &&
|
||||
current->euid != ipcp->uid && !capable(CAP_SYS_ADMIN))
|
||||
/* We _could_ check for CAP_CHOWN above, but we don't */
|
||||
goto out_unlock_up;
|
||||
|
||||
err = security_msg_queue_msgctl(msq, cmd);
|
||||
if (err)
|
||||
goto out_unlock_up;
|
||||
|
||||
switch (cmd) {
|
||||
case IPC_SET:
|
||||
{
|
||||
err = -EPERM;
|
||||
if (setbuf.qbytes > msg_ctlmnb && !capable(CAP_SYS_RESOURCE))
|
||||
goto out_unlock_up;
|
||||
|
||||
msq->q_qbytes = setbuf.qbytes;
|
||||
|
||||
ipcp->uid = setbuf.uid;
|
||||
ipcp->gid = setbuf.gid;
|
||||
ipcp->mode = (ipcp->mode & ~S_IRWXUGO) |
|
||||
(S_IRWXUGO & setbuf.mode);
|
||||
msq->q_ctime = get_seconds();
|
||||
/* sleeping receivers might be excluded by
|
||||
* stricter permissions.
|
||||
*/
|
||||
expunge_all(msq,-EAGAIN);
|
||||
/* sleeping senders might be able to send
|
||||
* due to a larger queue size.
|
||||
*/
|
||||
ss_wakeup(&msq->q_senders,0);
|
||||
msg_unlock(msq);
|
||||
break;
|
||||
}
|
||||
case IPC_RMID:
|
||||
freeque (msq, msqid);
|
||||
break;
|
||||
}
|
||||
err = 0;
|
||||
out_up:
|
||||
up(&msg_ids.sem);
|
||||
return err;
|
||||
out_unlock_up:
|
||||
msg_unlock(msq);
|
||||
goto out_up;
|
||||
out_unlock:
|
||||
msg_unlock(msq);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int testmsg(struct msg_msg* msg,long type,int mode)
|
||||
{
|
||||
switch(mode)
|
||||
{
|
||||
case SEARCH_ANY:
|
||||
return 1;
|
||||
case SEARCH_LESSEQUAL:
|
||||
if(msg->m_type <=type)
|
||||
return 1;
|
||||
break;
|
||||
case SEARCH_EQUAL:
|
||||
if(msg->m_type == type)
|
||||
return 1;
|
||||
break;
|
||||
case SEARCH_NOTEQUAL:
|
||||
if(msg->m_type != type)
|
||||
return 1;
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int pipelined_send(struct msg_queue* msq, struct msg_msg* msg)
|
||||
{
|
||||
struct list_head* tmp;
|
||||
|
||||
tmp = msq->q_receivers.next;
|
||||
while (tmp != &msq->q_receivers) {
|
||||
struct msg_receiver* msr;
|
||||
msr = list_entry(tmp,struct msg_receiver,r_list);
|
||||
tmp = tmp->next;
|
||||
if(testmsg(msg,msr->r_msgtype,msr->r_mode) &&
|
||||
!security_msg_queue_msgrcv(msq, msg, msr->r_tsk, msr->r_msgtype, msr->r_mode)) {
|
||||
list_del(&msr->r_list);
|
||||
if(msr->r_maxsize < msg->m_ts) {
|
||||
msr->r_msg = NULL;
|
||||
wake_up_process(msr->r_tsk);
|
||||
smp_mb();
|
||||
msr->r_msg = ERR_PTR(-E2BIG);
|
||||
} else {
|
||||
msr->r_msg = NULL;
|
||||
msq->q_lrpid = msr->r_tsk->pid;
|
||||
msq->q_rtime = get_seconds();
|
||||
wake_up_process(msr->r_tsk);
|
||||
smp_mb();
|
||||
msr->r_msg = msg;
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
asmlinkage long sys_msgsnd (int msqid, struct msgbuf __user *msgp, size_t msgsz, int msgflg)
|
||||
{
|
||||
struct msg_queue *msq;
|
||||
struct msg_msg *msg;
|
||||
long mtype;
|
||||
int err;
|
||||
|
||||
if (msgsz > msg_ctlmax || (long) msgsz < 0 || msqid < 0)
|
||||
return -EINVAL;
|
||||
if (get_user(mtype, &msgp->mtype))
|
||||
return -EFAULT;
|
||||
if (mtype < 1)
|
||||
return -EINVAL;
|
||||
|
||||
msg = load_msg(msgp->mtext, msgsz);
|
||||
if(IS_ERR(msg))
|
||||
return PTR_ERR(msg);
|
||||
|
||||
msg->m_type = mtype;
|
||||
msg->m_ts = msgsz;
|
||||
|
||||
msq = msg_lock(msqid);
|
||||
err=-EINVAL;
|
||||
if(msq==NULL)
|
||||
goto out_free;
|
||||
|
||||
err= -EIDRM;
|
||||
if (msg_checkid(msq,msqid))
|
||||
goto out_unlock_free;
|
||||
|
||||
for (;;) {
|
||||
struct msg_sender s;
|
||||
|
||||
err=-EACCES;
|
||||
if (ipcperms(&msq->q_perm, S_IWUGO))
|
||||
goto out_unlock_free;
|
||||
|
||||
err = security_msg_queue_msgsnd(msq, msg, msgflg);
|
||||
if (err)
|
||||
goto out_unlock_free;
|
||||
|
||||
if(msgsz + msq->q_cbytes <= msq->q_qbytes &&
|
||||
1 + msq->q_qnum <= msq->q_qbytes) {
|
||||
break;
|
||||
}
|
||||
|
||||
/* queue full, wait: */
|
||||
if(msgflg&IPC_NOWAIT) {
|
||||
err=-EAGAIN;
|
||||
goto out_unlock_free;
|
||||
}
|
||||
ss_add(msq, &s);
|
||||
ipc_rcu_getref(msq);
|
||||
msg_unlock(msq);
|
||||
schedule();
|
||||
|
||||
ipc_lock_by_ptr(&msq->q_perm);
|
||||
ipc_rcu_putref(msq);
|
||||
if (msq->q_perm.deleted) {
|
||||
err = -EIDRM;
|
||||
goto out_unlock_free;
|
||||
}
|
||||
ss_del(&s);
|
||||
|
||||
if (signal_pending(current)) {
|
||||
err=-ERESTARTNOHAND;
|
||||
goto out_unlock_free;
|
||||
}
|
||||
}
|
||||
|
||||
msq->q_lspid = current->tgid;
|
||||
msq->q_stime = get_seconds();
|
||||
|
||||
if(!pipelined_send(msq,msg)) {
|
||||
/* noone is waiting for this message, enqueue it */
|
||||
list_add_tail(&msg->m_list,&msq->q_messages);
|
||||
msq->q_cbytes += msgsz;
|
||||
msq->q_qnum++;
|
||||
atomic_add(msgsz,&msg_bytes);
|
||||
atomic_inc(&msg_hdrs);
|
||||
}
|
||||
|
||||
err = 0;
|
||||
msg = NULL;
|
||||
|
||||
out_unlock_free:
|
||||
msg_unlock(msq);
|
||||
out_free:
|
||||
if(msg!=NULL)
|
||||
free_msg(msg);
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline int convert_mode(long* msgtyp, int msgflg)
|
||||
{
|
||||
/*
|
||||
* find message of correct type.
|
||||
* msgtyp = 0 => get first.
|
||||
* msgtyp > 0 => get first message of matching type.
|
||||
* msgtyp < 0 => get message with least type must be < abs(msgtype).
|
||||
*/
|
||||
if(*msgtyp==0)
|
||||
return SEARCH_ANY;
|
||||
if(*msgtyp<0) {
|
||||
*msgtyp=-(*msgtyp);
|
||||
return SEARCH_LESSEQUAL;
|
||||
}
|
||||
if(msgflg & MSG_EXCEPT)
|
||||
return SEARCH_NOTEQUAL;
|
||||
return SEARCH_EQUAL;
|
||||
}
|
||||
|
||||
asmlinkage long sys_msgrcv (int msqid, struct msgbuf __user *msgp, size_t msgsz,
|
||||
long msgtyp, int msgflg)
|
||||
{
|
||||
struct msg_queue *msq;
|
||||
struct msg_msg *msg;
|
||||
int mode;
|
||||
|
||||
if (msqid < 0 || (long) msgsz < 0)
|
||||
return -EINVAL;
|
||||
mode = convert_mode(&msgtyp,msgflg);
|
||||
|
||||
msq = msg_lock(msqid);
|
||||
if(msq==NULL)
|
||||
return -EINVAL;
|
||||
|
||||
msg = ERR_PTR(-EIDRM);
|
||||
if (msg_checkid(msq,msqid))
|
||||
goto out_unlock;
|
||||
|
||||
for (;;) {
|
||||
struct msg_receiver msr_d;
|
||||
struct list_head* tmp;
|
||||
|
||||
msg = ERR_PTR(-EACCES);
|
||||
if (ipcperms (&msq->q_perm, S_IRUGO))
|
||||
goto out_unlock;
|
||||
|
||||
msg = ERR_PTR(-EAGAIN);
|
||||
tmp = msq->q_messages.next;
|
||||
while (tmp != &msq->q_messages) {
|
||||
struct msg_msg *walk_msg;
|
||||
walk_msg = list_entry(tmp,struct msg_msg,m_list);
|
||||
if(testmsg(walk_msg,msgtyp,mode) &&
|
||||
!security_msg_queue_msgrcv(msq, walk_msg, current, msgtyp, mode)) {
|
||||
msg = walk_msg;
|
||||
if(mode == SEARCH_LESSEQUAL && walk_msg->m_type != 1) {
|
||||
msg=walk_msg;
|
||||
msgtyp=walk_msg->m_type-1;
|
||||
} else {
|
||||
msg=walk_msg;
|
||||
break;
|
||||
}
|
||||
}
|
||||
tmp = tmp->next;
|
||||
}
|
||||
if(!IS_ERR(msg)) {
|
||||
/* Found a suitable message. Unlink it from the queue. */
|
||||
if ((msgsz < msg->m_ts) && !(msgflg & MSG_NOERROR)) {
|
||||
msg = ERR_PTR(-E2BIG);
|
||||
goto out_unlock;
|
||||
}
|
||||
list_del(&msg->m_list);
|
||||
msq->q_qnum--;
|
||||
msq->q_rtime = get_seconds();
|
||||
msq->q_lrpid = current->tgid;
|
||||
msq->q_cbytes -= msg->m_ts;
|
||||
atomic_sub(msg->m_ts,&msg_bytes);
|
||||
atomic_dec(&msg_hdrs);
|
||||
ss_wakeup(&msq->q_senders,0);
|
||||
msg_unlock(msq);
|
||||
break;
|
||||
}
|
||||
/* No message waiting. Wait for a message */
|
||||
if (msgflg & IPC_NOWAIT) {
|
||||
msg = ERR_PTR(-ENOMSG);
|
||||
goto out_unlock;
|
||||
}
|
||||
list_add_tail(&msr_d.r_list,&msq->q_receivers);
|
||||
msr_d.r_tsk = current;
|
||||
msr_d.r_msgtype = msgtyp;
|
||||
msr_d.r_mode = mode;
|
||||
if(msgflg & MSG_NOERROR)
|
||||
msr_d.r_maxsize = INT_MAX;
|
||||
else
|
||||
msr_d.r_maxsize = msgsz;
|
||||
msr_d.r_msg = ERR_PTR(-EAGAIN);
|
||||
current->state = TASK_INTERRUPTIBLE;
|
||||
msg_unlock(msq);
|
||||
|
||||
schedule();
|
||||
|
||||
/* Lockless receive, part 1:
|
||||
* Disable preemption. We don't hold a reference to the queue
|
||||
* and getting a reference would defeat the idea of a lockless
|
||||
* operation, thus the code relies on rcu to guarantee the
|
||||
* existance of msq:
|
||||
* Prior to destruction, expunge_all(-EIRDM) changes r_msg.
|
||||
* Thus if r_msg is -EAGAIN, then the queue not yet destroyed.
|
||||
* rcu_read_lock() prevents preemption between reading r_msg
|
||||
* and the spin_lock() inside ipc_lock_by_ptr().
|
||||
*/
|
||||
rcu_read_lock();
|
||||
|
||||
/* Lockless receive, part 2:
|
||||
* Wait until pipelined_send or expunge_all are outside of
|
||||
* wake_up_process(). There is a race with exit(), see
|
||||
* ipc/mqueue.c for the details.
|
||||
*/
|
||||
msg = (struct msg_msg*) msr_d.r_msg;
|
||||
while (msg == NULL) {
|
||||
cpu_relax();
|
||||
msg = (struct msg_msg*) msr_d.r_msg;
|
||||
}
|
||||
|
||||
/* Lockless receive, part 3:
|
||||
* If there is a message or an error then accept it without
|
||||
* locking.
|
||||
*/
|
||||
if(msg != ERR_PTR(-EAGAIN)) {
|
||||
rcu_read_unlock();
|
||||
break;
|
||||
}
|
||||
|
||||
/* Lockless receive, part 3:
|
||||
* Acquire the queue spinlock.
|
||||
*/
|
||||
ipc_lock_by_ptr(&msq->q_perm);
|
||||
rcu_read_unlock();
|
||||
|
||||
/* Lockless receive, part 4:
|
||||
* Repeat test after acquiring the spinlock.
|
||||
*/
|
||||
msg = (struct msg_msg*)msr_d.r_msg;
|
||||
if(msg != ERR_PTR(-EAGAIN))
|
||||
goto out_unlock;
|
||||
|
||||
list_del(&msr_d.r_list);
|
||||
if (signal_pending(current)) {
|
||||
msg = ERR_PTR(-ERESTARTNOHAND);
|
||||
out_unlock:
|
||||
msg_unlock(msq);
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (IS_ERR(msg))
|
||||
return PTR_ERR(msg);
|
||||
|
||||
msgsz = (msgsz > msg->m_ts) ? msg->m_ts : msgsz;
|
||||
if (put_user (msg->m_type, &msgp->mtype) ||
|
||||
store_msg(msgp->mtext, msg, msgsz)) {
|
||||
msgsz = -EFAULT;
|
||||
}
|
||||
free_msg(msg);
|
||||
return msgsz;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PROC_FS
|
||||
static int sysvipc_msg_read_proc(char *buffer, char **start, off_t offset, int length, int *eof, void *data)
|
||||
{
|
||||
off_t pos = 0;
|
||||
off_t begin = 0;
|
||||
int i, len = 0;
|
||||
|
||||
down(&msg_ids.sem);
|
||||
len += sprintf(buffer, " key msqid perms cbytes qnum lspid lrpid uid gid cuid cgid stime rtime ctime\n");
|
||||
|
||||
for(i = 0; i <= msg_ids.max_id; i++) {
|
||||
struct msg_queue * msq;
|
||||
msq = msg_lock(i);
|
||||
if(msq != NULL) {
|
||||
len += sprintf(buffer + len, "%10d %10d %4o %10lu %10lu %5u %5u %5u %5u %5u %5u %10lu %10lu %10lu\n",
|
||||
msq->q_perm.key,
|
||||
msg_buildid(i,msq->q_perm.seq),
|
||||
msq->q_perm.mode,
|
||||
msq->q_cbytes,
|
||||
msq->q_qnum,
|
||||
msq->q_lspid,
|
||||
msq->q_lrpid,
|
||||
msq->q_perm.uid,
|
||||
msq->q_perm.gid,
|
||||
msq->q_perm.cuid,
|
||||
msq->q_perm.cgid,
|
||||
msq->q_stime,
|
||||
msq->q_rtime,
|
||||
msq->q_ctime);
|
||||
msg_unlock(msq);
|
||||
|
||||
pos += len;
|
||||
if(pos < offset) {
|
||||
len = 0;
|
||||
begin = pos;
|
||||
}
|
||||
if(pos > offset + length)
|
||||
goto done;
|
||||
}
|
||||
|
||||
}
|
||||
*eof = 1;
|
||||
done:
|
||||
up(&msg_ids.sem);
|
||||
*start = buffer + (offset - begin);
|
||||
len -= (offset - begin);
|
||||
if(len > length)
|
||||
len = length;
|
||||
if(len < 0)
|
||||
len = 0;
|
||||
return len;
|
||||
}
|
||||
#endif
|
127
ipc/msgutil.c
Normal file
127
ipc/msgutil.c
Normal file
@@ -0,0 +1,127 @@
|
||||
/*
|
||||
* linux/ipc/util.c
|
||||
* Copyright (C) 1999, 2004 Manfred Spraul
|
||||
*
|
||||
* This file is released under GNU General Public Licence version 2 or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* See the file COPYING for more details.
|
||||
*/
|
||||
|
||||
#include <linux/spinlock.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/security.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/ipc.h>
|
||||
#include <asm/uaccess.h>
|
||||
|
||||
#include "util.h"
|
||||
|
||||
struct msg_msgseg {
|
||||
struct msg_msgseg* next;
|
||||
/* the next part of the message follows immediately */
|
||||
};
|
||||
|
||||
#define DATALEN_MSG (PAGE_SIZE-sizeof(struct msg_msg))
|
||||
#define DATALEN_SEG (PAGE_SIZE-sizeof(struct msg_msgseg))
|
||||
|
||||
struct msg_msg *load_msg(const void __user *src, int len)
|
||||
{
|
||||
struct msg_msg *msg;
|
||||
struct msg_msgseg **pseg;
|
||||
int err;
|
||||
int alen;
|
||||
|
||||
alen = len;
|
||||
if (alen > DATALEN_MSG)
|
||||
alen = DATALEN_MSG;
|
||||
|
||||
msg = (struct msg_msg *)kmalloc(sizeof(*msg) + alen, GFP_KERNEL);
|
||||
if (msg == NULL)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
msg->next = NULL;
|
||||
msg->security = NULL;
|
||||
|
||||
if (copy_from_user(msg + 1, src, alen)) {
|
||||
err = -EFAULT;
|
||||
goto out_err;
|
||||
}
|
||||
|
||||
len -= alen;
|
||||
src = ((char __user *)src) + alen;
|
||||
pseg = &msg->next;
|
||||
while (len > 0) {
|
||||
struct msg_msgseg *seg;
|
||||
alen = len;
|
||||
if (alen > DATALEN_SEG)
|
||||
alen = DATALEN_SEG;
|
||||
seg = (struct msg_msgseg *)kmalloc(sizeof(*seg) + alen,
|
||||
GFP_KERNEL);
|
||||
if (seg == NULL) {
|
||||
err = -ENOMEM;
|
||||
goto out_err;
|
||||
}
|
||||
*pseg = seg;
|
||||
seg->next = NULL;
|
||||
if (copy_from_user(seg + 1, src, alen)) {
|
||||
err = -EFAULT;
|
||||
goto out_err;
|
||||
}
|
||||
pseg = &seg->next;
|
||||
len -= alen;
|
||||
src = ((char __user *)src) + alen;
|
||||
}
|
||||
|
||||
err = security_msg_msg_alloc(msg);
|
||||
if (err)
|
||||
goto out_err;
|
||||
|
||||
return msg;
|
||||
|
||||
out_err:
|
||||
free_msg(msg);
|
||||
return ERR_PTR(err);
|
||||
}
|
||||
|
||||
int store_msg(void __user *dest, struct msg_msg *msg, int len)
|
||||
{
|
||||
int alen;
|
||||
struct msg_msgseg *seg;
|
||||
|
||||
alen = len;
|
||||
if (alen > DATALEN_MSG)
|
||||
alen = DATALEN_MSG;
|
||||
if (copy_to_user(dest, msg + 1, alen))
|
||||
return -1;
|
||||
|
||||
len -= alen;
|
||||
dest = ((char __user *)dest) + alen;
|
||||
seg = msg->next;
|
||||
while (len > 0) {
|
||||
alen = len;
|
||||
if (alen > DATALEN_SEG)
|
||||
alen = DATALEN_SEG;
|
||||
if (copy_to_user(dest, seg + 1, alen))
|
||||
return -1;
|
||||
len -= alen;
|
||||
dest = ((char __user *)dest) + alen;
|
||||
seg = seg->next;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void free_msg(struct msg_msg *msg)
|
||||
{
|
||||
struct msg_msgseg *seg;
|
||||
|
||||
security_msg_msg_free(msg);
|
||||
|
||||
seg = msg->next;
|
||||
kfree(msg);
|
||||
while (seg != NULL) {
|
||||
struct msg_msgseg *tmp = seg->next;
|
||||
kfree(seg);
|
||||
seg = tmp;
|
||||
}
|
||||
}
|
917
ipc/shm.c
Normal file
917
ipc/shm.c
Normal file
@@ -0,0 +1,917 @@
|
||||
/*
|
||||
* linux/ipc/shm.c
|
||||
* Copyright (C) 1992, 1993 Krishna Balasubramanian
|
||||
* Many improvements/fixes by Bruno Haible.
|
||||
* Replaced `struct shm_desc' by `struct vm_area_struct', July 1994.
|
||||
* Fixed the shm swap deallocation (shm_unuse()), August 1998 Andrea Arcangeli.
|
||||
*
|
||||
* /proc/sysvipc/shm support (c) 1999 Dragos Acostachioaie <dragos@iname.com>
|
||||
* BIGMEM support, Andrea Arcangeli <andrea@suse.de>
|
||||
* SMP thread shm, Jean-Luc Boyard <jean-luc.boyard@siemens.fr>
|
||||
* HIGHMEM support, Ingo Molnar <mingo@redhat.com>
|
||||
* Make shmmax, shmall, shmmni sysctl'able, Christoph Rohland <cr@sap.com>
|
||||
* Shared /dev/zero support, Kanoj Sarcar <kanoj@sgi.com>
|
||||
* Move the mm functionality over to mm/shmem.c, Christoph Rohland <cr@sap.com>
|
||||
*
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/mm.h>
|
||||
#include <linux/hugetlb.h>
|
||||
#include <linux/shm.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/file.h>
|
||||
#include <linux/mman.h>
|
||||
#include <linux/proc_fs.h>
|
||||
#include <linux/shmem_fs.h>
|
||||
#include <linux/security.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <linux/audit.h>
|
||||
#include <asm/uaccess.h>
|
||||
|
||||
#include "util.h"
|
||||
|
||||
#define shm_flags shm_perm.mode
|
||||
|
||||
static struct file_operations shm_file_operations;
|
||||
static struct vm_operations_struct shm_vm_ops;
|
||||
|
||||
static struct ipc_ids shm_ids;
|
||||
|
||||
#define shm_lock(id) ((struct shmid_kernel*)ipc_lock(&shm_ids,id))
|
||||
#define shm_unlock(shp) ipc_unlock(&(shp)->shm_perm)
|
||||
#define shm_get(id) ((struct shmid_kernel*)ipc_get(&shm_ids,id))
|
||||
#define shm_buildid(id, seq) \
|
||||
ipc_buildid(&shm_ids, id, seq)
|
||||
|
||||
static int newseg (key_t key, int shmflg, size_t size);
|
||||
static void shm_open (struct vm_area_struct *shmd);
|
||||
static void shm_close (struct vm_area_struct *shmd);
|
||||
#ifdef CONFIG_PROC_FS
|
||||
static int sysvipc_shm_read_proc(char *buffer, char **start, off_t offset, int length, int *eof, void *data);
|
||||
#endif
|
||||
|
||||
size_t shm_ctlmax = SHMMAX;
|
||||
size_t shm_ctlall = SHMALL;
|
||||
int shm_ctlmni = SHMMNI;
|
||||
|
||||
static int shm_tot; /* total number of shared memory pages */
|
||||
|
||||
void __init shm_init (void)
|
||||
{
|
||||
ipc_init_ids(&shm_ids, 1);
|
||||
#ifdef CONFIG_PROC_FS
|
||||
create_proc_read_entry("sysvipc/shm", 0, NULL, sysvipc_shm_read_proc, NULL);
|
||||
#endif
|
||||
}
|
||||
|
||||
static inline int shm_checkid(struct shmid_kernel *s, int id)
|
||||
{
|
||||
if (ipc_checkid(&shm_ids,&s->shm_perm,id))
|
||||
return -EIDRM;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline struct shmid_kernel *shm_rmid(int id)
|
||||
{
|
||||
return (struct shmid_kernel *)ipc_rmid(&shm_ids,id);
|
||||
}
|
||||
|
||||
static inline int shm_addid(struct shmid_kernel *shp)
|
||||
{
|
||||
return ipc_addid(&shm_ids, &shp->shm_perm, shm_ctlmni);
|
||||
}
|
||||
|
||||
|
||||
|
||||
static inline void shm_inc (int id) {
|
||||
struct shmid_kernel *shp;
|
||||
|
||||
if(!(shp = shm_lock(id)))
|
||||
BUG();
|
||||
shp->shm_atim = get_seconds();
|
||||
shp->shm_lprid = current->tgid;
|
||||
shp->shm_nattch++;
|
||||
shm_unlock(shp);
|
||||
}
|
||||
|
||||
/* This is called by fork, once for every shm attach. */
|
||||
static void shm_open (struct vm_area_struct *shmd)
|
||||
{
|
||||
shm_inc (shmd->vm_file->f_dentry->d_inode->i_ino);
|
||||
}
|
||||
|
||||
/*
|
||||
* shm_destroy - free the struct shmid_kernel
|
||||
*
|
||||
* @shp: struct to free
|
||||
*
|
||||
* It has to be called with shp and shm_ids.sem locked,
|
||||
* but returns with shp unlocked and freed.
|
||||
*/
|
||||
static void shm_destroy (struct shmid_kernel *shp)
|
||||
{
|
||||
shm_tot -= (shp->shm_segsz + PAGE_SIZE - 1) >> PAGE_SHIFT;
|
||||
shm_rmid (shp->id);
|
||||
shm_unlock(shp);
|
||||
if (!is_file_hugepages(shp->shm_file))
|
||||
shmem_lock(shp->shm_file, 0, shp->mlock_user);
|
||||
else
|
||||
user_shm_unlock(shp->shm_file->f_dentry->d_inode->i_size,
|
||||
shp->mlock_user);
|
||||
fput (shp->shm_file);
|
||||
security_shm_free(shp);
|
||||
ipc_rcu_putref(shp);
|
||||
}
|
||||
|
||||
/*
|
||||
* remove the attach descriptor shmd.
|
||||
* free memory for segment if it is marked destroyed.
|
||||
* The descriptor has already been removed from the current->mm->mmap list
|
||||
* and will later be kfree()d.
|
||||
*/
|
||||
static void shm_close (struct vm_area_struct *shmd)
|
||||
{
|
||||
struct file * file = shmd->vm_file;
|
||||
int id = file->f_dentry->d_inode->i_ino;
|
||||
struct shmid_kernel *shp;
|
||||
|
||||
down (&shm_ids.sem);
|
||||
/* remove from the list of attaches of the shm segment */
|
||||
if(!(shp = shm_lock(id)))
|
||||
BUG();
|
||||
shp->shm_lprid = current->tgid;
|
||||
shp->shm_dtim = get_seconds();
|
||||
shp->shm_nattch--;
|
||||
if(shp->shm_nattch == 0 &&
|
||||
shp->shm_flags & SHM_DEST)
|
||||
shm_destroy (shp);
|
||||
else
|
||||
shm_unlock(shp);
|
||||
up (&shm_ids.sem);
|
||||
}
|
||||
|
||||
static int shm_mmap(struct file * file, struct vm_area_struct * vma)
|
||||
{
|
||||
file_accessed(file);
|
||||
vma->vm_ops = &shm_vm_ops;
|
||||
shm_inc(file->f_dentry->d_inode->i_ino);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct file_operations shm_file_operations = {
|
||||
.mmap = shm_mmap
|
||||
};
|
||||
|
||||
static struct vm_operations_struct shm_vm_ops = {
|
||||
.open = shm_open, /* callback for a new vm-area open */
|
||||
.close = shm_close, /* callback for when the vm-area is released */
|
||||
.nopage = shmem_nopage,
|
||||
#ifdef CONFIG_NUMA
|
||||
.set_policy = shmem_set_policy,
|
||||
.get_policy = shmem_get_policy,
|
||||
#endif
|
||||
};
|
||||
|
||||
static int newseg (key_t key, int shmflg, size_t size)
|
||||
{
|
||||
int error;
|
||||
struct shmid_kernel *shp;
|
||||
int numpages = (size + PAGE_SIZE -1) >> PAGE_SHIFT;
|
||||
struct file * file;
|
||||
char name[13];
|
||||
int id;
|
||||
|
||||
if (size < SHMMIN || size > shm_ctlmax)
|
||||
return -EINVAL;
|
||||
|
||||
if (shm_tot + numpages >= shm_ctlall)
|
||||
return -ENOSPC;
|
||||
|
||||
shp = ipc_rcu_alloc(sizeof(*shp));
|
||||
if (!shp)
|
||||
return -ENOMEM;
|
||||
|
||||
shp->shm_perm.key = key;
|
||||
shp->shm_flags = (shmflg & S_IRWXUGO);
|
||||
shp->mlock_user = NULL;
|
||||
|
||||
shp->shm_perm.security = NULL;
|
||||
error = security_shm_alloc(shp);
|
||||
if (error) {
|
||||
ipc_rcu_putref(shp);
|
||||
return error;
|
||||
}
|
||||
|
||||
if (shmflg & SHM_HUGETLB) {
|
||||
/* hugetlb_zero_setup takes care of mlock user accounting */
|
||||
file = hugetlb_zero_setup(size);
|
||||
shp->mlock_user = current->user;
|
||||
} else {
|
||||
sprintf (name, "SYSV%08x", key);
|
||||
file = shmem_file_setup(name, size, VM_ACCOUNT);
|
||||
}
|
||||
error = PTR_ERR(file);
|
||||
if (IS_ERR(file))
|
||||
goto no_file;
|
||||
|
||||
error = -ENOSPC;
|
||||
id = shm_addid(shp);
|
||||
if(id == -1)
|
||||
goto no_id;
|
||||
|
||||
shp->shm_cprid = current->tgid;
|
||||
shp->shm_lprid = 0;
|
||||
shp->shm_atim = shp->shm_dtim = 0;
|
||||
shp->shm_ctim = get_seconds();
|
||||
shp->shm_segsz = size;
|
||||
shp->shm_nattch = 0;
|
||||
shp->id = shm_buildid(id,shp->shm_perm.seq);
|
||||
shp->shm_file = file;
|
||||
file->f_dentry->d_inode->i_ino = shp->id;
|
||||
if (shmflg & SHM_HUGETLB)
|
||||
set_file_hugepages(file);
|
||||
else
|
||||
file->f_op = &shm_file_operations;
|
||||
shm_tot += numpages;
|
||||
shm_unlock(shp);
|
||||
return shp->id;
|
||||
|
||||
no_id:
|
||||
fput(file);
|
||||
no_file:
|
||||
security_shm_free(shp);
|
||||
ipc_rcu_putref(shp);
|
||||
return error;
|
||||
}
|
||||
|
||||
asmlinkage long sys_shmget (key_t key, size_t size, int shmflg)
|
||||
{
|
||||
struct shmid_kernel *shp;
|
||||
int err, id = 0;
|
||||
|
||||
down(&shm_ids.sem);
|
||||
if (key == IPC_PRIVATE) {
|
||||
err = newseg(key, shmflg, size);
|
||||
} else if ((id = ipc_findkey(&shm_ids, key)) == -1) {
|
||||
if (!(shmflg & IPC_CREAT))
|
||||
err = -ENOENT;
|
||||
else
|
||||
err = newseg(key, shmflg, size);
|
||||
} else if ((shmflg & IPC_CREAT) && (shmflg & IPC_EXCL)) {
|
||||
err = -EEXIST;
|
||||
} else {
|
||||
shp = shm_lock(id);
|
||||
if(shp==NULL)
|
||||
BUG();
|
||||
if (shp->shm_segsz < size)
|
||||
err = -EINVAL;
|
||||
else if (ipcperms(&shp->shm_perm, shmflg))
|
||||
err = -EACCES;
|
||||
else {
|
||||
int shmid = shm_buildid(id, shp->shm_perm.seq);
|
||||
err = security_shm_associate(shp, shmflg);
|
||||
if (!err)
|
||||
err = shmid;
|
||||
}
|
||||
shm_unlock(shp);
|
||||
}
|
||||
up(&shm_ids.sem);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline unsigned long copy_shmid_to_user(void __user *buf, struct shmid64_ds *in, int version)
|
||||
{
|
||||
switch(version) {
|
||||
case IPC_64:
|
||||
return copy_to_user(buf, in, sizeof(*in));
|
||||
case IPC_OLD:
|
||||
{
|
||||
struct shmid_ds out;
|
||||
|
||||
ipc64_perm_to_ipc_perm(&in->shm_perm, &out.shm_perm);
|
||||
out.shm_segsz = in->shm_segsz;
|
||||
out.shm_atime = in->shm_atime;
|
||||
out.shm_dtime = in->shm_dtime;
|
||||
out.shm_ctime = in->shm_ctime;
|
||||
out.shm_cpid = in->shm_cpid;
|
||||
out.shm_lpid = in->shm_lpid;
|
||||
out.shm_nattch = in->shm_nattch;
|
||||
|
||||
return copy_to_user(buf, &out, sizeof(out));
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
struct shm_setbuf {
|
||||
uid_t uid;
|
||||
gid_t gid;
|
||||
mode_t mode;
|
||||
};
|
||||
|
||||
static inline unsigned long copy_shmid_from_user(struct shm_setbuf *out, void __user *buf, int version)
|
||||
{
|
||||
switch(version) {
|
||||
case IPC_64:
|
||||
{
|
||||
struct shmid64_ds tbuf;
|
||||
|
||||
if (copy_from_user(&tbuf, buf, sizeof(tbuf)))
|
||||
return -EFAULT;
|
||||
|
||||
out->uid = tbuf.shm_perm.uid;
|
||||
out->gid = tbuf.shm_perm.gid;
|
||||
out->mode = tbuf.shm_flags;
|
||||
|
||||
return 0;
|
||||
}
|
||||
case IPC_OLD:
|
||||
{
|
||||
struct shmid_ds tbuf_old;
|
||||
|
||||
if (copy_from_user(&tbuf_old, buf, sizeof(tbuf_old)))
|
||||
return -EFAULT;
|
||||
|
||||
out->uid = tbuf_old.shm_perm.uid;
|
||||
out->gid = tbuf_old.shm_perm.gid;
|
||||
out->mode = tbuf_old.shm_flags;
|
||||
|
||||
return 0;
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
static inline unsigned long copy_shminfo_to_user(void __user *buf, struct shminfo64 *in, int version)
|
||||
{
|
||||
switch(version) {
|
||||
case IPC_64:
|
||||
return copy_to_user(buf, in, sizeof(*in));
|
||||
case IPC_OLD:
|
||||
{
|
||||
struct shminfo out;
|
||||
|
||||
if(in->shmmax > INT_MAX)
|
||||
out.shmmax = INT_MAX;
|
||||
else
|
||||
out.shmmax = (int)in->shmmax;
|
||||
|
||||
out.shmmin = in->shmmin;
|
||||
out.shmmni = in->shmmni;
|
||||
out.shmseg = in->shmseg;
|
||||
out.shmall = in->shmall;
|
||||
|
||||
return copy_to_user(buf, &out, sizeof(out));
|
||||
}
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
|
||||
static void shm_get_stat(unsigned long *rss, unsigned long *swp)
|
||||
{
|
||||
int i;
|
||||
|
||||
*rss = 0;
|
||||
*swp = 0;
|
||||
|
||||
for (i = 0; i <= shm_ids.max_id; i++) {
|
||||
struct shmid_kernel *shp;
|
||||
struct inode *inode;
|
||||
|
||||
shp = shm_get(i);
|
||||
if(!shp)
|
||||
continue;
|
||||
|
||||
inode = shp->shm_file->f_dentry->d_inode;
|
||||
|
||||
if (is_file_hugepages(shp->shm_file)) {
|
||||
struct address_space *mapping = inode->i_mapping;
|
||||
*rss += (HPAGE_SIZE/PAGE_SIZE)*mapping->nrpages;
|
||||
} else {
|
||||
struct shmem_inode_info *info = SHMEM_I(inode);
|
||||
spin_lock(&info->lock);
|
||||
*rss += inode->i_mapping->nrpages;
|
||||
*swp += info->swapped;
|
||||
spin_unlock(&info->lock);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
asmlinkage long sys_shmctl (int shmid, int cmd, struct shmid_ds __user *buf)
|
||||
{
|
||||
struct shm_setbuf setbuf;
|
||||
struct shmid_kernel *shp;
|
||||
int err, version;
|
||||
|
||||
if (cmd < 0 || shmid < 0) {
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
version = ipc_parse_version(&cmd);
|
||||
|
||||
switch (cmd) { /* replace with proc interface ? */
|
||||
case IPC_INFO:
|
||||
{
|
||||
struct shminfo64 shminfo;
|
||||
|
||||
err = security_shm_shmctl(NULL, cmd);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
memset(&shminfo,0,sizeof(shminfo));
|
||||
shminfo.shmmni = shminfo.shmseg = shm_ctlmni;
|
||||
shminfo.shmmax = shm_ctlmax;
|
||||
shminfo.shmall = shm_ctlall;
|
||||
|
||||
shminfo.shmmin = SHMMIN;
|
||||
if(copy_shminfo_to_user (buf, &shminfo, version))
|
||||
return -EFAULT;
|
||||
/* reading a integer is always atomic */
|
||||
err= shm_ids.max_id;
|
||||
if(err<0)
|
||||
err = 0;
|
||||
goto out;
|
||||
}
|
||||
case SHM_INFO:
|
||||
{
|
||||
struct shm_info shm_info;
|
||||
|
||||
err = security_shm_shmctl(NULL, cmd);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
memset(&shm_info,0,sizeof(shm_info));
|
||||
down(&shm_ids.sem);
|
||||
shm_info.used_ids = shm_ids.in_use;
|
||||
shm_get_stat (&shm_info.shm_rss, &shm_info.shm_swp);
|
||||
shm_info.shm_tot = shm_tot;
|
||||
shm_info.swap_attempts = 0;
|
||||
shm_info.swap_successes = 0;
|
||||
err = shm_ids.max_id;
|
||||
up(&shm_ids.sem);
|
||||
if(copy_to_user (buf, &shm_info, sizeof(shm_info))) {
|
||||
err = -EFAULT;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = err < 0 ? 0 : err;
|
||||
goto out;
|
||||
}
|
||||
case SHM_STAT:
|
||||
case IPC_STAT:
|
||||
{
|
||||
struct shmid64_ds tbuf;
|
||||
int result;
|
||||
memset(&tbuf, 0, sizeof(tbuf));
|
||||
shp = shm_lock(shmid);
|
||||
if(shp==NULL) {
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
} else if(cmd==SHM_STAT) {
|
||||
err = -EINVAL;
|
||||
if (shmid > shm_ids.max_id)
|
||||
goto out_unlock;
|
||||
result = shm_buildid(shmid, shp->shm_perm.seq);
|
||||
} else {
|
||||
err = shm_checkid(shp,shmid);
|
||||
if(err)
|
||||
goto out_unlock;
|
||||
result = 0;
|
||||
}
|
||||
err=-EACCES;
|
||||
if (ipcperms (&shp->shm_perm, S_IRUGO))
|
||||
goto out_unlock;
|
||||
err = security_shm_shmctl(shp, cmd);
|
||||
if (err)
|
||||
goto out_unlock;
|
||||
kernel_to_ipc64_perm(&shp->shm_perm, &tbuf.shm_perm);
|
||||
tbuf.shm_segsz = shp->shm_segsz;
|
||||
tbuf.shm_atime = shp->shm_atim;
|
||||
tbuf.shm_dtime = shp->shm_dtim;
|
||||
tbuf.shm_ctime = shp->shm_ctim;
|
||||
tbuf.shm_cpid = shp->shm_cprid;
|
||||
tbuf.shm_lpid = shp->shm_lprid;
|
||||
if (!is_file_hugepages(shp->shm_file))
|
||||
tbuf.shm_nattch = shp->shm_nattch;
|
||||
else
|
||||
tbuf.shm_nattch = file_count(shp->shm_file) - 1;
|
||||
shm_unlock(shp);
|
||||
if(copy_shmid_to_user (buf, &tbuf, version))
|
||||
err = -EFAULT;
|
||||
else
|
||||
err = result;
|
||||
goto out;
|
||||
}
|
||||
case SHM_LOCK:
|
||||
case SHM_UNLOCK:
|
||||
{
|
||||
shp = shm_lock(shmid);
|
||||
if(shp==NULL) {
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
err = shm_checkid(shp,shmid);
|
||||
if(err)
|
||||
goto out_unlock;
|
||||
|
||||
if (!capable(CAP_IPC_LOCK)) {
|
||||
err = -EPERM;
|
||||
if (current->euid != shp->shm_perm.uid &&
|
||||
current->euid != shp->shm_perm.cuid)
|
||||
goto out_unlock;
|
||||
if (cmd == SHM_LOCK &&
|
||||
!current->signal->rlim[RLIMIT_MEMLOCK].rlim_cur)
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
err = security_shm_shmctl(shp, cmd);
|
||||
if (err)
|
||||
goto out_unlock;
|
||||
|
||||
if(cmd==SHM_LOCK) {
|
||||
struct user_struct * user = current->user;
|
||||
if (!is_file_hugepages(shp->shm_file)) {
|
||||
err = shmem_lock(shp->shm_file, 1, user);
|
||||
if (!err) {
|
||||
shp->shm_flags |= SHM_LOCKED;
|
||||
shp->mlock_user = user;
|
||||
}
|
||||
}
|
||||
} else if (!is_file_hugepages(shp->shm_file)) {
|
||||
shmem_lock(shp->shm_file, 0, shp->mlock_user);
|
||||
shp->shm_flags &= ~SHM_LOCKED;
|
||||
shp->mlock_user = NULL;
|
||||
}
|
||||
shm_unlock(shp);
|
||||
goto out;
|
||||
}
|
||||
case IPC_RMID:
|
||||
{
|
||||
/*
|
||||
* We cannot simply remove the file. The SVID states
|
||||
* that the block remains until the last person
|
||||
* detaches from it, then is deleted. A shmat() on
|
||||
* an RMID segment is legal in older Linux and if
|
||||
* we change it apps break...
|
||||
*
|
||||
* Instead we set a destroyed flag, and then blow
|
||||
* the name away when the usage hits zero.
|
||||
*/
|
||||
down(&shm_ids.sem);
|
||||
shp = shm_lock(shmid);
|
||||
err = -EINVAL;
|
||||
if (shp == NULL)
|
||||
goto out_up;
|
||||
err = shm_checkid(shp, shmid);
|
||||
if(err)
|
||||
goto out_unlock_up;
|
||||
|
||||
if (current->euid != shp->shm_perm.uid &&
|
||||
current->euid != shp->shm_perm.cuid &&
|
||||
!capable(CAP_SYS_ADMIN)) {
|
||||
err=-EPERM;
|
||||
goto out_unlock_up;
|
||||
}
|
||||
|
||||
err = security_shm_shmctl(shp, cmd);
|
||||
if (err)
|
||||
goto out_unlock_up;
|
||||
|
||||
if (shp->shm_nattch){
|
||||
shp->shm_flags |= SHM_DEST;
|
||||
/* Do not find it any more */
|
||||
shp->shm_perm.key = IPC_PRIVATE;
|
||||
shm_unlock(shp);
|
||||
} else
|
||||
shm_destroy (shp);
|
||||
up(&shm_ids.sem);
|
||||
goto out;
|
||||
}
|
||||
|
||||
case IPC_SET:
|
||||
{
|
||||
if (copy_shmid_from_user (&setbuf, buf, version)) {
|
||||
err = -EFAULT;
|
||||
goto out;
|
||||
}
|
||||
if ((err = audit_ipc_perms(0, setbuf.uid, setbuf.gid, setbuf.mode)))
|
||||
return err;
|
||||
down(&shm_ids.sem);
|
||||
shp = shm_lock(shmid);
|
||||
err=-EINVAL;
|
||||
if(shp==NULL)
|
||||
goto out_up;
|
||||
err = shm_checkid(shp,shmid);
|
||||
if(err)
|
||||
goto out_unlock_up;
|
||||
err=-EPERM;
|
||||
if (current->euid != shp->shm_perm.uid &&
|
||||
current->euid != shp->shm_perm.cuid &&
|
||||
!capable(CAP_SYS_ADMIN)) {
|
||||
goto out_unlock_up;
|
||||
}
|
||||
|
||||
err = security_shm_shmctl(shp, cmd);
|
||||
if (err)
|
||||
goto out_unlock_up;
|
||||
|
||||
shp->shm_perm.uid = setbuf.uid;
|
||||
shp->shm_perm.gid = setbuf.gid;
|
||||
shp->shm_flags = (shp->shm_flags & ~S_IRWXUGO)
|
||||
| (setbuf.mode & S_IRWXUGO);
|
||||
shp->shm_ctim = get_seconds();
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = 0;
|
||||
out_unlock_up:
|
||||
shm_unlock(shp);
|
||||
out_up:
|
||||
up(&shm_ids.sem);
|
||||
goto out;
|
||||
out_unlock:
|
||||
shm_unlock(shp);
|
||||
out:
|
||||
return err;
|
||||
}
|
||||
|
||||
/*
|
||||
* Fix shmaddr, allocate descriptor, map shm, add attach descriptor to lists.
|
||||
*
|
||||
* NOTE! Despite the name, this is NOT a direct system call entrypoint. The
|
||||
* "raddr" thing points to kernel space, and there has to be a wrapper around
|
||||
* this.
|
||||
*/
|
||||
long do_shmat(int shmid, char __user *shmaddr, int shmflg, ulong *raddr)
|
||||
{
|
||||
struct shmid_kernel *shp;
|
||||
unsigned long addr;
|
||||
unsigned long size;
|
||||
struct file * file;
|
||||
int err;
|
||||
unsigned long flags;
|
||||
unsigned long prot;
|
||||
unsigned long o_flags;
|
||||
int acc_mode;
|
||||
void *user_addr;
|
||||
|
||||
if (shmid < 0) {
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
} else if ((addr = (ulong)shmaddr)) {
|
||||
if (addr & (SHMLBA-1)) {
|
||||
if (shmflg & SHM_RND)
|
||||
addr &= ~(SHMLBA-1); /* round down */
|
||||
else
|
||||
#ifndef __ARCH_FORCE_SHMLBA
|
||||
if (addr & ~PAGE_MASK)
|
||||
#endif
|
||||
return -EINVAL;
|
||||
}
|
||||
flags = MAP_SHARED | MAP_FIXED;
|
||||
} else {
|
||||
if ((shmflg & SHM_REMAP))
|
||||
return -EINVAL;
|
||||
|
||||
flags = MAP_SHARED;
|
||||
}
|
||||
|
||||
if (shmflg & SHM_RDONLY) {
|
||||
prot = PROT_READ;
|
||||
o_flags = O_RDONLY;
|
||||
acc_mode = S_IRUGO;
|
||||
} else {
|
||||
prot = PROT_READ | PROT_WRITE;
|
||||
o_flags = O_RDWR;
|
||||
acc_mode = S_IRUGO | S_IWUGO;
|
||||
}
|
||||
if (shmflg & SHM_EXEC) {
|
||||
prot |= PROT_EXEC;
|
||||
acc_mode |= S_IXUGO;
|
||||
}
|
||||
|
||||
/*
|
||||
* We cannot rely on the fs check since SYSV IPC does have an
|
||||
* additional creator id...
|
||||
*/
|
||||
shp = shm_lock(shmid);
|
||||
if(shp == NULL) {
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
err = shm_checkid(shp,shmid);
|
||||
if (err) {
|
||||
shm_unlock(shp);
|
||||
goto out;
|
||||
}
|
||||
if (ipcperms(&shp->shm_perm, acc_mode)) {
|
||||
shm_unlock(shp);
|
||||
err = -EACCES;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = security_shm_shmat(shp, shmaddr, shmflg);
|
||||
if (err) {
|
||||
shm_unlock(shp);
|
||||
return err;
|
||||
}
|
||||
|
||||
file = shp->shm_file;
|
||||
size = i_size_read(file->f_dentry->d_inode);
|
||||
shp->shm_nattch++;
|
||||
shm_unlock(shp);
|
||||
|
||||
down_write(¤t->mm->mmap_sem);
|
||||
if (addr && !(shmflg & SHM_REMAP)) {
|
||||
user_addr = ERR_PTR(-EINVAL);
|
||||
if (find_vma_intersection(current->mm, addr, addr + size))
|
||||
goto invalid;
|
||||
/*
|
||||
* If shm segment goes below stack, make sure there is some
|
||||
* space left for the stack to grow (at least 4 pages).
|
||||
*/
|
||||
if (addr < current->mm->start_stack &&
|
||||
addr > current->mm->start_stack - size - PAGE_SIZE * 5)
|
||||
goto invalid;
|
||||
}
|
||||
|
||||
user_addr = (void*) do_mmap (file, addr, size, prot, flags, 0);
|
||||
|
||||
invalid:
|
||||
up_write(¤t->mm->mmap_sem);
|
||||
|
||||
down (&shm_ids.sem);
|
||||
if(!(shp = shm_lock(shmid)))
|
||||
BUG();
|
||||
shp->shm_nattch--;
|
||||
if(shp->shm_nattch == 0 &&
|
||||
shp->shm_flags & SHM_DEST)
|
||||
shm_destroy (shp);
|
||||
else
|
||||
shm_unlock(shp);
|
||||
up (&shm_ids.sem);
|
||||
|
||||
*raddr = (unsigned long) user_addr;
|
||||
err = 0;
|
||||
if (IS_ERR(user_addr))
|
||||
err = PTR_ERR(user_addr);
|
||||
out:
|
||||
return err;
|
||||
}
|
||||
|
||||
/*
|
||||
* detach and kill segment if marked destroyed.
|
||||
* The work is done in shm_close.
|
||||
*/
|
||||
asmlinkage long sys_shmdt(char __user *shmaddr)
|
||||
{
|
||||
struct mm_struct *mm = current->mm;
|
||||
struct vm_area_struct *vma, *next;
|
||||
unsigned long addr = (unsigned long)shmaddr;
|
||||
loff_t size = 0;
|
||||
int retval = -EINVAL;
|
||||
|
||||
down_write(&mm->mmap_sem);
|
||||
|
||||
/*
|
||||
* This function tries to be smart and unmap shm segments that
|
||||
* were modified by partial mlock or munmap calls:
|
||||
* - It first determines the size of the shm segment that should be
|
||||
* unmapped: It searches for a vma that is backed by shm and that
|
||||
* started at address shmaddr. It records it's size and then unmaps
|
||||
* it.
|
||||
* - Then it unmaps all shm vmas that started at shmaddr and that
|
||||
* are within the initially determined size.
|
||||
* Errors from do_munmap are ignored: the function only fails if
|
||||
* it's called with invalid parameters or if it's called to unmap
|
||||
* a part of a vma. Both calls in this function are for full vmas,
|
||||
* the parameters are directly copied from the vma itself and always
|
||||
* valid - therefore do_munmap cannot fail. (famous last words?)
|
||||
*/
|
||||
/*
|
||||
* If it had been mremap()'d, the starting address would not
|
||||
* match the usual checks anyway. So assume all vma's are
|
||||
* above the starting address given.
|
||||
*/
|
||||
vma = find_vma(mm, addr);
|
||||
|
||||
while (vma) {
|
||||
next = vma->vm_next;
|
||||
|
||||
/*
|
||||
* Check if the starting address would match, i.e. it's
|
||||
* a fragment created by mprotect() and/or munmap(), or it
|
||||
* otherwise it starts at this address with no hassles.
|
||||
*/
|
||||
if ((vma->vm_ops == &shm_vm_ops || is_vm_hugetlb_page(vma)) &&
|
||||
(vma->vm_start - addr)/PAGE_SIZE == vma->vm_pgoff) {
|
||||
|
||||
|
||||
size = vma->vm_file->f_dentry->d_inode->i_size;
|
||||
do_munmap(mm, vma->vm_start, vma->vm_end - vma->vm_start);
|
||||
/*
|
||||
* We discovered the size of the shm segment, so
|
||||
* break out of here and fall through to the next
|
||||
* loop that uses the size information to stop
|
||||
* searching for matching vma's.
|
||||
*/
|
||||
retval = 0;
|
||||
vma = next;
|
||||
break;
|
||||
}
|
||||
vma = next;
|
||||
}
|
||||
|
||||
/*
|
||||
* We need look no further than the maximum address a fragment
|
||||
* could possibly have landed at. Also cast things to loff_t to
|
||||
* prevent overflows and make comparisions vs. equal-width types.
|
||||
*/
|
||||
while (vma && (loff_t)(vma->vm_end - addr) <= size) {
|
||||
next = vma->vm_next;
|
||||
|
||||
/* finding a matching vma now does not alter retval */
|
||||
if ((vma->vm_ops == &shm_vm_ops || is_vm_hugetlb_page(vma)) &&
|
||||
(vma->vm_start - addr)/PAGE_SIZE == vma->vm_pgoff)
|
||||
|
||||
do_munmap(mm, vma->vm_start, vma->vm_end - vma->vm_start);
|
||||
vma = next;
|
||||
}
|
||||
|
||||
up_write(&mm->mmap_sem);
|
||||
return retval;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PROC_FS
|
||||
static int sysvipc_shm_read_proc(char *buffer, char **start, off_t offset, int length, int *eof, void *data)
|
||||
{
|
||||
off_t pos = 0;
|
||||
off_t begin = 0;
|
||||
int i, len = 0;
|
||||
|
||||
down(&shm_ids.sem);
|
||||
len += sprintf(buffer, " key shmid perms size cpid lpid nattch uid gid cuid cgid atime dtime ctime\n");
|
||||
|
||||
for(i = 0; i <= shm_ids.max_id; i++) {
|
||||
struct shmid_kernel* shp;
|
||||
|
||||
shp = shm_lock(i);
|
||||
if(shp!=NULL) {
|
||||
#define SMALL_STRING "%10d %10d %4o %10u %5u %5u %5d %5u %5u %5u %5u %10lu %10lu %10lu\n"
|
||||
#define BIG_STRING "%10d %10d %4o %21u %5u %5u %5d %5u %5u %5u %5u %10lu %10lu %10lu\n"
|
||||
char *format;
|
||||
|
||||
if (sizeof(size_t) <= sizeof(int))
|
||||
format = SMALL_STRING;
|
||||
else
|
||||
format = BIG_STRING;
|
||||
len += sprintf(buffer + len, format,
|
||||
shp->shm_perm.key,
|
||||
shm_buildid(i, shp->shm_perm.seq),
|
||||
shp->shm_flags,
|
||||
shp->shm_segsz,
|
||||
shp->shm_cprid,
|
||||
shp->shm_lprid,
|
||||
is_file_hugepages(shp->shm_file) ? (file_count(shp->shm_file) - 1) : shp->shm_nattch,
|
||||
shp->shm_perm.uid,
|
||||
shp->shm_perm.gid,
|
||||
shp->shm_perm.cuid,
|
||||
shp->shm_perm.cgid,
|
||||
shp->shm_atim,
|
||||
shp->shm_dtim,
|
||||
shp->shm_ctim);
|
||||
shm_unlock(shp);
|
||||
|
||||
pos += len;
|
||||
if(pos < offset) {
|
||||
len = 0;
|
||||
begin = pos;
|
||||
}
|
||||
if(pos > offset + length)
|
||||
goto done;
|
||||
}
|
||||
}
|
||||
*eof = 1;
|
||||
done:
|
||||
up(&shm_ids.sem);
|
||||
*start = buffer + (offset - begin);
|
||||
len -= (offset - begin);
|
||||
if(len > length)
|
||||
len = length;
|
||||
if(len < 0)
|
||||
len = 0;
|
||||
return len;
|
||||
}
|
||||
#endif
|
580
ipc/util.c
Normal file
580
ipc/util.c
Normal file
@@ -0,0 +1,580 @@
|
||||
/*
|
||||
* linux/ipc/util.c
|
||||
* Copyright (C) 1992 Krishna Balasubramanian
|
||||
*
|
||||
* Sep 1997 - Call suser() last after "normal" permission checks so we
|
||||
* get BSD style process accounting right.
|
||||
* Occurs in several places in the IPC code.
|
||||
* Chris Evans, <chris@ferret.lmh.ox.ac.uk>
|
||||
* Nov 1999 - ipc helper functions, unified SMP locking
|
||||
* Manfred Spraul <manfreds@colorfullife.com>
|
||||
* Oct 2002 - One lock per IPC id. RCU ipc_free for lock-free grow_ary().
|
||||
* Mingming Cao <cmm@us.ibm.com>
|
||||
*/
|
||||
|
||||
#include <linux/config.h>
|
||||
#include <linux/mm.h>
|
||||
#include <linux/shm.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/msg.h>
|
||||
#include <linux/smp_lock.h>
|
||||
#include <linux/vmalloc.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/highuid.h>
|
||||
#include <linux/security.h>
|
||||
#include <linux/rcupdate.h>
|
||||
#include <linux/workqueue.h>
|
||||
|
||||
#include <asm/unistd.h>
|
||||
|
||||
#include "util.h"
|
||||
|
||||
/**
|
||||
* ipc_init - initialise IPC subsystem
|
||||
*
|
||||
* The various system5 IPC resources (semaphores, messages and shared
|
||||
* memory are initialised
|
||||
*/
|
||||
|
||||
static int __init ipc_init(void)
|
||||
{
|
||||
sem_init();
|
||||
msg_init();
|
||||
shm_init();
|
||||
return 0;
|
||||
}
|
||||
__initcall(ipc_init);
|
||||
|
||||
/**
|
||||
* ipc_init_ids - initialise IPC identifiers
|
||||
* @ids: Identifier set
|
||||
* @size: Number of identifiers
|
||||
*
|
||||
* Given a size for the ipc identifier range (limited below IPCMNI)
|
||||
* set up the sequence range to use then allocate and initialise the
|
||||
* array itself.
|
||||
*/
|
||||
|
||||
void __init ipc_init_ids(struct ipc_ids* ids, int size)
|
||||
{
|
||||
int i;
|
||||
sema_init(&ids->sem,1);
|
||||
|
||||
if(size > IPCMNI)
|
||||
size = IPCMNI;
|
||||
ids->in_use = 0;
|
||||
ids->max_id = -1;
|
||||
ids->seq = 0;
|
||||
{
|
||||
int seq_limit = INT_MAX/SEQ_MULTIPLIER;
|
||||
if(seq_limit > USHRT_MAX)
|
||||
ids->seq_max = USHRT_MAX;
|
||||
else
|
||||
ids->seq_max = seq_limit;
|
||||
}
|
||||
|
||||
ids->entries = ipc_rcu_alloc(sizeof(struct kern_ipc_perm *)*size +
|
||||
sizeof(struct ipc_id_ary));
|
||||
|
||||
if(ids->entries == NULL) {
|
||||
printk(KERN_ERR "ipc_init_ids() failed, ipc service disabled.\n");
|
||||
size = 0;
|
||||
ids->entries = &ids->nullentry;
|
||||
}
|
||||
ids->entries->size = size;
|
||||
for(i=0;i<size;i++)
|
||||
ids->entries->p[i] = NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_findkey - find a key in an ipc identifier set
|
||||
* @ids: Identifier set
|
||||
* @key: The key to find
|
||||
*
|
||||
* Requires ipc_ids.sem locked.
|
||||
* Returns the identifier if found or -1 if not.
|
||||
*/
|
||||
|
||||
int ipc_findkey(struct ipc_ids* ids, key_t key)
|
||||
{
|
||||
int id;
|
||||
struct kern_ipc_perm* p;
|
||||
int max_id = ids->max_id;
|
||||
|
||||
/*
|
||||
* rcu_dereference() is not needed here
|
||||
* since ipc_ids.sem is held
|
||||
*/
|
||||
for (id = 0; id <= max_id; id++) {
|
||||
p = ids->entries->p[id];
|
||||
if(p==NULL)
|
||||
continue;
|
||||
if (key == p->key)
|
||||
return id;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
/*
|
||||
* Requires ipc_ids.sem locked
|
||||
*/
|
||||
static int grow_ary(struct ipc_ids* ids, int newsize)
|
||||
{
|
||||
struct ipc_id_ary* new;
|
||||
struct ipc_id_ary* old;
|
||||
int i;
|
||||
int size = ids->entries->size;
|
||||
|
||||
if(newsize > IPCMNI)
|
||||
newsize = IPCMNI;
|
||||
if(newsize <= size)
|
||||
return newsize;
|
||||
|
||||
new = ipc_rcu_alloc(sizeof(struct kern_ipc_perm *)*newsize +
|
||||
sizeof(struct ipc_id_ary));
|
||||
if(new == NULL)
|
||||
return size;
|
||||
new->size = newsize;
|
||||
memcpy(new->p, ids->entries->p, sizeof(struct kern_ipc_perm *)*size +
|
||||
sizeof(struct ipc_id_ary));
|
||||
for(i=size;i<newsize;i++) {
|
||||
new->p[i] = NULL;
|
||||
}
|
||||
old = ids->entries;
|
||||
|
||||
/*
|
||||
* Use rcu_assign_pointer() to make sure the memcpyed contents
|
||||
* of the new array are visible before the new array becomes visible.
|
||||
*/
|
||||
rcu_assign_pointer(ids->entries, new);
|
||||
|
||||
ipc_rcu_putref(old);
|
||||
return newsize;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_addid - add an IPC identifier
|
||||
* @ids: IPC identifier set
|
||||
* @new: new IPC permission set
|
||||
* @size: new size limit for the id array
|
||||
*
|
||||
* Add an entry 'new' to the IPC arrays. The permissions object is
|
||||
* initialised and the first free entry is set up and the id assigned
|
||||
* is returned. The list is returned in a locked state on success.
|
||||
* On failure the list is not locked and -1 is returned.
|
||||
*
|
||||
* Called with ipc_ids.sem held.
|
||||
*/
|
||||
|
||||
int ipc_addid(struct ipc_ids* ids, struct kern_ipc_perm* new, int size)
|
||||
{
|
||||
int id;
|
||||
|
||||
size = grow_ary(ids,size);
|
||||
|
||||
/*
|
||||
* rcu_dereference()() is not needed here since
|
||||
* ipc_ids.sem is held
|
||||
*/
|
||||
for (id = 0; id < size; id++) {
|
||||
if(ids->entries->p[id] == NULL)
|
||||
goto found;
|
||||
}
|
||||
return -1;
|
||||
found:
|
||||
ids->in_use++;
|
||||
if (id > ids->max_id)
|
||||
ids->max_id = id;
|
||||
|
||||
new->cuid = new->uid = current->euid;
|
||||
new->gid = new->cgid = current->egid;
|
||||
|
||||
new->seq = ids->seq++;
|
||||
if(ids->seq > ids->seq_max)
|
||||
ids->seq = 0;
|
||||
|
||||
spin_lock_init(&new->lock);
|
||||
new->deleted = 0;
|
||||
rcu_read_lock();
|
||||
spin_lock(&new->lock);
|
||||
ids->entries->p[id] = new;
|
||||
return id;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_rmid - remove an IPC identifier
|
||||
* @ids: identifier set
|
||||
* @id: Identifier to remove
|
||||
*
|
||||
* The identifier must be valid, and in use. The kernel will panic if
|
||||
* fed an invalid identifier. The entry is removed and internal
|
||||
* variables recomputed. The object associated with the identifier
|
||||
* is returned.
|
||||
* ipc_ids.sem and the spinlock for this ID is hold before this function
|
||||
* is called, and remain locked on the exit.
|
||||
*/
|
||||
|
||||
struct kern_ipc_perm* ipc_rmid(struct ipc_ids* ids, int id)
|
||||
{
|
||||
struct kern_ipc_perm* p;
|
||||
int lid = id % SEQ_MULTIPLIER;
|
||||
if(lid >= ids->entries->size)
|
||||
BUG();
|
||||
|
||||
/*
|
||||
* do not need a rcu_dereference()() here to force ordering
|
||||
* on Alpha, since the ipc_ids.sem is held.
|
||||
*/
|
||||
p = ids->entries->p[lid];
|
||||
ids->entries->p[lid] = NULL;
|
||||
if(p==NULL)
|
||||
BUG();
|
||||
ids->in_use--;
|
||||
|
||||
if (lid == ids->max_id) {
|
||||
do {
|
||||
lid--;
|
||||
if(lid == -1)
|
||||
break;
|
||||
} while (ids->entries->p[lid] == NULL);
|
||||
ids->max_id = lid;
|
||||
}
|
||||
p->deleted = 1;
|
||||
return p;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_alloc - allocate ipc space
|
||||
* @size: size desired
|
||||
*
|
||||
* Allocate memory from the appropriate pools and return a pointer to it.
|
||||
* NULL is returned if the allocation fails
|
||||
*/
|
||||
|
||||
void* ipc_alloc(int size)
|
||||
{
|
||||
void* out;
|
||||
if(size > PAGE_SIZE)
|
||||
out = vmalloc(size);
|
||||
else
|
||||
out = kmalloc(size, GFP_KERNEL);
|
||||
return out;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_free - free ipc space
|
||||
* @ptr: pointer returned by ipc_alloc
|
||||
* @size: size of block
|
||||
*
|
||||
* Free a block created with ipc_alloc. The caller must know the size
|
||||
* used in the allocation call.
|
||||
*/
|
||||
|
||||
void ipc_free(void* ptr, int size)
|
||||
{
|
||||
if(size > PAGE_SIZE)
|
||||
vfree(ptr);
|
||||
else
|
||||
kfree(ptr);
|
||||
}
|
||||
|
||||
/*
|
||||
* rcu allocations:
|
||||
* There are three headers that are prepended to the actual allocation:
|
||||
* - during use: ipc_rcu_hdr.
|
||||
* - during the rcu grace period: ipc_rcu_grace.
|
||||
* - [only if vmalloc]: ipc_rcu_sched.
|
||||
* Their lifetime doesn't overlap, thus the headers share the same memory.
|
||||
* Unlike a normal union, they are right-aligned, thus some container_of
|
||||
* forward/backward casting is necessary:
|
||||
*/
|
||||
struct ipc_rcu_hdr
|
||||
{
|
||||
int refcount;
|
||||
int is_vmalloc;
|
||||
void *data[0];
|
||||
};
|
||||
|
||||
|
||||
struct ipc_rcu_grace
|
||||
{
|
||||
struct rcu_head rcu;
|
||||
/* "void *" makes sure alignment of following data is sane. */
|
||||
void *data[0];
|
||||
};
|
||||
|
||||
struct ipc_rcu_sched
|
||||
{
|
||||
struct work_struct work;
|
||||
/* "void *" makes sure alignment of following data is sane. */
|
||||
void *data[0];
|
||||
};
|
||||
|
||||
#define HDRLEN_KMALLOC (sizeof(struct ipc_rcu_grace) > sizeof(struct ipc_rcu_hdr) ? \
|
||||
sizeof(struct ipc_rcu_grace) : sizeof(struct ipc_rcu_hdr))
|
||||
#define HDRLEN_VMALLOC (sizeof(struct ipc_rcu_sched) > HDRLEN_KMALLOC ? \
|
||||
sizeof(struct ipc_rcu_sched) : HDRLEN_KMALLOC)
|
||||
|
||||
static inline int rcu_use_vmalloc(int size)
|
||||
{
|
||||
/* Too big for a single page? */
|
||||
if (HDRLEN_KMALLOC + size > PAGE_SIZE)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_rcu_alloc - allocate ipc and rcu space
|
||||
* @size: size desired
|
||||
*
|
||||
* Allocate memory for the rcu header structure + the object.
|
||||
* Returns the pointer to the object.
|
||||
* NULL is returned if the allocation fails.
|
||||
*/
|
||||
|
||||
void* ipc_rcu_alloc(int size)
|
||||
{
|
||||
void* out;
|
||||
/*
|
||||
* We prepend the allocation with the rcu struct, and
|
||||
* workqueue if necessary (for vmalloc).
|
||||
*/
|
||||
if (rcu_use_vmalloc(size)) {
|
||||
out = vmalloc(HDRLEN_VMALLOC + size);
|
||||
if (out) {
|
||||
out += HDRLEN_VMALLOC;
|
||||
container_of(out, struct ipc_rcu_hdr, data)->is_vmalloc = 1;
|
||||
container_of(out, struct ipc_rcu_hdr, data)->refcount = 1;
|
||||
}
|
||||
} else {
|
||||
out = kmalloc(HDRLEN_KMALLOC + size, GFP_KERNEL);
|
||||
if (out) {
|
||||
out += HDRLEN_KMALLOC;
|
||||
container_of(out, struct ipc_rcu_hdr, data)->is_vmalloc = 0;
|
||||
container_of(out, struct ipc_rcu_hdr, data)->refcount = 1;
|
||||
}
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
void ipc_rcu_getref(void *ptr)
|
||||
{
|
||||
container_of(ptr, struct ipc_rcu_hdr, data)->refcount++;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_schedule_free - free ipc + rcu space
|
||||
*
|
||||
* Since RCU callback function is called in bh,
|
||||
* we need to defer the vfree to schedule_work
|
||||
*/
|
||||
static void ipc_schedule_free(struct rcu_head *head)
|
||||
{
|
||||
struct ipc_rcu_grace *grace =
|
||||
container_of(head, struct ipc_rcu_grace, rcu);
|
||||
struct ipc_rcu_sched *sched =
|
||||
container_of(&(grace->data[0]), struct ipc_rcu_sched, data[0]);
|
||||
|
||||
INIT_WORK(&sched->work, vfree, sched);
|
||||
schedule_work(&sched->work);
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc_immediate_free - free ipc + rcu space
|
||||
*
|
||||
* Free from the RCU callback context
|
||||
*
|
||||
*/
|
||||
static void ipc_immediate_free(struct rcu_head *head)
|
||||
{
|
||||
struct ipc_rcu_grace *free =
|
||||
container_of(head, struct ipc_rcu_grace, rcu);
|
||||
kfree(free);
|
||||
}
|
||||
|
||||
void ipc_rcu_putref(void *ptr)
|
||||
{
|
||||
if (--container_of(ptr, struct ipc_rcu_hdr, data)->refcount > 0)
|
||||
return;
|
||||
|
||||
if (container_of(ptr, struct ipc_rcu_hdr, data)->is_vmalloc) {
|
||||
call_rcu(&container_of(ptr, struct ipc_rcu_grace, data)->rcu,
|
||||
ipc_schedule_free);
|
||||
} else {
|
||||
call_rcu(&container_of(ptr, struct ipc_rcu_grace, data)->rcu,
|
||||
ipc_immediate_free);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* ipcperms - check IPC permissions
|
||||
* @ipcp: IPC permission set
|
||||
* @flag: desired permission set.
|
||||
*
|
||||
* Check user, group, other permissions for access
|
||||
* to ipc resources. return 0 if allowed
|
||||
*/
|
||||
|
||||
int ipcperms (struct kern_ipc_perm *ipcp, short flag)
|
||||
{ /* flag will most probably be 0 or S_...UGO from <linux/stat.h> */
|
||||
int requested_mode, granted_mode;
|
||||
|
||||
requested_mode = (flag >> 6) | (flag >> 3) | flag;
|
||||
granted_mode = ipcp->mode;
|
||||
if (current->euid == ipcp->cuid || current->euid == ipcp->uid)
|
||||
granted_mode >>= 6;
|
||||
else if (in_group_p(ipcp->cgid) || in_group_p(ipcp->gid))
|
||||
granted_mode >>= 3;
|
||||
/* is there some bit set in requested_mode but not in granted_mode? */
|
||||
if ((requested_mode & ~granted_mode & 0007) &&
|
||||
!capable(CAP_IPC_OWNER))
|
||||
return -1;
|
||||
|
||||
return security_ipc_permission(ipcp, flag);
|
||||
}
|
||||
|
||||
/*
|
||||
* Functions to convert between the kern_ipc_perm structure and the
|
||||
* old/new ipc_perm structures
|
||||
*/
|
||||
|
||||
/**
|
||||
* kernel_to_ipc64_perm - convert kernel ipc permissions to user
|
||||
* @in: kernel permissions
|
||||
* @out: new style IPC permissions
|
||||
*
|
||||
* Turn the kernel object 'in' into a set of permissions descriptions
|
||||
* for returning to userspace (out).
|
||||
*/
|
||||
|
||||
|
||||
void kernel_to_ipc64_perm (struct kern_ipc_perm *in, struct ipc64_perm *out)
|
||||
{
|
||||
out->key = in->key;
|
||||
out->uid = in->uid;
|
||||
out->gid = in->gid;
|
||||
out->cuid = in->cuid;
|
||||
out->cgid = in->cgid;
|
||||
out->mode = in->mode;
|
||||
out->seq = in->seq;
|
||||
}
|
||||
|
||||
/**
|
||||
* ipc64_perm_to_ipc_perm - convert old ipc permissions to new
|
||||
* @in: new style IPC permissions
|
||||
* @out: old style IPC permissions
|
||||
*
|
||||
* Turn the new style permissions object in into a compatibility
|
||||
* object and store it into the 'out' pointer.
|
||||
*/
|
||||
|
||||
void ipc64_perm_to_ipc_perm (struct ipc64_perm *in, struct ipc_perm *out)
|
||||
{
|
||||
out->key = in->key;
|
||||
SET_UID(out->uid, in->uid);
|
||||
SET_GID(out->gid, in->gid);
|
||||
SET_UID(out->cuid, in->cuid);
|
||||
SET_GID(out->cgid, in->cgid);
|
||||
out->mode = in->mode;
|
||||
out->seq = in->seq;
|
||||
}
|
||||
|
||||
/*
|
||||
* So far only shm_get_stat() calls ipc_get() via shm_get(), so ipc_get()
|
||||
* is called with shm_ids.sem locked. Since grow_ary() is also called with
|
||||
* shm_ids.sem down(for Shared Memory), there is no need to add read
|
||||
* barriers here to gurantee the writes in grow_ary() are seen in order
|
||||
* here (for Alpha).
|
||||
*
|
||||
* However ipc_get() itself does not necessary require ipc_ids.sem down. So
|
||||
* if in the future ipc_get() is used by other places without ipc_ids.sem
|
||||
* down, then ipc_get() needs read memery barriers as ipc_lock() does.
|
||||
*/
|
||||
struct kern_ipc_perm* ipc_get(struct ipc_ids* ids, int id)
|
||||
{
|
||||
struct kern_ipc_perm* out;
|
||||
int lid = id % SEQ_MULTIPLIER;
|
||||
if(lid >= ids->entries->size)
|
||||
return NULL;
|
||||
out = ids->entries->p[lid];
|
||||
return out;
|
||||
}
|
||||
|
||||
struct kern_ipc_perm* ipc_lock(struct ipc_ids* ids, int id)
|
||||
{
|
||||
struct kern_ipc_perm* out;
|
||||
int lid = id % SEQ_MULTIPLIER;
|
||||
struct ipc_id_ary* entries;
|
||||
|
||||
rcu_read_lock();
|
||||
entries = rcu_dereference(ids->entries);
|
||||
if(lid >= entries->size) {
|
||||
rcu_read_unlock();
|
||||
return NULL;
|
||||
}
|
||||
out = entries->p[lid];
|
||||
if(out == NULL) {
|
||||
rcu_read_unlock();
|
||||
return NULL;
|
||||
}
|
||||
spin_lock(&out->lock);
|
||||
|
||||
/* ipc_rmid() may have already freed the ID while ipc_lock
|
||||
* was spinning: here verify that the structure is still valid
|
||||
*/
|
||||
if (out->deleted) {
|
||||
spin_unlock(&out->lock);
|
||||
rcu_read_unlock();
|
||||
return NULL;
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
void ipc_lock_by_ptr(struct kern_ipc_perm *perm)
|
||||
{
|
||||
rcu_read_lock();
|
||||
spin_lock(&perm->lock);
|
||||
}
|
||||
|
||||
void ipc_unlock(struct kern_ipc_perm* perm)
|
||||
{
|
||||
spin_unlock(&perm->lock);
|
||||
rcu_read_unlock();
|
||||
}
|
||||
|
||||
int ipc_buildid(struct ipc_ids* ids, int id, int seq)
|
||||
{
|
||||
return SEQ_MULTIPLIER*seq + id;
|
||||
}
|
||||
|
||||
int ipc_checkid(struct ipc_ids* ids, struct kern_ipc_perm* ipcp, int uid)
|
||||
{
|
||||
if(uid/SEQ_MULTIPLIER != ipcp->seq)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef __ARCH_WANT_IPC_PARSE_VERSION
|
||||
|
||||
|
||||
/**
|
||||
* ipc_parse_version - IPC call version
|
||||
* @cmd: pointer to command
|
||||
*
|
||||
* Return IPC_64 for new style IPC and IPC_OLD for old style IPC.
|
||||
* The cmd value is turned from an encoding command and version into
|
||||
* just the command code.
|
||||
*/
|
||||
|
||||
int ipc_parse_version (int *cmd)
|
||||
{
|
||||
if (*cmd & IPC_64) {
|
||||
*cmd ^= IPC_64;
|
||||
return IPC_64;
|
||||
} else {
|
||||
return IPC_OLD;
|
||||
}
|
||||
}
|
||||
|
||||
#endif /* __ARCH_WANT_IPC_PARSE_VERSION */
|
81
ipc/util.h
Normal file
81
ipc/util.h
Normal file
@@ -0,0 +1,81 @@
|
||||
/*
|
||||
* linux/ipc/util.h
|
||||
* Copyright (C) 1999 Christoph Rohland
|
||||
*
|
||||
* ipc helper functions (c) 1999 Manfred Spraul <manfreds@colorfullife.com>
|
||||
*/
|
||||
|
||||
#ifndef _IPC_UTIL_H
|
||||
#define _IPC_UTIL_H
|
||||
|
||||
#define USHRT_MAX 0xffff
|
||||
#define SEQ_MULTIPLIER (IPCMNI)
|
||||
|
||||
void sem_init (void);
|
||||
void msg_init (void);
|
||||
void shm_init (void);
|
||||
|
||||
struct ipc_id_ary {
|
||||
int size;
|
||||
struct kern_ipc_perm *p[0];
|
||||
};
|
||||
|
||||
struct ipc_ids {
|
||||
int in_use;
|
||||
int max_id;
|
||||
unsigned short seq;
|
||||
unsigned short seq_max;
|
||||
struct semaphore sem;
|
||||
struct ipc_id_ary nullentry;
|
||||
struct ipc_id_ary* entries;
|
||||
};
|
||||
|
||||
void __init ipc_init_ids(struct ipc_ids* ids, int size);
|
||||
|
||||
/* must be called with ids->sem acquired.*/
|
||||
int ipc_findkey(struct ipc_ids* ids, key_t key);
|
||||
int ipc_addid(struct ipc_ids* ids, struct kern_ipc_perm* new, int size);
|
||||
|
||||
/* must be called with both locks acquired. */
|
||||
struct kern_ipc_perm* ipc_rmid(struct ipc_ids* ids, int id);
|
||||
|
||||
int ipcperms (struct kern_ipc_perm *ipcp, short flg);
|
||||
|
||||
/* for rare, potentially huge allocations.
|
||||
* both function can sleep
|
||||
*/
|
||||
void* ipc_alloc(int size);
|
||||
void ipc_free(void* ptr, int size);
|
||||
|
||||
/*
|
||||
* For allocation that need to be freed by RCU.
|
||||
* Objects are reference counted, they start with reference count 1.
|
||||
* getref increases the refcount, the putref call that reduces the recount
|
||||
* to 0 schedules the rcu destruction. Caller must guarantee locking.
|
||||
*/
|
||||
void* ipc_rcu_alloc(int size);
|
||||
void ipc_rcu_getref(void *ptr);
|
||||
void ipc_rcu_putref(void *ptr);
|
||||
|
||||
struct kern_ipc_perm* ipc_get(struct ipc_ids* ids, int id);
|
||||
struct kern_ipc_perm* ipc_lock(struct ipc_ids* ids, int id);
|
||||
void ipc_lock_by_ptr(struct kern_ipc_perm *ipcp);
|
||||
void ipc_unlock(struct kern_ipc_perm* perm);
|
||||
int ipc_buildid(struct ipc_ids* ids, int id, int seq);
|
||||
int ipc_checkid(struct ipc_ids* ids, struct kern_ipc_perm* ipcp, int uid);
|
||||
|
||||
void kernel_to_ipc64_perm(struct kern_ipc_perm *in, struct ipc64_perm *out);
|
||||
void ipc64_perm_to_ipc_perm(struct ipc64_perm *in, struct ipc_perm *out);
|
||||
|
||||
#if defined(__ia64__) || defined(__x86_64__) || defined(__hppa__)
|
||||
/* On IA-64, we always use the "64-bit version" of the IPC structures. */
|
||||
# define ipc_parse_version(cmd) IPC_64
|
||||
#else
|
||||
int ipc_parse_version (int *cmd);
|
||||
#endif
|
||||
|
||||
extern void free_msg(struct msg_msg *msg);
|
||||
extern struct msg_msg *load_msg(const void __user *src, int len);
|
||||
extern int store_msg(void __user *dest, struct msg_msg *msg, int len);
|
||||
|
||||
#endif
|
Reference in New Issue
Block a user