| /* |
| * Copyright (c) 2002, Intel Corporation. All rights reserved. |
| * Created by: julie.n.fleischer REMOVE-THIS AT intel DOT com |
| * This file is licensed under the GPL license. For the full content |
| * of this license, see the COPYING file at the top level of this |
| * source tree. |
| */ |
| |
| /* |
| * Test mq_send() will set errno == EINTR if it is interrupted by a signal. |
| * |
| * Have a child send signals until it starts to block. At that point, have |
| * the parent send a signal to interrupt the child. Test passes if errno == |
| * EINVAL. |
| * |
| * Test very similar to 5-2.c except we don't have the additional test to |
| * verify the child _was_ blocking. |
| */ |
| |
| #include <stdio.h> |
| #include <mqueue.h> |
| #include <fcntl.h> |
| #include <sys/stat.h> |
| #include <sys/types.h> |
| #include <sys/wait.h> |
| #include <unistd.h> |
| #include <string.h> |
| #include <stdlib.h> |
| #include <stdint.h> |
| #include <signal.h> |
| #include <errno.h> |
| #include "posixtest.h" |
| |
| #define NAMESIZE 50 |
| #define MSGSTR "0123456789" |
| #define BUFFER 40 |
| #define MAXMSG 5 |
| |
| #define CHILDPASS 1 |
| #define CHILDFAIL 0 |
| |
| char gqname[NAMESIZE]; |
| mqd_t gqueue; |
| |
| /* |
| * This handler is just used to catch the signal and stop sleep (so the |
| * parent knows the child is still busy sending signals). |
| */ |
| void justreturn_handler(int signo) |
| { |
| return; |
| } |
| |
| int main() |
| { |
| int pid; |
| const char *msgptr = MSGSTR; |
| struct mq_attr attr; |
| struct sigaction act; |
| |
| sprintf(gqname, "/mq_send_12-1_%d", getpid()); |
| |
| attr.mq_maxmsg = MAXMSG; |
| attr.mq_msgsize = BUFFER; |
| gqueue = mq_open(gqname, O_CREAT |O_RDWR, S_IRUSR | S_IWUSR, &attr); |
| if (gqueue == (mqd_t)-1) { |
| perror("mq_open() did not return success"); |
| return PTS_UNRESOLVED; |
| } |
| |
| /* parent and child use justreturn_handler to just return out of |
| * situations -- parent uses to stop it's sleep and wait again for |
| * the child; child uses to stop its mq_send |
| */ |
| act.sa_handler=justreturn_handler; |
| act.sa_flags=0; |
| sigemptyset(&act.sa_mask); |
| sigaction(SIGABRT, &act, 0); |
| |
| if ((pid = fork()) == 0) { |
| /* child here */ |
| int i; |
| |
| sleep(1); // give parent time to set up handler |
| for (i=0; i<MAXMSG+1; i++) { |
| if (mq_send(gqueue, msgptr, strlen(msgptr), 1) == -1) { |
| if (errno == EINTR) { |
| printf("mq_send interrupted by signal\n"); |
| return CHILDPASS; |
| } else { |
| printf("mq_send not interrupted by signal\n"); |
| return CHILDFAIL; |
| } |
| } |
| /* send signal to parent each time message is sent */ |
| kill(getppid(), SIGABRT); |
| } |
| |
| printf("Child never blocked\n"); |
| return CHILDFAIL; |
| } else { |
| /* parent here */ |
| int j,k; |
| |
| for (j=0; j<MAXMSG+1; j++) { |
| if (sleep(3) == 0) { |
| /* If sleep finished, child is probably blocking */ |
| kill(pid, SIGABRT); //signal child |
| break; |
| } |
| } |
| mq_close(gqueue); |
| if (mq_unlink(gqname) != 0) { |
| perror("mq_unlink()"); |
| return PTS_UNRESOLVED; |
| } |
| |
| if (wait(&k) == -1) { |
| perror("Error waiting for child to exit\n"); |
| kill(pid, SIGKILL); //kill child if not gone |
| return PTS_UNRESOLVED; |
| } |
| if (!WIFEXITED(k) || !WEXITSTATUS(k)) { |
| printf("Test FAILED\n"); |
| return PTS_FAIL; |
| } |
| |
| printf("Test PASSED\n"); |
| return PTS_PASS; |
| } |
| |
| return PTS_UNRESOLVED; |
| } |