summaryrefslogtreecommitdiff
path: root/sys/kern/sysv_msg.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/kern/sysv_msg.c')
-rw-r--r--sys/kern/sysv_msg.c34
1 files changed, 26 insertions, 8 deletions
diff --git a/sys/kern/sysv_msg.c b/sys/kern/sysv_msg.c
index a2d0da4f89c..88efdb47f06 100644
--- a/sys/kern/sysv_msg.c
+++ b/sys/kern/sysv_msg.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: sysv_msg.c,v 1.24 2011/05/20 16:06:25 blambert Exp $ */
+/* $OpenBSD: sysv_msg.c,v 1.25 2012/02/05 18:12:10 blambert Exp $ */
/* $NetBSD: sysv_msg.c,v 1.19 1996/02/09 19:00:18 christos Exp $ */
/*
* Copyright (c) 2009 Bret S. Lambert <blambert@openbsd.org>
@@ -230,7 +230,7 @@ again:
goto again;
found:
- *retval = que->que_id;
+ *retval = IXSEQ_TO_IPCID(que->que_ix, que->msqid_ds.msg_perm);
return (error);
}
@@ -387,7 +387,8 @@ out:
struct que *
que_create(key_t key, struct ucred *cred, int mode)
{
- struct que *que;
+ struct que *que, *que2;
+ int nextix = 1;
que = malloc(sizeof(*que), M_TEMP, M_WAIT|M_ZERO);
@@ -397,6 +398,14 @@ que_create(key_t key, struct ucred *cred, int mode)
return (NULL);
}
+ /* find next available "index" */
+ TAILQ_FOREACH(que2, &msg_queues, que_next) {
+ if (nextix < que2->que_ix)
+ break;
+ nextix = que2->que_ix + 1;
+ }
+ que->que_ix = nextix;
+
que->msqid_ds.msg_perm.key = key;
que->msqid_ds.msg_perm.cuid = cred->cr_uid;
que->msqid_ds.msg_perm.uid = cred->cr_uid;
@@ -409,7 +418,11 @@ que_create(key_t key, struct ucred *cred, int mode)
TAILQ_INIT(&que->que_msgs);
- TAILQ_INSERT_TAIL(&msg_queues, que, que_next);
+ /* keep queues in "index" order */
+ if (que2)
+ TAILQ_INSERT_BEFORE(que2, que, que_next);
+ else
+ TAILQ_INSERT_TAIL(&msg_queues, que, que_next);
num_ques++;
return (que);
@@ -421,7 +434,7 @@ que_lookup(int id)
struct que *que;
TAILQ_FOREACH(que, &msg_queues, que_next)
- if (que->que_id == id)
+ if (que->que_ix == IPCID_TO_IX(id))
break;
/* don't return queues marked for removal */
@@ -511,7 +524,7 @@ msg_lookup(struct que *que, int msgtyp)
/*
* Three different matches are performed based on the value of msgtyp:
* 1) msgtyp > 0 => match exactly
- * 2> msgtyp = 0 => match any
+ * 2) msgtyp = 0 => match any
* 3) msgtyp < 0 => match any up to absolute value of msgtyp
*/
TAILQ_FOREACH(msg, &que->que_msgs, msg_next)
@@ -647,7 +660,7 @@ sysctl_sysvmsg(int *name, u_int namelen, void *where, size_t *sizep)
struct msg_sysctl_info *info;
struct que *que;
size_t infolen;
- int error, i = 0;
+ int error;
switch (*name) {
case KERN_SYSVIPC_MSG_INFO:
@@ -693,8 +706,13 @@ sysctl_sysvmsg(int *name, u_int namelen, void *where, size_t *sizep)
bcopy(&msginfo, &info->msginfo, sizeof(struct msginfo));
+ /*
+ * Special case #3: the previous array-based implementation
+ * exported the array indices and userland has come to rely
+ * upon these indices, so keep behavior consisitent.
+ */
TAILQ_FOREACH(que, &msg_queues, que_next)
- bcopy(&que->msqid_ds, &info->msgids[i++],
+ bcopy(&que->msqid_ds, &info->msgids[que->que_ix],
sizeof(struct msqid_ds));
error = copyout(info, where, infolen);