2000-11-24 23:30:50 +00:00
|
|
|
/*
|
2000-11-28 23:23:41 +00:00
|
|
|
dropin.c -- a set of drop-in replacements for libc functions
|
2006-04-26 13:52:58 +00:00
|
|
|
Copyright (C) 2000-2005 Ivo Timmermans,
|
2016-04-14 15:20:36 +00:00
|
|
|
2000-2016 Guus Sliepen <guus@tinc-vpn.org>
|
2000-11-24 23:30:50 +00:00
|
|
|
|
|
|
|
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 2 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.
|
|
|
|
|
2009-09-24 22:01:00 +00:00
|
|
|
You should have received a copy of the GNU General Public License along
|
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2000-11-24 23:30:50 +00:00
|
|
|
*/
|
|
|
|
|
2003-07-17 15:06:27 +00:00
|
|
|
#include "system.h"
|
2000-11-24 23:30:50 +00:00
|
|
|
|
2003-07-21 13:14:02 +00:00
|
|
|
#include "xalloc.h"
|
|
|
|
|
2000-11-24 23:30:50 +00:00
|
|
|
#ifndef HAVE_DAEMON
|
2000-11-29 00:33:15 +00:00
|
|
|
/*
|
|
|
|
Replacement for the daemon() function.
|
2012-10-10 15:17:49 +00:00
|
|
|
|
2000-11-29 00:33:15 +00:00
|
|
|
The daemon() function is for programs wishing to detach themselves
|
|
|
|
from the controlling terminal and run in the background as system
|
|
|
|
daemons.
|
|
|
|
|
|
|
|
Unless the argument nochdir is non-zero, daemon() changes the
|
|
|
|
current working directory to the root (``/'').
|
|
|
|
|
|
|
|
Unless the argument noclose is non-zero, daemon() will redirect
|
|
|
|
standard input, standard output and standard error to /dev/null.
|
|
|
|
*/
|
2009-09-24 22:14:03 +00:00
|
|
|
int daemon(int nochdir, int noclose) {
|
2003-07-28 22:06:09 +00:00
|
|
|
#ifdef HAVE_FORK
|
2002-09-09 21:49:16 +00:00
|
|
|
pid_t pid;
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
pid = fork();
|
|
|
|
|
|
|
|
/* Check if forking failed */
|
|
|
|
if(pid < 0) {
|
|
|
|
perror("fork");
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we are the parent, terminate */
|
|
|
|
if(pid)
|
|
|
|
exit(0);
|
|
|
|
|
|
|
|
/* Detach by becoming the new process group leader */
|
|
|
|
if(setsid() < 0) {
|
|
|
|
perror("setsid");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Change working directory to the root (to avoid keeping mount
|
|
|
|
points busy) */
|
|
|
|
if(!nochdir) {
|
|
|
|
chdir("/");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Redirect stdin/out/err to /dev/null */
|
|
|
|
if(!noclose) {
|
|
|
|
fd = open("/dev/null", O_RDWR);
|
|
|
|
|
|
|
|
if(fd < 0) {
|
|
|
|
perror("opening /dev/null");
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
dup2(fd, 0);
|
|
|
|
dup2(fd, 1);
|
|
|
|
dup2(fd, 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2003-07-28 22:06:09 +00:00
|
|
|
#else
|
|
|
|
return -1;
|
|
|
|
#endif
|
2000-11-24 23:30:50 +00:00
|
|
|
}
|
|
|
|
#endif
|
2000-11-28 23:23:41 +00:00
|
|
|
|
2001-11-05 19:06:07 +00:00
|
|
|
#ifndef HAVE_ASPRINTF
|
2009-09-08 16:16:58 +00:00
|
|
|
int asprintf(char **buf, const char *fmt, ...) {
|
|
|
|
int result;
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
result = vasprintf(buf, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int vasprintf(char **buf, const char *fmt, va_list ap) {
|
2002-09-09 21:49:16 +00:00
|
|
|
int status;
|
2009-09-08 16:16:58 +00:00
|
|
|
va_list aq;
|
2002-09-09 21:49:16 +00:00
|
|
|
int len;
|
|
|
|
|
|
|
|
len = 4096;
|
|
|
|
*buf = xmalloc(len);
|
2001-11-05 19:06:07 +00:00
|
|
|
|
2009-09-08 16:16:58 +00:00
|
|
|
va_copy(aq, ap);
|
|
|
|
status = vsnprintf(*buf, len, fmt, aq);
|
|
|
|
va_end(aq);
|
2001-11-05 19:06:07 +00:00
|
|
|
|
2002-09-09 21:49:16 +00:00
|
|
|
if(status >= 0)
|
2003-07-29 22:59:01 +00:00
|
|
|
*buf = xrealloc(*buf, status + 1);
|
2001-11-05 19:06:07 +00:00
|
|
|
|
2002-09-09 21:49:16 +00:00
|
|
|
if(status > len - 1) {
|
|
|
|
len = status;
|
2009-09-08 16:16:58 +00:00
|
|
|
va_copy(aq, ap);
|
|
|
|
status = vsnprintf(*buf, len, fmt, aq);
|
|
|
|
va_end(aq);
|
2002-09-09 21:49:16 +00:00
|
|
|
}
|
2001-11-05 19:06:07 +00:00
|
|
|
|
2002-09-09 21:49:16 +00:00
|
|
|
return status;
|
2001-11-05 19:06:07 +00:00
|
|
|
}
|
|
|
|
#endif
|
2003-07-28 22:06:09 +00:00
|
|
|
|
|
|
|
#ifndef HAVE_GETTIMEOFDAY
|
|
|
|
int gettimeofday(struct timeval *tv, void *tz) {
|
2013-01-21 15:12:18 +00:00
|
|
|
#ifdef HAVE_MINGW
|
|
|
|
FILETIME ft;
|
|
|
|
GetSystemTimeAsFileTime(&ft);
|
|
|
|
uint64_t lt = (uint64_t)ft.dwLowDateTime | ((uint64_t)ft.dwHighDateTime << 32);
|
|
|
|
lt -= 116444736000000000ULL;
|
|
|
|
tv->tv_sec = lt / 10000000;
|
|
|
|
tv->tv_usec = (lt / 10) % 1000000;
|
|
|
|
#else
|
|
|
|
#warning No high resolution time source!
|
2003-07-28 22:06:09 +00:00
|
|
|
tv->tv_sec = time(NULL);
|
|
|
|
tv->tv_usec = 0;
|
2013-01-21 15:12:18 +00:00
|
|
|
#endif
|
2003-07-28 22:06:09 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
2010-11-13 14:50:39 +00:00
|
|
|
|
2016-04-14 15:20:36 +00:00
|
|
|
#ifndef HAVE_NANOSLEEP
|
|
|
|
int nanosleep(const struct timespec *req, struct timespec *rem) {
|
|
|
|
struct timeval tv = {req->tv_sec, req->tv_nsec / 1000};
|
|
|
|
return select(0, NULL, NULL, NULL, &tv);
|
2010-11-13 14:50:39 +00:00
|
|
|
}
|
|
|
|
#endif
|