1 // SPDX-License-Identifier: GPL-2.0
9 #include "../kselftest.h"
11 #define MAX_MSG_SIZE 32
16 char mtext
[MAX_MSG_SIZE
];
19 #define TEST_STRING "Test sysv5 msg"
22 #define ANOTHER_TEST_STRING "Yet another test sysv5 msg"
23 #define ANOTHER_MSG_TYPE 26538
31 struct msg1
*messages
;
34 int restore_queue(struct msgque_data
*msgque
)
39 fd
= open("/proc/sys/kernel/msg_next_id", O_WRONLY
);
41 printf("Failed to open /proc/sys/kernel/msg_next_id\n");
44 sprintf(buf
, "%d", msgque
->msq_id
);
46 ret
= write(fd
, buf
, strlen(buf
));
47 if (ret
!= strlen(buf
)) {
48 printf("Failed to write to /proc/sys/kernel/msg_next_id\n");
52 id
= msgget(msgque
->key
, msgque
->mode
| IPC_CREAT
| IPC_EXCL
);
54 printf("Failed to create queue\n");
58 if (id
!= msgque
->msq_id
) {
59 printf("Restored queue has wrong id (%d instead of %d)\n",
65 for (i
= 0; i
< msgque
->qnum
; i
++) {
66 if (msgsnd(msgque
->msq_id
, &msgque
->messages
[i
].mtype
,
67 msgque
->messages
[i
].msize
, IPC_NOWAIT
) != 0) {
68 printf("msgsnd failed (%m)\n");
76 if (msgctl(id
, IPC_RMID
, 0))
77 printf("Failed to destroy queue: %d\n", -errno
);
81 int check_and_destroy_queue(struct msgque_data
*msgque
)
87 ret
= msgrcv(msgque
->msq_id
, &message
.mtype
, MAX_MSG_SIZE
,
92 printf("Failed to read IPC message: %m\n");
96 if (ret
!= msgque
->messages
[cnt
].msize
) {
97 printf("Wrong message size: %d (expected %d)\n", ret
,
98 msgque
->messages
[cnt
].msize
);
102 if (message
.mtype
!= msgque
->messages
[cnt
].mtype
) {
103 printf("Wrong message type\n");
107 if (memcmp(message
.mtext
, msgque
->messages
[cnt
].mtext
, ret
)) {
108 printf("Wrong message content\n");
115 if (cnt
!= msgque
->qnum
) {
116 printf("Wrong message number\n");
123 if (msgctl(msgque
->msq_id
, IPC_RMID
, 0)) {
124 printf("Failed to destroy queue: %d\n", -errno
);
130 int dump_queue(struct msgque_data
*msgque
)
132 struct msqid64_ds ds
;
136 for (kern_id
= 0; kern_id
< 256; kern_id
++) {
137 ret
= msgctl(kern_id
, MSG_STAT
, &ds
);
139 if (errno
== -EINVAL
)
141 printf("Failed to get stats for IPC queue with id %d\n",
146 if (ret
== msgque
->msq_id
)
150 msgque
->messages
= malloc(sizeof(struct msg1
) * ds
.msg_qnum
);
151 if (msgque
->messages
== NULL
) {
152 printf("Failed to get stats for IPC queue\n");
156 msgque
->qnum
= ds
.msg_qnum
;
157 msgque
->mode
= ds
.msg_perm
.mode
;
158 msgque
->qbytes
= ds
.msg_qbytes
;
160 for (i
= 0; i
< msgque
->qnum
; i
++) {
161 ret
= msgrcv(msgque
->msq_id
, &msgque
->messages
[i
].mtype
,
162 MAX_MSG_SIZE
, i
, IPC_NOWAIT
| MSG_COPY
);
164 printf("Failed to copy IPC message: %m (%d)\n", errno
);
167 msgque
->messages
[i
].msize
= ret
;
172 int fill_msgque(struct msgque_data
*msgque
)
176 msgbuf
.mtype
= MSG_TYPE
;
177 memcpy(msgbuf
.mtext
, TEST_STRING
, sizeof(TEST_STRING
));
178 if (msgsnd(msgque
->msq_id
, &msgbuf
.mtype
, sizeof(TEST_STRING
),
180 printf("First message send failed (%m)\n");
184 msgbuf
.mtype
= ANOTHER_MSG_TYPE
;
185 memcpy(msgbuf
.mtext
, ANOTHER_TEST_STRING
, sizeof(ANOTHER_TEST_STRING
));
186 if (msgsnd(msgque
->msq_id
, &msgbuf
.mtype
, sizeof(ANOTHER_TEST_STRING
),
188 printf("Second message send failed (%m)\n");
194 int main(int argc
, char **argv
)
197 struct msgque_data msgque
;
200 printf("Please run the test as root - Exiting.\n");
201 return ksft_exit_fail();
204 msgque
.key
= ftok(argv
[0], 822155650);
205 if (msgque
.key
== -1) {
206 printf("Can't make key: %d\n", -errno
);
207 return ksft_exit_fail();
210 msgque
.msq_id
= msgget(msgque
.key
, IPC_CREAT
| IPC_EXCL
| 0666);
211 if (msgque
.msq_id
== -1) {
213 printf("Can't create queue: %d\n", err
);
217 err
= fill_msgque(&msgque
);
219 printf("Failed to fill queue: %d\n", err
);
223 err
= dump_queue(&msgque
);
225 printf("Failed to dump queue: %d\n", err
);
229 err
= check_and_destroy_queue(&msgque
);
231 printf("Failed to check and destroy queue: %d\n", err
);
235 err
= restore_queue(&msgque
);
237 printf("Failed to restore queue: %d\n", err
);
241 err
= check_and_destroy_queue(&msgque
);
243 printf("Failed to test queue: %d\n", err
);
246 return ksft_exit_pass();
249 if (msgctl(msgque
.msq_id
, IPC_RMID
, 0)) {
250 printf("Failed to destroy queue: %d\n", -errno
);
251 return ksft_exit_fail();
254 return ksft_exit_fail();