You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
181 lines
3.9 KiB
181 lines
3.9 KiB
/* |
|
* Copyright 2014 Con Kolivas |
|
* |
|
* This program is free software; you can redistribute it 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. See COPYING for more details. |
|
*/ |
|
|
|
#include <sys/prctl.h> |
|
#include <sys/socket.h> |
|
#include <sys/stat.h> |
|
#include <sys/types.h> |
|
#include <signal.h> |
|
#include <stdio.h> |
|
#include <stdlib.h> |
|
#include <string.h> |
|
#include <unistd.h> |
|
|
|
#include "ckpool.h" |
|
#include "libckpool.h" |
|
#include "generator.h" |
|
|
|
static void rename_proc(const char *name) |
|
{ |
|
char buf[16]; |
|
|
|
snprintf(buf, 16, "ckp@%s", name); |
|
prctl(PR_SET_NAME, buf, 0, 0, 0); |
|
} |
|
|
|
static void *listener(void *arg) |
|
{ |
|
proc_instance_t *pi = (proc_instance_t *)arg; |
|
unixsock_t *us = &pi->us; |
|
int sockd; |
|
|
|
rename_proc(pi->sockname); |
|
retry: |
|
sockd = accept(us->sockd, NULL, NULL); |
|
if (sockd < 0) { |
|
if (interrupted()) |
|
goto retry; |
|
LOGERR("Failed to accept on socket in listener"); |
|
goto out; |
|
} |
|
/* Insert parsing and repeat code here */ |
|
out: |
|
if (sockd >= 0) |
|
close(sockd); |
|
close_unix_socket(us->sockd, us->path); |
|
return NULL; |
|
} |
|
|
|
/* Open the file in path, check if there is a pid in there that still exists |
|
* and if not, write the pid into that file. */ |
|
static bool write_pid(const char *path, pid_t pid) |
|
{ |
|
struct stat statbuf; |
|
FILE *fp; |
|
int ret; |
|
|
|
if (!stat(path, &statbuf)) { |
|
int oldpid; |
|
|
|
LOGWARNING("File %s exists", path); |
|
fp = fopen(path, "r"); |
|
if (!fp) { |
|
LOGERR("Failed to open file %s", path); |
|
return false; |
|
} |
|
ret = fscanf(fp, "%d", &oldpid); |
|
fclose(fp); |
|
if (ret == 1 && !(kill(oldpid, 0))) { |
|
LOGWARNING("Process %s pid %d still exists", path, oldpid); |
|
return false; |
|
} |
|
} |
|
fp = fopen(path, "w"); |
|
if (!fp) { |
|
LOGERR("Failed to open file %s", path); |
|
return false; |
|
} |
|
fprintf(fp, "%d", pid); |
|
fclose(fp); |
|
|
|
return true; |
|
} |
|
|
|
static void create_process_unixsock(proc_instance_t *pi) |
|
{ |
|
unixsock_t *us = &pi->us; |
|
|
|
us->path = strdup(pi->ckp->socket_dir); |
|
realloc_strcat(&us->path, pi->sockname); |
|
LOGDEBUG("Opening %s", us->path); |
|
us->sockd = open_unix_server(us->path); |
|
if (unlikely(us->sockd < 0)) |
|
quit(1, "Failed to open %s socket", pi->sockname); |
|
} |
|
|
|
static void write_namepid(proc_instance_t *pi) |
|
{ |
|
int pid = getpid(); |
|
char s[1024]; |
|
|
|
sprintf(s, "%s%s.pid", pi->ckp->socket_dir, pi->processname); |
|
if (!write_pid(s, pid)) |
|
quit(1, "Failed to write %s pid %d", pi->processname, pid); |
|
} |
|
|
|
static void launch_process(proc_instance_t *pi) |
|
{ |
|
pid_t pid; |
|
|
|
pid = fork(); |
|
if (pid < 0) |
|
quit(1, "Failed to fork %s in launch_process", pi->processname); |
|
if (!pid) { |
|
rename_proc(pi->processname); |
|
write_namepid(pi); |
|
create_process_unixsock(pi); |
|
exit(pi->process(pi)); |
|
} |
|
} |
|
|
|
int main(int argc, char **argv) |
|
{ |
|
proc_instance_t proc_main; |
|
proc_instance_t proc_generator; |
|
pthread_t pth_listener; |
|
ckpool_t ckp; |
|
int c, ret; |
|
|
|
memset(&ckp, 0, sizeof(ckp)); |
|
while ((c = getopt(argc, argv, "c:gn:s:")) != -1) { |
|
switch (c) { |
|
case 'c': |
|
ckp.config = optarg; |
|
break; |
|
case 'g': |
|
/* Launch generator only */ |
|
break; |
|
case 'n': |
|
ckp.name = optarg; |
|
break; |
|
case 's': |
|
ckp.socket_dir = strdup(optarg); |
|
break; |
|
} |
|
} |
|
|
|
if (!ckp.socket_dir) |
|
ckp.socket_dir = strdup("/tmp/ckpool"); |
|
|
|
realloc_strcat(&ckp.socket_dir, "/"); |
|
ret = mkdir(ckp.socket_dir, 0700); |
|
if (ret && errno != EEXIST) |
|
quit(1, "Failed to make directory %s", ckp.socket_dir); |
|
|
|
proc_main.ckp = &ckp; |
|
proc_main.processname = strdup("main"); |
|
proc_main.sockname = strdup("listener"); |
|
write_namepid(&proc_main); |
|
create_process_unixsock(&proc_main); |
|
|
|
create_pthread(&pth_listener, listener, &proc_main); |
|
|
|
/* Launch separate processes from here */ |
|
proc_generator.ckp = &ckp; |
|
proc_generator.processname = strdup("generator"); |
|
proc_generator.sockname = proc_generator.processname; |
|
proc_generator.process = &generator; |
|
launch_process(&proc_generator); |
|
|
|
/* Shutdown from here */ |
|
join_pthread(pth_listener); |
|
dealloc(ckp.socket_dir); |
|
|
|
return 0; |
|
}
|
|
|