/* 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));
Timeout_setTimeout(timeout, NULL, 2000, eb, alloc);
EventBase_beginLoop(eb);
return 0;
}