2005-04-17 02:20:36 +04:00
/*
* linux / ipc / msg . c
2006-07-30 14:04:11 +04:00
* Copyright ( C ) 1992 Krishna Balasubramanian
2005-04-17 02:20:36 +04:00
*
* 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
2006-01-15 04:43:54 +03:00
* ( c ) 1999 Manfred Spraul < manfred @ colorfullife . com >
2006-04-03 01:07:33 +04:00
*
* support for audit of ipc object properties and permission changes
* Dustin Kirkland < dustin . kirkland @ us . ibm . com >
2005-04-17 02:20:36 +04:00
*/
2006-01-11 23:17:46 +03:00
# include <linux/capability.h>
2005-04-17 02:20:36 +04:00
# 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>
2005-09-07 02:17:10 +04:00
# include <linux/seq_file.h>
2006-03-26 13:37:17 +04:00
# include <linux/mutex.h>
2005-04-17 02:20:36 +04:00
# include <asm/current.h>
# include <asm/uaccess.h>
# include "util.h"
/* sysctl: */
int msg_ctlmax = MSGMAX ;
int msg_ctlmnb = MSGMNB ;
int msg_ctlmni = MSGMNI ;
2006-07-30 14:04:11 +04:00
/*
* one msg_receiver structure for each sleeping receiver :
*/
2005-04-17 02:20:36 +04:00
struct msg_receiver {
2006-07-30 14:04:11 +04:00
struct list_head r_list ;
struct task_struct * r_tsk ;
2005-04-17 02:20:36 +04:00
2006-07-30 14:04:11 +04:00
int r_mode ;
long r_msgtype ;
long r_maxsize ;
2005-04-17 02:20:36 +04:00
2006-07-30 14:04:11 +04:00
volatile struct msg_msg * r_msg ;
2005-04-17 02:20:36 +04:00
} ;
/* one msg_sender for each sleeping sender */
struct msg_sender {
2006-07-30 14:04:11 +04:00
struct list_head list ;
struct task_struct * tsk ;
2005-04-17 02:20:36 +04:00
} ;
# define SEARCH_ANY 1
# define SEARCH_EQUAL 2
# define SEARCH_NOTEQUAL 3
# define SEARCH_LESSEQUAL 4
2006-07-30 14:04:11 +04:00
static atomic_t msg_bytes = ATOMIC_INIT ( 0 ) ;
static atomic_t msg_hdrs = ATOMIC_INIT ( 0 ) ;
2005-04-17 02:20:36 +04:00
static struct ipc_ids msg_ids ;
2006-07-30 14:04:11 +04:00
# 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)
2005-04-17 02:20:36 +04:00
2006-07-30 14:04:11 +04:00
static void freeque ( struct msg_queue * msq , int id ) ;
static int newque ( key_t key , int msgflg ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_PROC_FS
2005-09-07 02:17:10 +04:00
static int sysvipc_msg_proc_show ( struct seq_file * s , void * it ) ;
2005-04-17 02:20:36 +04:00
# endif
2006-07-30 14:04:11 +04:00
void __init msg_init ( void )
2005-04-17 02:20:36 +04:00
{
2006-07-30 14:04:11 +04:00
ipc_init_ids ( & msg_ids , msg_ctlmni ) ;
2005-09-07 02:17:10 +04:00
ipc_init_proc_interface ( " sysvipc/msg " ,
" key msqid perms cbytes qnum lspid lrpid uid gid cuid cgid stime rtime ctime \n " ,
& msg_ids ,
sysvipc_msg_proc_show ) ;
2005-04-17 02:20:36 +04:00
}
2006-07-30 14:04:11 +04:00
static int newque ( key_t key , int msgflg )
2005-04-17 02:20:36 +04:00
{
struct msg_queue * msq ;
2006-07-30 14:04:11 +04:00
int id , retval ;
2005-04-17 02:20:36 +04:00
2006-07-30 14:04:11 +04:00
msq = ipc_rcu_alloc ( sizeof ( * msq ) ) ;
if ( ! msq )
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
2006-07-30 14:04:11 +04:00
msq - > q_perm . mode = msgflg & S_IRWXUGO ;
2005-04-17 02:20:36 +04:00
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 ) ;
2006-07-30 14:04:11 +04:00
if ( id = = - 1 ) {
2005-04-17 02:20:36 +04:00
security_msg_queue_free ( msq ) ;
ipc_rcu_putref ( msq ) ;
return - ENOSPC ;
}
2006-07-30 14:04:11 +04:00
msq - > q_id = msg_buildid ( id , msq - > q_perm . seq ) ;
2005-04-17 02:20:36 +04:00
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 ) ;
2005-09-07 02:17:10 +04:00
return msq - > q_id ;
2005-04-17 02:20:36 +04:00
}
2006-07-30 14:04:11 +04:00
static inline void ss_add ( struct msg_queue * msq , struct msg_sender * mss )
2005-04-17 02:20:36 +04:00
{
2006-07-30 14:04:11 +04:00
mss - > tsk = current ;
current - > state = TASK_INTERRUPTIBLE ;
list_add_tail ( & mss - > list , & msq - > q_senders ) ;
2005-04-17 02:20:36 +04:00
}
2006-07-30 14:04:11 +04:00
static inline void ss_del ( struct msg_sender * mss )
2005-04-17 02:20:36 +04:00
{
2006-07-30 14:04:11 +04:00
if ( mss - > list . next ! = NULL )
2005-04-17 02:20:36 +04:00
list_del ( & mss - > list ) ;
}
2006-07-30 14:04:11 +04:00
static void ss_wakeup ( struct list_head * h , int kill )
2005-04-17 02:20:36 +04:00
{
struct list_head * tmp ;
tmp = h - > next ;
while ( tmp ! = h ) {
2006-07-30 14:04:11 +04:00
struct msg_sender * mss ;
mss = list_entry ( tmp , struct msg_sender , list ) ;
2005-04-17 02:20:36 +04:00
tmp = tmp - > next ;
2006-07-30 14:04:11 +04:00
if ( kill )
mss - > list . next = NULL ;
2005-04-17 02:20:36 +04:00
wake_up_process ( mss - > tsk ) ;
}
}
2006-07-30 14:04:11 +04:00
static void expunge_all ( struct msg_queue * msq , int res )
2005-04-17 02:20:36 +04:00
{
struct list_head * tmp ;
tmp = msq - > q_receivers . next ;
while ( tmp ! = & msq - > q_receivers ) {
2006-07-30 14:04:11 +04:00
struct msg_receiver * msr ;
msr = list_entry ( tmp , struct msg_receiver , r_list ) ;
2005-04-17 02:20:36 +04:00
tmp = tmp - > next ;
msr - > r_msg = NULL ;
wake_up_process ( msr - > r_tsk ) ;
smp_mb ( ) ;
msr - > r_msg = ERR_PTR ( res ) ;
}
}
2006-07-30 14:04:11 +04:00
/*
* freeque ( ) wakes up waiters on the sender and receiver waiting queue ,
* removes the message queue from message queue ID
2005-04-17 02:20:36 +04:00
* array , and cleans up all the messages associated with this queue .
*
2006-03-26 13:37:17 +04:00
* msg_ids . mutex and the spinlock for this message queue is hold
* before freeque ( ) is called . msg_ids . mutex remains locked on exit .
2005-04-17 02:20:36 +04:00
*/
2006-07-30 14:04:11 +04:00
static void freeque ( struct msg_queue * msq , int id )
2005-04-17 02:20:36 +04:00
{
struct list_head * tmp ;
2006-07-30 14:04:11 +04:00
expunge_all ( msq , - EIDRM ) ;
ss_wakeup ( & msq - > q_senders , 1 ) ;
2005-04-17 02:20:36 +04:00
msq = msg_rmid ( id ) ;
msg_unlock ( msq ) ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
tmp = msq - > q_messages . next ;
2006-07-30 14:04:11 +04:00
while ( tmp ! = & msq - > q_messages ) {
struct msg_msg * msg = list_entry ( tmp , struct msg_msg , m_list ) ;
2005-04-17 02:20:36 +04:00
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 ) ;
}
2006-07-30 14:04:11 +04:00
asmlinkage long sys_msgget ( key_t key , int msgflg )
2005-04-17 02:20:36 +04:00
{
struct msg_queue * msq ;
2006-07-30 14:04:11 +04:00
int id , ret = - EPERM ;
2005-04-17 02:20:36 +04:00
2006-03-26 13:37:17 +04:00
mutex_lock ( & msg_ids . mutex ) ;
2005-04-17 02:20:36 +04:00
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 ) ;
2006-07-30 14:04:11 +04:00
BUG_ON ( msq = = NULL ) ;
2005-04-17 02:20:36 +04:00
if ( ipcperms ( & msq - > q_perm , msgflg ) )
ret = - EACCES ;
else {
int qid = msg_buildid ( id , msq - > q_perm . seq ) ;
2006-07-30 14:04:11 +04:00
ret = security_msg_queue_associate ( msq , msgflg ) ;
2005-04-17 02:20:36 +04:00
if ( ! ret )
ret = qid ;
}
msg_unlock ( msq ) ;
}
2006-03-26 13:37:17 +04:00
mutex_unlock ( & msg_ids . mutex ) ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
return ret ;
}
2006-07-30 14:04:11 +04:00
static inline unsigned long
copy_msqid_to_user ( void __user * buf , struct msqid64_ds * in , int version )
2005-04-17 02:20:36 +04:00
{
switch ( version ) {
case IPC_64 :
2006-07-30 14:04:11 +04:00
return copy_to_user ( buf , in , sizeof ( * in ) ) ;
2005-04-17 02:20:36 +04:00
case IPC_OLD :
2006-07-30 14:04:11 +04:00
{
2005-04-17 02:20:36 +04:00
struct msqid_ds out ;
2006-07-30 14:04:11 +04:00
memset ( & out , 0 , sizeof ( out ) ) ;
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
if ( in - > msg_cbytes > USHRT_MAX )
2005-04-17 02:20:36 +04:00
out . msg_cbytes = USHRT_MAX ;
else
out . msg_cbytes = in - > msg_cbytes ;
out . msg_lcbytes = in - > msg_cbytes ;
2006-07-30 14:04:11 +04:00
if ( in - > msg_qnum > USHRT_MAX )
2005-04-17 02:20:36 +04:00
out . msg_qnum = USHRT_MAX ;
else
out . msg_qnum = in - > msg_qnum ;
2006-07-30 14:04:11 +04:00
if ( in - > msg_qbytes > USHRT_MAX )
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
return copy_to_user ( buf , & out , sizeof ( out ) ) ;
}
2005-04-17 02:20:36 +04:00
default :
return - EINVAL ;
}
}
struct msq_setbuf {
unsigned long qbytes ;
uid_t uid ;
gid_t gid ;
mode_t mode ;
} ;
2006-07-30 14:04:11 +04:00
static inline unsigned long
copy_msqid_from_user ( struct msq_setbuf * out , void __user * buf , int version )
2005-04-17 02:20:36 +04:00
{
switch ( version ) {
case IPC_64 :
2006-07-30 14:04:11 +04:00
{
2005-04-17 02:20:36 +04:00
struct msqid64_ds tbuf ;
2006-07-30 14:04:11 +04:00
if ( copy_from_user ( & tbuf , buf , sizeof ( tbuf ) ) )
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
}
2005-04-17 02:20:36 +04:00
case IPC_OLD :
2006-07-30 14:04:11 +04:00
{
2005-04-17 02:20:36 +04:00
struct msqid_ds tbuf_old ;
2006-07-30 14:04:11 +04:00
if ( copy_from_user ( & tbuf_old , buf , sizeof ( tbuf_old ) ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
out - > uid = tbuf_old . msg_perm . uid ;
out - > gid = tbuf_old . msg_perm . gid ;
out - > mode = tbuf_old . msg_perm . mode ;
2006-07-30 14:04:11 +04:00
if ( tbuf_old . msg_qbytes = = 0 )
2005-04-17 02:20:36 +04:00
out - > qbytes = tbuf_old . msg_lqbytes ;
else
out - > qbytes = tbuf_old . msg_qbytes ;
return 0 ;
2006-07-30 14:04:11 +04:00
}
2005-04-17 02:20:36 +04:00
default :
return - EINVAL ;
}
}
2006-07-30 14:04:11 +04:00
asmlinkage long sys_msgctl ( int msqid , int cmd , struct msqid_ds __user * buf )
2005-04-17 02:20:36 +04:00
{
struct kern_ipc_perm * ipcp ;
2006-07-30 14:04:11 +04:00
struct msq_setbuf setbuf ;
struct msg_queue * msq ;
int err , version ;
2005-04-17 02:20:36 +04:00
if ( msqid < 0 | | cmd < 0 )
return - EINVAL ;
version = ipc_parse_version ( & cmd ) ;
switch ( cmd ) {
2006-07-30 14:04:11 +04:00
case IPC_INFO :
case MSG_INFO :
{
2005-04-17 02:20:36 +04:00
struct msginfo msginfo ;
int max_id ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
if ( ! buf )
return - EFAULT ;
2006-07-30 14:04:11 +04:00
/*
* We must not return kernel stack data .
2005-04-17 02:20:36 +04:00
* due to padding , it ' s not enough
* to set all member fields .
*/
err = security_msg_queue_msgctl ( NULL , cmd ) ;
if ( err )
return err ;
2006-07-30 14:04:11 +04:00
memset ( & msginfo , 0 , sizeof ( msginfo ) ) ;
2005-04-17 02:20:36 +04:00
msginfo . msgmni = msg_ctlmni ;
msginfo . msgmax = msg_ctlmax ;
msginfo . msgmnb = msg_ctlmnb ;
msginfo . msgssz = MSGSSZ ;
msginfo . msgseg = MSGSEG ;
2006-03-26 13:37:17 +04:00
mutex_lock ( & msg_ids . mutex ) ;
2005-04-17 02:20:36 +04:00
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 ;
2006-03-26 13:37:17 +04:00
mutex_unlock ( & msg_ids . mutex ) ;
2006-07-30 14:04:11 +04:00
if ( copy_to_user ( buf , & msginfo , sizeof ( struct msginfo ) ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
2006-07-30 14:04:11 +04:00
return ( max_id < 0 ) ? 0 : max_id ;
2005-04-17 02:20:36 +04:00
}
case MSG_STAT :
case IPC_STAT :
{
struct msqid64_ds tbuf ;
int success_return ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
if ( ! buf )
return - EFAULT ;
2006-07-30 14:04:11 +04:00
if ( cmd = = MSG_STAT & & msqid > = msg_ids . entries - > size )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
2006-07-30 14:04:11 +04:00
memset ( & tbuf , 0 , sizeof ( tbuf ) ) ;
2005-04-17 02:20:36 +04:00
msq = msg_lock ( msqid ) ;
if ( msq = = NULL )
return - EINVAL ;
2006-07-30 14:04:11 +04:00
if ( cmd = = MSG_STAT ) {
2005-04-17 02:20:36 +04:00
success_return = msg_buildid ( msqid , msq - > q_perm . seq ) ;
} else {
err = - EIDRM ;
2006-07-30 14:04:11 +04:00
if ( msg_checkid ( msq , msqid ) )
2005-04-17 02:20:36 +04:00
goto out_unlock ;
success_return = 0 ;
}
err = - EACCES ;
2006-07-30 14:04:11 +04:00
if ( ipcperms ( & msq - > q_perm , S_IRUGO ) )
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
if ( copy_msqid_from_user ( & setbuf , buf , version ) )
2005-04-17 02:20:36 +04:00
return - EFAULT ;
break ;
case IPC_RMID :
break ;
default :
return - EINVAL ;
}
2006-03-26 13:37:17 +04:00
mutex_lock ( & msg_ids . mutex ) ;
2005-04-17 02:20:36 +04:00
msq = msg_lock ( msqid ) ;
2006-07-30 14:04:11 +04:00
err = - EINVAL ;
2005-04-17 02:20:36 +04:00
if ( msq = = NULL )
goto out_up ;
err = - EIDRM ;
2006-07-30 14:04:11 +04:00
if ( msg_checkid ( msq , msqid ) )
2005-04-17 02:20:36 +04:00
goto out_unlock_up ;
ipcp = & msq - > q_perm ;
2006-04-03 01:07:33 +04:00
err = audit_ipc_obj ( ipcp ) ;
if ( err )
goto out_unlock_up ;
2006-05-17 06:03:48 +04:00
if ( cmd = = IPC_SET ) {
2006-07-30 14:04:11 +04:00
err = audit_ipc_set_perm ( setbuf . qbytes , setbuf . uid , setbuf . gid ,
setbuf . mode ) ;
2006-05-17 06:03:48 +04:00
if ( err )
goto out_unlock_up ;
}
2006-04-03 01:07:33 +04:00
2005-04-17 02:20:36 +04:00
err = - EPERM ;
2006-07-30 14:04:11 +04:00
if ( current - > euid ! = ipcp - > cuid & &
2005-04-17 02:20:36 +04:00
current - > euid ! = ipcp - > uid & & ! capable ( CAP_SYS_ADMIN ) )
2006-07-30 14:04:11 +04:00
/* We _could_ check for CAP_CHOWN above, but we don't */
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
ipcp - > mode = ( ipcp - > mode & ~ S_IRWXUGO ) |
( S_IRWXUGO & setbuf . mode ) ;
2005-04-17 02:20:36 +04:00
msq - > q_ctime = get_seconds ( ) ;
/* sleeping receivers might be excluded by
* stricter permissions .
*/
2006-07-30 14:04:11 +04:00
expunge_all ( msq , - EAGAIN ) ;
2005-04-17 02:20:36 +04:00
/* sleeping senders might be able to send
* due to a larger queue size .
*/
2006-07-30 14:04:11 +04:00
ss_wakeup ( & msq - > q_senders , 0 ) ;
2005-04-17 02:20:36 +04:00
msg_unlock ( msq ) ;
break ;
}
case IPC_RMID :
2006-07-30 14:04:11 +04:00
freeque ( msq , msqid ) ;
2005-04-17 02:20:36 +04:00
break ;
}
err = 0 ;
out_up :
2006-03-26 13:37:17 +04:00
mutex_unlock ( & msg_ids . mutex ) ;
2005-04-17 02:20:36 +04:00
return err ;
out_unlock_up :
msg_unlock ( msq ) ;
goto out_up ;
out_unlock :
msg_unlock ( msq ) ;
return err ;
}
2006-07-30 14:04:11 +04:00
static int testmsg ( struct msg_msg * msg , long type , int mode )
2005-04-17 02:20:36 +04:00
{
switch ( mode )
{
case SEARCH_ANY :
return 1 ;
case SEARCH_LESSEQUAL :
2006-07-30 14:04:11 +04:00
if ( msg - > m_type < = type )
2005-04-17 02:20:36 +04:00
return 1 ;
break ;
case SEARCH_EQUAL :
2006-07-30 14:04:11 +04:00
if ( msg - > m_type = = type )
2005-04-17 02:20:36 +04:00
return 1 ;
break ;
case SEARCH_NOTEQUAL :
2006-07-30 14:04:11 +04:00
if ( msg - > m_type ! = type )
2005-04-17 02:20:36 +04:00
return 1 ;
break ;
}
return 0 ;
}
2006-07-30 14:04:11 +04:00
static inline int pipelined_send ( struct msg_queue * msq , struct msg_msg * msg )
2005-04-17 02:20:36 +04:00
{
2006-07-30 14:04:11 +04:00
struct list_head * tmp ;
2005-04-17 02:20:36 +04:00
tmp = msq - > q_receivers . next ;
while ( tmp ! = & msq - > q_receivers ) {
2006-07-30 14:04:11 +04:00
struct msg_receiver * msr ;
msr = list_entry ( tmp , struct msg_receiver , r_list ) ;
2005-04-17 02:20:36 +04:00
tmp = tmp - > next ;
2006-07-30 14:04:11 +04:00
if ( testmsg ( msg , msr - > r_msgtype , msr - > r_mode ) & &
! security_msg_queue_msgrcv ( msq , msg , msr - > r_tsk ,
msr - > r_msgtype , msr - > r_mode ) ) {
2005-04-17 02:20:36 +04:00
list_del ( & msr - > r_list ) ;
2006-07-30 14:04:11 +04:00
if ( msr - > r_maxsize < msg - > m_ts ) {
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
return 1 ;
}
}
}
return 0 ;
}
2006-07-30 14:04:11 +04:00
asmlinkage long
sys_msgsnd ( int msqid , struct msgbuf __user * msgp , size_t msgsz , int msgflg )
2005-04-17 02:20:36 +04:00
{
struct msg_queue * msq ;
struct msg_msg * msg ;
long mtype ;
int err ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
if ( msgsz > msg_ctlmax | | ( long ) msgsz < 0 | | msqid < 0 )
return - EINVAL ;
if ( get_user ( mtype , & msgp - > mtype ) )
2006-07-30 14:04:11 +04:00
return - EFAULT ;
2005-04-17 02:20:36 +04:00
if ( mtype < 1 )
return - EINVAL ;
msg = load_msg ( msgp - > mtext , msgsz ) ;
2006-07-30 14:04:11 +04:00
if ( IS_ERR ( msg ) )
2005-04-17 02:20:36 +04:00
return PTR_ERR ( msg ) ;
msg - > m_type = mtype ;
msg - > m_ts = msgsz ;
msq = msg_lock ( msqid ) ;
2006-07-30 14:04:11 +04:00
err = - EINVAL ;
if ( msq = = NULL )
2005-04-17 02:20:36 +04:00
goto out_free ;
err = - EIDRM ;
2006-07-30 14:04:11 +04:00
if ( msg_checkid ( msq , msqid ) )
2005-04-17 02:20:36 +04:00
goto out_unlock_free ;
for ( ; ; ) {
struct msg_sender s ;
2006-07-30 14:04:11 +04:00
err = - EACCES ;
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
if ( msgsz + msq - > q_cbytes < = msq - > q_qbytes & &
2005-04-17 02:20:36 +04:00
1 + msq - > q_qnum < = msq - > q_qbytes ) {
break ;
}
/* queue full, wait: */
2006-07-30 14:04:11 +04:00
if ( msgflg & IPC_NOWAIT ) {
err = - EAGAIN ;
2005-04-17 02:20:36 +04:00
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 ) ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
if ( signal_pending ( current ) ) {
2006-07-30 14:04:11 +04:00
err = - ERESTARTNOHAND ;
2005-04-17 02:20:36 +04:00
goto out_unlock_free ;
}
}
msq - > q_lspid = current - > tgid ;
msq - > q_stime = get_seconds ( ) ;
2006-07-30 14:04:11 +04:00
if ( ! pipelined_send ( msq , msg ) ) {
2005-04-17 02:20:36 +04:00
/* noone is waiting for this message, enqueue it */
2006-07-30 14:04:11 +04:00
list_add_tail ( & msg - > m_list , & msq - > q_messages ) ;
2005-04-17 02:20:36 +04:00
msq - > q_cbytes + = msgsz ;
msq - > q_qnum + + ;
2006-07-30 14:04:11 +04:00
atomic_add ( msgsz , & msg_bytes ) ;
2005-04-17 02:20:36 +04:00
atomic_inc ( & msg_hdrs ) ;
}
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
err = 0 ;
msg = NULL ;
out_unlock_free :
msg_unlock ( msq ) ;
out_free :
2006-07-30 14:04:11 +04:00
if ( msg ! = NULL )
2005-04-17 02:20:36 +04:00
free_msg ( msg ) ;
return err ;
}
2006-07-30 14:04:11 +04:00
static inline int convert_mode ( long * msgtyp , int msgflg )
2005-04-17 02:20:36 +04:00
{
2006-07-30 14:04:11 +04:00
/*
2005-04-17 02:20:36 +04:00
* find message of correct type .
* msgtyp = 0 = > get first .
* msgtyp > 0 = > get first message of matching type .
2006-07-30 14:04:11 +04:00
* msgtyp < 0 = > get message with least type must be < abs ( msgtype ) .
2005-04-17 02:20:36 +04:00
*/
2006-07-30 14:04:11 +04:00
if ( * msgtyp = = 0 )
2005-04-17 02:20:36 +04:00
return SEARCH_ANY ;
2006-07-30 14:04:11 +04:00
if ( * msgtyp < 0 ) {
* msgtyp = - * msgtyp ;
2005-04-17 02:20:36 +04:00
return SEARCH_LESSEQUAL ;
}
2006-07-30 14:04:11 +04:00
if ( msgflg & MSG_EXCEPT )
2005-04-17 02:20:36 +04:00
return SEARCH_NOTEQUAL ;
return SEARCH_EQUAL ;
}
2006-07-30 14:04:11 +04:00
asmlinkage long sys_msgrcv ( int msqid , struct msgbuf __user * msgp , size_t msgsz ,
long msgtyp , int msgflg )
2005-04-17 02:20:36 +04:00
{
struct msg_queue * msq ;
struct msg_msg * msg ;
int mode ;
if ( msqid < 0 | | ( long ) msgsz < 0 )
return - EINVAL ;
2006-07-30 14:04:11 +04:00
mode = convert_mode ( & msgtyp , msgflg ) ;
2005-04-17 02:20:36 +04:00
msq = msg_lock ( msqid ) ;
2006-07-30 14:04:11 +04:00
if ( msq = = NULL )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
msg = ERR_PTR ( - EIDRM ) ;
2006-07-30 14:04:11 +04:00
if ( msg_checkid ( msq , msqid ) )
2005-04-17 02:20:36 +04:00
goto out_unlock ;
for ( ; ; ) {
struct msg_receiver msr_d ;
2006-07-30 14:04:11 +04:00
struct list_head * tmp ;
2005-04-17 02:20:36 +04:00
msg = ERR_PTR ( - EACCES ) ;
2006-07-30 14:04:11 +04:00
if ( ipcperms ( & msq - > q_perm , S_IRUGO ) )
2005-04-17 02:20:36 +04:00
goto out_unlock ;
msg = ERR_PTR ( - EAGAIN ) ;
tmp = msq - > q_messages . next ;
while ( tmp ! = & msq - > q_messages ) {
struct msg_msg * walk_msg ;
2006-07-30 14:04:11 +04:00
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 ) ) {
2005-04-17 02:20:36 +04:00
msg = walk_msg ;
2006-07-30 14:04:11 +04:00
if ( mode = = SEARCH_LESSEQUAL & &
walk_msg - > m_type ! = 1 ) {
msg = walk_msg ;
msgtyp = walk_msg - > m_type - 1 ;
2005-04-17 02:20:36 +04:00
} else {
2006-07-30 14:04:11 +04:00
msg = walk_msg ;
2005-04-17 02:20:36 +04:00
break ;
}
}
tmp = tmp - > next ;
}
2006-07-30 14:04:11 +04:00
if ( ! IS_ERR ( msg ) ) {
/*
* Found a suitable message .
* Unlink it from the queue .
*/
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
atomic_sub ( msg - > m_ts , & msg_bytes ) ;
2005-04-17 02:20:36 +04:00
atomic_dec ( & msg_hdrs ) ;
2006-07-30 14:04:11 +04:00
ss_wakeup ( & msq - > q_senders , 0 ) ;
2005-04-17 02:20:36 +04:00
msg_unlock ( msq ) ;
break ;
}
/* No message waiting. Wait for a message */
if ( msgflg & IPC_NOWAIT ) {
msg = ERR_PTR ( - ENOMSG ) ;
goto out_unlock ;
}
2006-07-30 14:04:11 +04:00
list_add_tail ( & msr_d . r_list , & msq - > q_receivers ) ;
2005-04-17 02:20:36 +04:00
msr_d . r_tsk = current ;
msr_d . r_msgtype = msgtyp ;
msr_d . r_mode = mode ;
2006-07-30 14:04:11 +04:00
if ( msgflg & MSG_NOERROR )
2005-04-17 02:20:36 +04:00
msr_d . r_maxsize = INT_MAX ;
2006-07-30 14:04:11 +04:00
else
2005-04-17 02:20:36 +04:00
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 .
*/
2006-07-30 14:04:11 +04:00
msg = ( struct msg_msg * ) msr_d . r_msg ;
2005-04-17 02:20:36 +04:00
while ( msg = = NULL ) {
cpu_relax ( ) ;
2006-07-30 14:04:11 +04:00
msg = ( struct msg_msg * ) msr_d . r_msg ;
2005-04-17 02:20:36 +04:00
}
/* Lockless receive, part 3:
* If there is a message or an error then accept it without
* locking .
*/
2006-07-30 14:04:11 +04:00
if ( msg ! = ERR_PTR ( - EAGAIN ) ) {
2005-04-17 02:20:36 +04:00
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 ;
2006-07-30 14:04:11 +04:00
if ( msg ! = ERR_PTR ( - EAGAIN ) )
2005-04-17 02:20:36 +04:00
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 ) )
2006-07-30 14:04:11 +04:00
return PTR_ERR ( msg ) ;
2005-04-17 02:20:36 +04:00
msgsz = ( msgsz > msg - > m_ts ) ? msg - > m_ts : msgsz ;
if ( put_user ( msg - > m_type , & msgp - > mtype ) | |
store_msg ( msgp - > mtext , msg , msgsz ) ) {
2006-07-30 14:04:11 +04:00
msgsz = - EFAULT ;
2005-04-17 02:20:36 +04:00
}
free_msg ( msg ) ;
2006-07-30 14:04:11 +04:00
2005-04-17 02:20:36 +04:00
return msgsz ;
}
# ifdef CONFIG_PROC_FS
2005-09-07 02:17:10 +04:00
static int sysvipc_msg_proc_show ( struct seq_file * s , void * it )
2005-04-17 02:20:36 +04:00
{
2005-09-07 02:17:10 +04:00
struct msg_queue * msq = it ;
return seq_printf ( s ,
2006-07-30 14:04:11 +04:00
" %10d %10d %4o %10lu %10lu %5u %5u %5u %5u %5u %5u %10lu %10lu %10lu \n " ,
msq - > q_perm . key ,
msq - > q_id ,
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 ) ;
2005-04-17 02:20:36 +04:00
}
# endif