2 * This code is mainly taken from Doug Potter's page
4 * http://www-theorie.physik.unizh.ch/~dpotter/howto/daemonize
6 * I contacted him 2007-04-16 about the license for the original code,
7 * he replied it is Public Domain. Use the URL above to get the original
8 * Public Domain version if you want it.
10 * This version is LGPL2.1+SLE like the rest of libwebsockets and is
11 * Copyright (c)2006 - 2013 Andy Green <andy@warmcat.com>
18 #include <sys/types.h>
25 #include "private-libwebsockets.h"
28 static char *lock_path;
30 int get_daemonize_pid()
36 child_handler(int signum)
43 case SIGALRM: /* timed out daemonizing */
47 case SIGUSR1: /* positive confirmation we daemonized well */
48 /* Create the lock file as the current user */
50 fd = open(lock_path, O_TRUNC | O_RDWR | O_CREAT, 0640);
53 "unable to create lock file %s, code=%d (%s)\n",
54 lock_path, errno, strerror(errno));
57 len = sprintf(sz, "%u", pid_daemon);
58 sent = write(fd, sz, len);
61 "unable to write pid to lock file %s, code=%d (%s)\n",
62 lock_path, errno, strerror(errno));
68 case SIGCHLD: /* daemonization failed */
74 static void lws_daemon_closing(int sigact)
76 if (getpid() == pid_daemon)
79 lws_free_set_NULL(lock_path);
82 kill(getpid(), SIGKILL);
86 * You just need to call this from your main(), when it
87 * returns you are all set "in the background" decoupled
88 * from the console you were started from.
90 * The process context you called from has been terminated then.
94 lws_daemonize(const char *_lock_path)
101 /* already a daemon */
102 // if (getppid() == 1)
105 fd = open(_lock_path, O_RDONLY);
107 n = read(fd, buf, sizeof(buf));
114 "Daemon already running from pid %d\n", n);
118 "Removing stale lock file %s from dead pid %d\n",
124 n = strlen(_lock_path) + 1;
125 lock_path = lws_malloc(n);
127 fprintf(stderr, "Out of mem in lws_daemonize\n");
130 strcpy(lock_path, _lock_path);
132 /* Trap signals that we expect to receive */
133 signal(SIGCHLD, child_handler); /* died */
134 signal(SIGUSR1, child_handler); /* was happy */
135 signal(SIGALRM, child_handler); /* timeout daemonizing */
137 /* Fork off the parent process */
139 if (pid_daemon < 0) {
140 fprintf(stderr, "unable to fork daemon, code=%d (%s)",
141 errno, strerror(errno));
145 /* If we got a good PID, then we can exit the parent process. */
149 /* At this point we are executing as the child process */
151 pid_daemon = getpid();
153 /* Cancel certain signals */
154 signal(SIGCHLD, SIG_DFL); /* A child process dies */
155 signal(SIGTSTP, SIG_IGN); /* Various TTY signals */
156 signal(SIGTTOU, SIG_IGN);
157 signal(SIGTTIN, SIG_IGN);
158 signal(SIGHUP, SIG_IGN); /* Ignore hangup signal */
160 /* Change the file mode mask */
163 /* Create a new SID for the child process */
167 "unable to create a new session, code %d (%s)",
168 errno, strerror(errno));
173 * Change the current working directory. This prevents the current
174 * directory from being locked; hence not being able to remove it.
176 if (chdir("/tmp") < 0) {
178 "unable to change directory to %s, code %d (%s)",
179 "/", errno, strerror(errno));
183 /* Redirect standard files to /dev/null */
184 if (!freopen("/dev/null", "r", stdin))
185 fprintf(stderr, "unable to freopen() stdin, code %d (%s)",
186 errno, strerror(errno));
188 if (!freopen("/dev/null", "w", stdout))
189 fprintf(stderr, "unable to freopen() stdout, code %d (%s)",
190 errno, strerror(errno));
192 if (!freopen("/dev/null", "w", stderr))
193 fprintf(stderr, "unable to freopen() stderr, code %d (%s)",
194 errno, strerror(errno));
196 /* Tell the parent process that we are A-okay */
197 kill(parent, SIGUSR1);
199 act.sa_handler = lws_daemon_closing;
200 sigemptyset(&act.sa_mask);
203 sigaction(SIGTERM, &act, NULL);
205 /* return to continue what is now "the daemon" */