FTL/src/daemon.c

294 lines
6.7 KiB
C

/* Pi-hole: A black hole for Internet advertisements
* (c) 2017 Pi-hole, LLC (https://pi-hole.net)
* Network-wide ad blocking via your own hardware.
*
* FTL Engine
* Daemon routines
*
* This file is copyright under the latest version of the EUPL.
* Please see LICENSE file for your rights under this license. */
#include "FTL.h"
#include "daemon.h"
#include "config.h"
#include "log.h"
// sleepms()
#include "timers.h"
// gravityDB_close()
#include "database/gravity-db.h"
// destroy_shmem()
#include "shmem.h"
// uname()
#include <sys/utsname.h>
// killed
#include "signals.h"
// sysinfo()
#include <sys/sysinfo.h>
#include <errno.h>
pthread_t threads[THREADS_MAX] = { 0 };
pthread_t api_threads[MAX_API_THREADS] = { 0 };
bool resolver_ready = false;
void go_daemon(void)
{
// Create child process
pid_t process_id = fork();
// Indication of fork() failure
if (process_id < 0)
{
logg("fork failed!\n");
// Return failure in exit status
exit(EXIT_FAILURE);
}
// PARENT PROCESS. Need to kill it.
if (process_id > 0)
{
printf("FTL started!\n");
// return success in exit status
exit(EXIT_SUCCESS);
}
//unmask the file mode
umask(0);
//set new session
// creates a session and sets the process group ID
const pid_t sid = setsid();
if(sid < 0)
{
// Return failure
logg("setsid failed!\n");
exit(EXIT_FAILURE);
}
// Create grandchild process
// Fork a second child and exit immediately to prevent zombies. This
// causes the second child process to be orphaned, making the init
// process responsible for its cleanup. And, since the first child is
// a session leader without a controlling terminal, it's possible for
// it to acquire one by opening a terminal in the future (System V-
// based systems). This second fork guarantees that the child is no
// longer a session leader, preventing the daemon from ever acquiring
// a controlling terminal.
process_id = fork();
// Indication of fork() failure
if (process_id < 0)
{
logg("fork failed!\n");
// Return failure in exit status
exit(EXIT_FAILURE);
}
// PARENT PROCESS. Need to kill it.
if (process_id > 0)
{
// return success in exit status
exit(EXIT_SUCCESS);
}
savepid();
// Closing stdin, stdout and stderr is handled by dnsmasq
}
void savepid(void)
{
FILE *f;
const pid_t pid = getpid();
if((f = fopen(FTLfiles.pid, "w+")) == NULL)
{
logg("WARNING: Unable to write PID to file.");
logg(" Continuing anyway...");
}
else
{
fprintf(f, "%i", (int)pid);
fclose(f);
}
logg("PID of FTL process: %i", (int)pid);
}
static void removepid(void)
{
FILE *f;
if((f = fopen(FTLfiles.pid, "w")) == NULL)
{
logg("WARNING: Unable to empty PID file");
return;
}
fclose(f);
}
char *getUserName(void)
{
char *name;
// the getpwuid() function shall search the user database for an entry with a matching uid
// the geteuid() function shall return the effective user ID of the calling process - this is used as the search criteria for the getpwuid() function
const uid_t euid = geteuid();
const struct passwd *pw = getpwuid(euid);
if(pw)
{
name = strdup(pw->pw_name);
}
else
{
if(asprintf(&name, "%u", euid) < 0)
return NULL;
}
return name;
}
// "man 7 hostname" says:
//
// Each element of the hostname must be from 1 to 63 characters long and the
// entire hostname, including the dots, can be at most 253 characters long.
//
// Valid characters for hostnames are ASCII(7) letters from a to z, the
// digits from 0 to 9, and the hyphen (-). A hostname may not start with a
// hyphen.
#define HOSTNAMESIZE 256
static char nodename[HOSTNAMESIZE] = { 0 };
const char *hostname(void)
{
// Ask kernel for node name if not known
// This is equivalent to "uname -n"
if(nodename[0] == '\0')
{
struct utsname buf;
if(uname(&buf) == 0)
strncpy(nodename, buf.nodename, HOSTNAMESIZE);
nodename[HOSTNAMESIZE-1] = '\0';
}
return nodename;
}
void delay_startup(void)
{
// Exit early if not sleeping
if(config.delay_startup == 0u)
return;
// Get uptime of system
struct sysinfo info = { 0 };
if(sysinfo(&info) == 0)
{
if(info.uptime > DELAY_UPTIME)
{
logg("Not sleeping as system has finished booting");
return;
}
}
else
{
logg("Unable to obtain sysinfo: %s (%i)", strerror(errno), errno);
}
// Sleep if requested by DELAY_STARTUP
logg("Sleeping for %d seconds as requested by configuration ...", config.delay_startup);
if(sleep(config.delay_startup) != 0)
{
logg("FATAL: Sleeping was interrupted by an external signal");
cleanup(EXIT_FAILURE);
exit(EXIT_FAILURE);
}
logg("Done sleeping, continuing startup of resolver...");
}
// Is this a fork?
bool __attribute__ ((const)) is_fork(const pid_t mpid, const pid_t pid)
{
return mpid > -1 && mpid != pid;
}
pid_t FTL_gettid(void)
{
#ifdef SYS_gettid
return (pid_t)syscall(SYS_gettid);
#else
#warning SYS_gettid is not available on this system
return -1;
#endif // SYS_gettid
}
static void terminate_threads(void)
{
struct timespec ts;
// Terminate threads before closing database connections and finishing shared memory
killed = true;
// Try to join threads to ensure cancellation has succeeded
logg("Waiting for threads to join");
for(int i = 0; i < THREADS_MAX; i++)
{
if(thread_cancellable[i])
{
logg("Thread %s (%d) is idle, terminating it.",
thread_names[i], i);
pthread_cancel(threads[i]);
}
if (clock_gettime(CLOCK_REALTIME, &ts) == -1)
{
logg("Thread %s (%d) is busy, cancelling it (cannot set timeout).",
thread_names[i], i);
pthread_cancel(threads[i]);
continue;
}
// Timeout for joining is 2 seconds for each thread
ts.tv_sec += 2;
const int s = pthread_timedjoin_np(threads[i], NULL, &ts);
if(s != 0)
{
logg("Thread %s (%d) is still busy, cancelling it.",
thread_names[i], i);
pthread_cancel(threads[i]);
continue;
}
}
logg("All threads joined");
}
// Clean up on exit
void cleanup(const int ret)
{
// Do proper cleanup only if FTL started successfully
if(resolver_ready)
{
// Terminate threads
terminate_threads();
// Cancel and join possibly still running API worker threads
for(unsigned int tid = 0; tid < MAX_API_THREADS; tid++)
{
// Otherwise, cancel and join the thread
logg("Joining API worker thread %d", tid);
pthread_cancel(api_threads[tid]);
pthread_join(api_threads[tid], NULL);
}
// Close database connection
lock_shm();
gravityDB_close();
unlock_shm();
}
// Remove PID file
removepid();
// Remove shared memory objects
// Important: This invalidated all objects such as
// counters-> ... etc.
// This should be the last action when cleaning up
destroy_shmem();
char buffer[42] = { 0 };
format_time(buffer, 0, timer_elapsed_msec(EXIT_TIMER));
logg("########## FTL terminated after%s (code %i)! ##########", buffer, ret);
}