/* 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 "benc/String.h" #include "crypto/random/Random.h" #include "rust/cjdns_sys/Rffi.h" #include "util/Defined.h" #include "util/events/EventBase.h" #include "util/events/Pipe.h" #include "util/events/Socket.h" #include "util/events/Timeout.h" #include "memory/Allocator.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 "interface/addressable/AddrIface.h" #include #include #include #include #include #include #define MESSAGE "IT WORKS!" #define MESSAGEB "INDEED" struct Context { struct Iface iface; struct Allocator* alloc; struct Allocator* rootAlloc; EventBase_t* base; struct Log* log; // Parent only int fd; // child only char* name; Identity }; static void onConnectionParent(void* context, const struct Sockaddr* addr) { printf("onConnectionParent\n"); struct Context* c = Identity_check((struct Context*) context); struct Allocator* alloc = Allocator_child(c->alloc); Message_t* msg = Message_new(0, 256, alloc); Err_assert(Message_epush(msg, MESSAGE, CString_strlen(MESSAGE) + 1)); Err_assert(AddrIface_pushAddr(msg, addr)); if (!Defined(win32)) { Message_setAssociatedFd(msg, c->fd); } printf("Parent sending message [%s] len [%d]\n", MESSAGE, Message_getLength(msg)); Iface_send(&c->iface, msg); Allocator_free(alloc); } static int g_childStopped = 0; static struct Context* g_context = NULL; static Iface_DEFUN receiveMessageParent(Message_t* msg, struct Iface* iface) { struct Context* c = Identity_check((struct Context*) iface); Err_assert(AddrIface_popAddr(NULL, msg)); printf("msg length is %d\n", Message_getLength(msg)); Assert_true(Message_getLength(msg) == (int)CString_strlen(MESSAGEB)+1); Assert_true(!Bits_memcmp(Message_bytes(msg), MESSAGEB, CString_strlen(MESSAGEB)+1)); g_context = c; if (g_childStopped) { printf("Parent stopped in receiveMessageParent\n"); Allocator_free(c->rootAlloc); } return NULL; } static void timeout(void* vNULL) { Assert_true(!"timed out."); } static void childDone(struct Allocator_OnFreeJob* ofj) { printf("Shutdown child\n"); struct Context* c = Identity_check((struct Context*) ofj->userData); EventBase_endLoop(c->base); // This causes a segfault because the logger calles into freed memory // Allocator_free(c->rootAlloc); } static Iface_DEFUN receiveMessageChild(Message_t* msg, struct Iface* iface) { struct Context* c = Identity_check((struct Context*) iface); Allocator_t* alloc = Allocator_child(c->alloc); Allocator_t* alloc1 = Allocator_child(alloc); Message_t* m = Message_clone(msg, alloc1); printf("Child received message\n"); Assert_true(Message_getLength(m) == (int)CString_strlen(MESSAGE)+1); Assert_true(!Bits_memcmp(Message_bytes(m), MESSAGE, CString_strlen(MESSAGE)+1)); if (!Defined(win32)) { int fd = Message_getAssociatedFd(msg); Assert_true(fd > -1); if (lseek(fd, 0, SEEK_SET) < 0) { printf("lseek(%d) failed: errno %s\n", fd, strerror(errno)); Assert_failure("lseek()"); } uint8_t* buf = Allocator_calloc(Message_getAlloc(msg), 2048, 1); if (read(fd, buf, 1024) < 0) { printf("read(%d) failed: errno %s\n", fd, strerror(errno)); Assert_failure("read()"); } if (CString_strncmp(buf, c->name, 1024)) { printf("want: %s\n" "got: %s", c->name, buf); Assert_failure("file content is wrong"); } } Err_assert(Message_eshift(m, -((int)CString_strlen(MESSAGE)))); Err_assert(Message_epush(m, MESSAGEB, CString_strlen(MESSAGEB))); Allocator_onFree(alloc1, childDone, c); Iface_send(&c->iface, m); Allocator_free(alloc); // exit(0); // shutdown // Allocator_free(c->rootAlloc); return NULL; } static void child(char* name, struct Context* ctx) { Iface_t* iface = NULL; Err_assert(Socket_connect(&iface, name, ctx->alloc)); ctx->iface.send = receiveMessageChild; ctx->name = name; Iface_plumb(&ctx->iface, iface); Timeout_setTimeout(timeout, NULL, 10000, ctx->base, ctx->alloc); EventBase_beginLoop(ctx->base); } static void onChildExit(int64_t exit_status, int term_signal) { printf("\n\n\nChild process exit status = %d term_signal = %d\n\n\n", (int)exit_status, term_signal); Assert_true(exit_status == 0); g_childStopped = true; if (g_context) { printf("Parent stopped in onChildExit\n"); EventBase_endLoop(g_context->base); // Allocator_free(g_context->rootAlloc); } } int main(int argc, char** argv) { struct Allocator* allocator = Allocator_new(1<<20); EventBase_t* eb = EventBase_new(allocator); struct Allocator* alloc = Allocator_child(allocator); struct Log* log = FileWriterLog_new(stdout, alloc); Rffi_setLogger(log); struct Context* ctx = Allocator_calloc(alloc, sizeof(struct Context), 1); Identity_set(ctx); ctx->alloc = alloc; ctx->rootAlloc = allocator; 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 = NULL; Err_assert(Random_new(&rand, alloc, log)); char randName[32] = {0}; Random_base32(rand, (uint8_t*)randName, 31); String* name = String_printf(alloc, "%s%scjdns-test-%s", Pipe_PATH, Pipe_PATH_SEP, randName); if (!Defined(win32)) { String* textName = String_printf(alloc, "%s%scjdns-test-%s.txt", Pipe_PATH, Pipe_PATH_SEP, randName); int fd = open(textName->bytes, O_CREAT | O_TRUNC | O_RDWR, 0600); Assert_true(fd >= 0); Assert_true(write(fd, name->bytes, name->len) == ((ssize_t)name->len)); ctx->fd = fd; unlink(textName->bytes); } Socket_Server_t* ss = NULL; Err_assert(Socket_server(&ss, name->bytes, alloc)); Socket_serverOnConnect(ss, onConnectionParent, ctx); Iface_plumb(&ctx->iface, ss->iface); const 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 #elif netbsd Assert_true(path[0] == 'b'); // Process_getPath returns relative paths on netbsd too #else Assert_true(path[0] == '/'); #endif const char* args[] = { "Process_test", "child", name->bytes, NULL }; Assert_true(!Process_spawn(path, args, alloc, onChildExit)); Timeout_setTimeout(timeout, NULL, 10000, eb, alloc); EventBase_beginLoop(eb); unlink(name->bytes); return 0; }