/* * Copyright (c) 2009, Stathis Kamperis * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include #include #include #include #include #include #include /* fork() */ #include #define MQNAME "/tmqpc" mqd_t md; static void diep(const char *s); int main(void) { const char msg[] = "Parent says hello"; int rv; pid_t pid; /* Create a message queue for write only with default parameters. */ md = mq_open(MQNAME, O_CREAT | O_EXCL | O_WRONLY, 0700, NULL); if (md == -1) diep("mq_open"); /* Send message. */ rv = mq_send(md, msg, sizeof(msg), /* priority */ 0); if (rv == -1) diep("mq_send"); /* Disassociate with message queue. */ rv = mq_close(md); if (rv == -1) diep("mq_close"); /* Fork and have child read the message from parent. */ pid = fork(); if (pid == -1) { diep("fork"); } else if (pid == 0) { /* We are inside the child. */ md = mq_open(MQNAME, O_RDONLY); if (md == -1) diep("child: mq_open"); char msg_recvd[8192]; /* Implementation defined. */ rv = mq_receive(md, msg_recvd, sizeof(msg_recvd), NULL); if (rv == -1) diep("child: mq_receive"); assert(strcmp(msg_recvd, msg) == 0); /* Disassociate with message queue. */ rv = mq_close(md); if (rv == -1) diep("child: mq_close"); /* Remove the message queue from the system. */ rv = mq_unlink(MQNAME); if (rv == -1) diep("mq_unlink"); printf("passed\n"); } else { /* We are inside the parent. */ int status; wait(&status); } return EXIT_SUCCESS; } void diep(const char *s) { perror(s); /* * Message queues' name & resources are persistent, i.e., they live * even after the process dies. That's why, disassociate and destroy * the queue on failure, or else we might end up with zombie queues and * hit the limit of max open queues. * Also, we don't care about the return value of the following calls. */ mq_close(md); mq_unlink(MQNAME); exit(EXIT_FAILURE); }