1 1.3 kamil /* $NetBSD: msg.h,v 1.3 2020/03/06 14:06:56 kamil Exp $ */ 2 1.1 kamil 3 1.1 kamil /*- 4 1.1 kamil * Copyright (c) 2016 The NetBSD Foundation, Inc. 5 1.1 kamil * All rights reserved. 6 1.1 kamil * 7 1.1 kamil * This code is derived from software contributed to The NetBSD Foundation 8 1.1 kamil * by Christos Zoulas. 9 1.1 kamil * 10 1.1 kamil * Redistribution and use in source and binary forms, with or without 11 1.1 kamil * modification, are permitted provided that the following conditions 12 1.1 kamil * are met: 13 1.1 kamil * 1. Redistributions of source code must retain the above copyright 14 1.1 kamil * notice, this list of conditions and the following disclaimer. 15 1.1 kamil * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 kamil * notice, this list of conditions and the following disclaimer in the 17 1.1 kamil * documentation and/or other materials provided with the distribution. 18 1.1 kamil * 19 1.1 kamil * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 kamil * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 kamil * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 kamil * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 kamil * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 kamil * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 kamil * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 kamil * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 kamil * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 kamil * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 kamil * POSSIBILITY OF SUCH DAMAGE. 30 1.1 kamil */ 31 1.1 kamil 32 1.1 kamil struct msg_fds { 33 1.1 kamil int pfd[2]; 34 1.1 kamil int cfd[2]; 35 1.1 kamil }; 36 1.1 kamil 37 1.1 kamil #define CLOSEFD(fd) do { \ 38 1.1 kamil if (fd != -1) { \ 39 1.1 kamil close(fd); \ 40 1.1 kamil fd = -1; \ 41 1.1 kamil } \ 42 1.1 kamil } while (/*CONSTCOND*/ 0) 43 1.1 kamil 44 1.3 kamil static int __used 45 1.1 kamil msg_open(struct msg_fds *fds) 46 1.1 kamil { 47 1.1 kamil if (pipe(fds->pfd) == -1) 48 1.1 kamil return -1; 49 1.1 kamil if (pipe(fds->cfd) == -1) { 50 1.1 kamil close(fds->pfd[0]); 51 1.1 kamil close(fds->pfd[1]); 52 1.1 kamil return -1; 53 1.1 kamil } 54 1.1 kamil return 0; 55 1.1 kamil } 56 1.1 kamil 57 1.3 kamil static void __used 58 1.1 kamil msg_close(struct msg_fds *fds) 59 1.1 kamil { 60 1.1 kamil CLOSEFD(fds->pfd[0]); 61 1.1 kamil CLOSEFD(fds->pfd[1]); 62 1.1 kamil CLOSEFD(fds->cfd[0]); 63 1.1 kamil CLOSEFD(fds->cfd[1]); 64 1.1 kamil } 65 1.1 kamil 66 1.3 kamil static int __used 67 1.1 kamil msg_write_child(const char *info, struct msg_fds *fds, void *msg, size_t len) 68 1.1 kamil { 69 1.1 kamil ssize_t rv; 70 1.1 kamil CLOSEFD(fds->cfd[1]); 71 1.1 kamil CLOSEFD(fds->pfd[0]); 72 1.1 kamil 73 1.2 kamil // printf("Send %s\n", info); 74 1.1 kamil rv = write(fds->pfd[1], msg, len); 75 1.1 kamil if (rv != (ssize_t)len) 76 1.1 kamil return 1; 77 1.1 kamil // printf("Wait %s\n", info); 78 1.1 kamil rv = read(fds->cfd[0], msg, len); 79 1.1 kamil if (rv != (ssize_t)len) 80 1.1 kamil return 1; 81 1.1 kamil return 0; 82 1.1 kamil } 83 1.1 kamil 84 1.3 kamil static int __used 85 1.1 kamil msg_write_parent(const char *info, struct msg_fds *fds, void *msg, size_t len) 86 1.1 kamil { 87 1.1 kamil ssize_t rv; 88 1.1 kamil CLOSEFD(fds->pfd[1]); 89 1.1 kamil CLOSEFD(fds->cfd[0]); 90 1.1 kamil 91 1.2 kamil // printf("Send %s\n", info); 92 1.1 kamil rv = write(fds->cfd[1], msg, len); 93 1.1 kamil if (rv != (ssize_t)len) 94 1.1 kamil return 1; 95 1.1 kamil // printf("Wait %s\n", info); 96 1.1 kamil rv = read(fds->pfd[0], msg, len); 97 1.1 kamil if (rv != (ssize_t)len) 98 1.1 kamil return 1; 99 1.1 kamil return 0; 100 1.1 kamil } 101 1.1 kamil 102 1.3 kamil static int __used 103 1.1 kamil msg_read_parent(const char *info, struct msg_fds *fds, void *msg, size_t len) 104 1.1 kamil { 105 1.1 kamil ssize_t rv; 106 1.1 kamil CLOSEFD(fds->pfd[1]); 107 1.1 kamil CLOSEFD(fds->cfd[0]); 108 1.1 kamil 109 1.2 kamil // printf("Wait %s\n", info); 110 1.1 kamil rv = read(fds->pfd[0], msg, len); 111 1.1 kamil if (rv != (ssize_t)len) 112 1.1 kamil return 1; 113 1.1 kamil // printf("Send %s\n", info); 114 1.1 kamil rv = write(fds->cfd[1], msg, len); 115 1.1 kamil if (rv != (ssize_t)len) 116 1.1 kamil return 1; 117 1.1 kamil return 0; 118 1.1 kamil } 119 1.1 kamil 120 1.3 kamil static int __used 121 1.1 kamil msg_read_child(const char *info, struct msg_fds *fds, void *msg, size_t len) 122 1.1 kamil { 123 1.1 kamil ssize_t rv; 124 1.1 kamil CLOSEFD(fds->cfd[1]); 125 1.1 kamil CLOSEFD(fds->pfd[0]); 126 1.1 kamil 127 1.2 kamil // printf("Wait %s\n", info); 128 1.1 kamil rv = read(fds->cfd[0], msg, len); 129 1.1 kamil if (rv != (ssize_t)len) 130 1.1 kamil return 1; 131 1.1 kamil // printf("Send %s\n", info); 132 1.1 kamil rv = write(fds->pfd[1], msg, len); 133 1.1 kamil if (rv != (ssize_t)len) 134 1.1 kamil return 1; 135 1.1 kamil return 0; 136 1.1 kamil } 137 1.3 kamil 138 1.3 kamil #define PARENT_TO_CHILD(info, fds, msg) \ 139 1.3 kamil SYSCALL_REQUIRE(msg_write_child(info " to child " # fds, &fds, &msg, \ 140 1.3 kamil sizeof(msg)) == 0) 141 1.3 kamil 142 1.3 kamil #define CHILD_FROM_PARENT(info, fds, msg) \ 143 1.3 kamil FORKEE_ASSERT(msg_read_parent(info " from parent " # fds, &fds, &msg, \ 144 1.3 kamil sizeof(msg)) == 0) 145 1.3 kamil 146 1.3 kamil #define CHILD_TO_PARENT(info, fds, msg) \ 147 1.3 kamil FORKEE_ASSERT(msg_write_parent(info " to parent " # fds, &fds, &msg, \ 148 1.3 kamil sizeof(msg)) == 0) 149 1.3 kamil 150 1.3 kamil #define PARENT_FROM_CHILD(info, fds, msg) \ 151 1.3 kamil SYSCALL_REQUIRE(msg_read_child(info " from parent " # fds, &fds, &msg, \ 152 1.3 kamil sizeof(msg)) == 0) 153