summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFerenc Wagner <wferi@niif.hu>2009-09-05 18:42:18 +0200
committerH. Peter Anvin <hpa@zytor.com>2009-09-14 14:41:13 -0700
commitc86f82532ec439678e4dd329b7d9c59d3f9ad6bb (patch)
tree11b3414791243ccbd1a087cc8a828b07a7bb2781
parent5a27e30ec25b6ff49de9e5bc65d0f358dd69275b (diff)
downloadtftp-hpa-c86f82532ec439678e4dd329b7d9c59d3f9ad6bb.tar.gz
Implement the --pidfile option
Setting the umask moved later, right before entering the select loop, so that it does not affect the permissions of the pid file. Signed-off-by: Ferenc Wagner <wferi@niif.hu> Signed-off-by: H. Peter Anvin <hpa@zytor.com>
-rw-r--r--tftpd/tftpd.8.in5
-rw-r--r--tftpd/tftpd.c49
2 files changed, 50 insertions, 4 deletions
diff --git a/tftpd/tftpd.8.in b/tftpd/tftpd.8.in
index 367f7ab..8486934 100644
--- a/tftpd/tftpd.8.in
+++ b/tftpd/tftpd.8.in
@@ -133,6 +133,11 @@ system-provided access controls for the user specified via the
.B \-\-user
option.
.TP
+\fB\-\-pidfile\fP \fIpidfile\fP, \fB\-P\fP \fIpidfile\fP When run in
+standalone mode, write the process ID of the listening server into
+\fIpidfile\fP. On normal termination (SIGTERM or SIGINT) the pid file
+is automatically removed.
+.TP
\fB\-\-timeout\fP \fItimeout\fP, \fB\-t\fP \fItimeout\fP
When run from
.B inetd
diff --git a/tftpd/tftpd.c b/tftpd/tftpd.c
index ff39c85..69ff121 100644
--- a/tftpd/tftpd.c
+++ b/tftpd/tftpd.c
@@ -144,6 +144,13 @@ static void handle_sighup(int sig)
caught_sighup = 1;
}
+/* Handle exit requests by SIGTERM and SIGINT */
+static volatile sig_atomic_t exit_signal = 0;
+static void handle_exit(int sig)
+{
+ exit_signal = sig;
+}
+
/* Handle timeout signal or timeout event */
void timer(int sig)
{
@@ -318,9 +325,10 @@ static struct option long_options[] = {
{ "retransmit", 1, NULL, 'T' },
{ "port-range", 1, NULL, 'R' },
{ "map-file", 1, NULL, 'm' },
+ { "pidfile", 1, NULL, 'P' },
{ NULL, 0, NULL, 0 }
};
-static const char short_options[] = "46cspvVlLa:B:u:U:r:t:T:R:m:";
+static const char short_options[] = "46cspvVlLa:B:u:U:r:t:T:R:m:P:";
int main(int argc, char **argv)
{
@@ -352,6 +360,7 @@ int main(int argc, char **argv)
#ifdef WITH_REGEX
char *rewrite_file = NULL;
#endif
+ const char *pidfile = NULL;
u_short tp_opcode;
/* basename() is way too much of a pain from a portability standpoint */
@@ -475,6 +484,9 @@ int main(int argc, char **argv)
printf("%s\n", TFTPD_CONFIG_STR);
exit(0);
break;
+ case 'P':
+ pidfile = optarg;
+ break;
default:
syslog(LOG_ERR, "Unknown option: '%c'", optopt);
break;
@@ -507,16 +519,19 @@ int main(int argc, char **argv)
exit(EX_NOUSER);
}
- if (spec_umask || !unixperms)
- umask(my_umask);
-
#ifdef WITH_REGEX
if (rewrite_file)
rewrite_rules = read_remap_rules(rewrite_file);
#endif
+ if (pidfile && !standalone) {
+ syslog(LOG_WARNING, "not in standalone mode, ignoring pid file");
+ pidfile = NULL;
+ }
+
/* If we're running standalone, set up the input port */
if (standalone) {
+ FILE *pf;
#ifdef HAVE_IPV6
if (ai_fam != AF_INET6) {
#endif
@@ -702,6 +717,20 @@ int main(int argc, char **argv)
syslog(LOG_ERR, "cannot daemonize: %m");
exit(EX_OSERR);
}
+ set_signal(SIGTERM, handle_exit, 0);
+ set_signal(SIGINT, handle_exit, 0);
+ if (pidfile) {
+ pf = fopen (pidfile, "w");
+ if (!pf) {
+ syslog(LOG_ERR, "cannot open pid file '%s' for writing: %m", pidfile);
+ pidfile = NULL;
+ } else {
+ if (fprintf(pf, "%d\n", getpid()) < 0)
+ syslog(LOG_ERR, "error writing pid file '%s': %m", pidfile);
+ if (fclose(pf))
+ syslog(LOG_ERR, "error closing pid file '%s': %m", pidfile);
+ }
+ }
if (fd6 > fd4)
fdmax = fd6;
else
@@ -734,11 +763,23 @@ int main(int argc, char **argv)
lose packets as a result. */
set_signal(SIGHUP, handle_sighup, 0);
+ if (spec_umask || !unixperms)
+ umask(my_umask);
+
while (1) {
fd_set readset;
struct timeval tv_waittime;
int rv;
+ if (exit_signal) { /* happens in standalone mode only */
+ if (pidfile && unlink(pidfile)) {
+ syslog(LOG_WARNING, "error removing pid file '%s': %m", pidfile);
+ exit(EX_OSERR);
+ } else {
+ exit(0);
+ }
+ }
+
if (caught_sighup) {
caught_sighup = 0;
if (standalone) {