/* vim: set expandtab ts=4 sw=4: */ /* * You may redistribute this program and/or modify it under the terms of * the GNU General Public License as published by the Free Software Foundation, * either version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "crypto/random/Random.h" #include "util/events/EventBase.h" #include "util/events/Pipe.h" #include "util/events/Timeout.h" #include "memory/Allocator.h" #include "memory/MallocAllocator.h" #include "util/events/Process.h" #include "util/log/Log.h" #include "util/log/FileWriterLog.h" #include "util/CString.h" #include "util/Assert.h" #include "wire/Message.h" #include "wire/Error.h" #include #include #include #define MESSAGE "IT WORKS!" #define MESSAGEB "INDEED" struct Context { struct Iface iface; struct Allocator* alloc; struct EventBase* base; struct Log* log; Identity }; static void onConnectionParent(struct Pipe* p, int status) { Assert_true(!status); struct Context* c = Identity_check((struct Context*) p->userData); struct Allocator* alloc = Allocator_child(c->alloc); uint8_t* bytes = Allocator_calloc(alloc, CString_strlen(MESSAGE) + 1, 1); Bits_memcpy(bytes, MESSAGE, CString_strlen(MESSAGE)); struct Message* m = Allocator_clone(alloc, (&(struct Message) { .length = CString_strlen(MESSAGE), .padding = 0, .capacity = CString_strlen(MESSAGE), .alloc = alloc, .bytes = bytes })); printf("Parent sending message [%s]\n", bytes); Iface_send(&c->iface, m); Allocator_free(alloc); } static Iface_DEFUN receiveMessageParent(struct Message* msg, struct Iface* iface) { struct Context* c = Identity_check((struct Context*) iface); Assert_true(msg->length == (int)CString_strlen(MESSAGEB)); Assert_true(!Bits_memcmp(msg->bytes, MESSAGEB, CString_strlen(MESSAGEB))); Allocator_free(c->alloc); return NULL; } static void timeout(void* vNULL) { Assert_true(!"timed out."); } static void onConnectionChild(struct Pipe* p, int status) { Assert_true(!status); printf("Child connected\n"); } static Iface_DEFUN receiveMessageChild(struct Message* msg, struct Iface* iface) { struct Context* c = Identity_check((struct Context*) iface); struct Message* m = Message_clone(msg, c->alloc); printf("Child received message\n"); Assert_true(m->length == (int)CString_strlen(MESSAGE)); Assert_true(!Bits_memcmp(m->bytes, MESSAGE, CString_strlen(MESSAGE))); Message_shift(m, -((int)CString_strlen(MESSAGE)), NULL); Message_push(m, MESSAGEB, CString_strlen(MESSAGEB), NULL); Iface_send(&c->iface, m); // shutdown Allocator_free(c->alloc); return NULL; } static void child(char* name, struct Context* ctx) { struct Pipe* pipe = Pipe_named(name, ctx->base, NULL, ctx->alloc); pipe->logger = ctx->log; pipe->onConnection = onConnectionChild; pipe->userData = ctx; ctx->iface.send = receiveMessageChild; Iface_plumb(&ctx->iface, &pipe->iface); Timeout_setTimeout(timeout, NULL, 2000, ctx->base, ctx->alloc); EventBase_beginLoop(ctx->base); } int main(int argc, char** argv) { struct Allocator* alloc = MallocAllocator_new(1<<20); struct EventBase* eb = EventBase_new(alloc); struct Log* log = FileWriterLog_new(stdout, alloc); struct Context* ctx = Allocator_calloc(alloc, sizeof(struct Context), 1); Identity_set(ctx); ctx->alloc = alloc; ctx->base = eb; ctx->log = log; ctx->iface.send = receiveMessageParent; if (argc > 3 && !CString_strcmp("Process_test", argv[1]) && !CString_strcmp("child", argv[2])) { child(argv[3], ctx); return 0; } struct Random* rand = Random_new(alloc, log, NULL); char name[32] = {0}; Random_base32(rand, (uint8_t*)name, 31); struct Pipe* pipe = Pipe_named(name, eb, NULL, alloc); pipe->logger = log; pipe->userData = ctx; pipe->onConnection = onConnectionParent; Iface_plumb(&ctx->iface, &pipe->iface); char* path = Process_getPath(alloc); Assert_true(path != NULL); #ifdef win32 Assert_true(CString_strstr(path, ":\\") == path + 1); /* C:\ */ Assert_true(CString_strstr(path, ".exe")); #elif openbsd Assert_true(path[0] == 'b'); // Process_getPath returns relative paths on openbsd #else Assert_true(path[0] == '/'); #endif char* args[] = { "Process_test", "child", name, NULL }; Assert_true(!Process_spawn(path, args, eb, alloc, NULL)); Timeout_setTimeout(timeout, NULL, 2000, eb, alloc); EventBase_beginLoop(eb); return 0; }