/* 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 Allocator* alloc; struct EventBase* base; struct Log* log; }; static void onConnectionParent(struct Pipe* p, int status) { Assert_true(!status); struct Context* c = p->iface.receiverContext; 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); Interface_sendMessage(&p->iface, m); Allocator_free(alloc); } static uint8_t receiveMessageParent(struct Message* msg, struct Interface* iface) { struct Context* c = iface->receiverContext; Assert_true(msg->length == (int)CString_strlen(MESSAGEB)); Assert_true(!Bits_memcmp(msg->bytes, MESSAGEB, CString_strlen(MESSAGEB))); Allocator_free(c->alloc); return Error_NONE; } 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 uint8_t receiveMessageChild(struct Message* m, struct Interface* iface) { 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); Interface_sendMessage(iface, m); // shutdown struct Context* ctx = iface->receiverContext; Allocator_free(ctx->alloc); return Error_NONE; } static void child(char* name, struct Context* ctx) { struct Pipe* pipe = Pipe_named(name, ctx->base, NULL, ctx->alloc); pipe->logger = ctx->log; pipe->iface.receiveMessage = receiveMessageChild; pipe->iface.receiverContext = ctx; pipe->onConnection = onConnectionChild; 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); ctx->alloc = alloc; ctx->base = eb; ctx->log = log; 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->iface.receiveMessage = receiveMessageParent; pipe->iface.receiverContext = ctx; pipe->onConnection = onConnectionParent; 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)); Timeout_setTimeout(timeout, NULL, 2000, eb, alloc); EventBase_beginLoop(eb); return 0; }