2 * A rewrite of the original Debian's start-stop-daemon Perl script
3 * in C (faster - it is executed many times during system startup).
5 * Written by Marek Michalkiewicz <marekm@i17linuxb.ists.pwr.wroc.pl>,
6 * public domain. Based conceptually on start-stop-daemon.pl, by Ian
7 * Jackson <ijackson@gnu.ai.mit.edu>. May be used and distributed
8 * freely for any purpose. Changes by Christian Schwarz
9 * <schwarz@monet.m.isar.de>, to make output conform to the Debian
10 * Console Message Standard, also placed in public domain. Minor
11 * changes by Klee Dienes <klee@debian.org>, also placed in the Public
14 * Changes by Ben Collins <bcollins@debian.org>, added --chuid, --background
15 * and --make-pidfile options, placed in public domain aswell.
17 * Port to OpenBSD by Sontri Tomo Huynh <huynh.29@osu.edu>
18 * and Andreas Schuldei <andreas@schuldei.org>
20 * Changes by Ian Jackson: added --retry (and associated rearrangements).
22 * Modified for Gentoo rc-scripts by Donny Davies <woodchip@gentoo.org>:
23 * I removed the BSD/Hurd/OtherOS stuff, added #include <stddef.h>
24 * and stuck in a #define VERSION "1.9.18". Now it compiles without
25 * the whole automake/config.h dance.
39 #define VERSION "1.9.18"
41 #define MIN_POLL_INTERVAL 20000 /*us*/
52 #include <sys/queue.h>
57 #include <sys/ioctl.h>
58 #include <sys/types.h>
65 #include <linux/sched.h>
68 static int testmode
= 0;
69 static int quietmode
= 0;
70 static int exitnodo
= 1;
73 static int background
= 0;
74 static int mpidfile
= 0;
75 static int signal_nr
= 15;
76 static const char *signal_str
= NULL
;
77 static int user_id
= -1;
78 static int runas_uid
= -1;
79 static int runas_gid
= -1;
80 static const char *userspec
= NULL
;
81 static char *changeuser
= NULL
;
82 static const char *changegroup
= NULL
;
83 static char *changeroot
= NULL
;
84 static const char *cmdname
= NULL
;
85 static char *execname
= NULL
;
86 static char *startas
= NULL
;
87 static const char *pidfile
= NULL
;
88 static char what_stop
[1024];
89 static const char *schedule_str
= NULL
;
90 static const char *progname
= "";
91 static int nicelevel
= 0;
93 static struct stat exec_stat
;
96 struct pid_list
*next
;
100 static struct pid_list
*found
= NULL
;
101 static struct pid_list
*killed
= NULL
;
103 struct schedule_item
{
104 enum { sched_timeout
, sched_signal
, sched_goto
, sched_forever
} type
;
105 int value
; /* seconds, signal no., or index into array */
106 /* sched_forever is only seen within parse_schedule and callees */
109 static int schedule_length
;
110 static struct schedule_item
*schedule
= NULL
;
112 LIST_HEAD(namespace_head
, namespace);
116 LIST_ENTRY(namespace) list
;
121 static struct namespace_head namespace_head
;
123 static void *xmalloc(int size
);
124 static void push(struct pid_list
**list
, pid_t pid
);
125 static void do_help(void);
126 static void parse_options(int argc
, char *const *argv
);
127 static int pid_is_user(pid_t pid
, uid_t uid
);
128 static int pid_is_cmd(pid_t pid
, const char *name
);
129 static void check(pid_t pid
);
130 static void do_pidfile(const char *name
);
131 static void do_stop(int signal_nr
, int quietmode
, int *n_killed
,
132 int *n_notkilled
, int retry_nr
);
133 static int pid_is_exec(pid_t pid
, const struct stat
*esb
);
136 static void fatal(const char *format
, ...)
137 __attribute__((noreturn
, format(printf
, 1, 2)));
138 static void badusage(const char *msg
) __attribute__((noreturn
));
140 static void fatal(const char *format
, ...);
141 static void badusage(const char *msg
);
144 /* This next part serves only to construct the TVCALC macro, which
145 * is used for doing arithmetic on struct timeval's. It works like this:
146 * TVCALC(result, expression);
147 * where result is a struct timeval (and must be an lvalue) and
148 * expression is the single expression for both components. In this
149 * expression you can use the special values TVELEM, which when fed a
150 * const struct timeval* gives you the relevant component, and
151 * TVADJUST. TVADJUST is necessary when subtracting timevals, to make
152 * it easier to renormalise. Whenver you subtract timeval elements,
153 * you must make sure that TVADJUST is added to the result of the
154 * subtraction (before any resulting multiplication or what have you).
155 * TVELEM must be linear in TVADJUST.
157 typedef long tvselector(const struct timeval
*);
158 static long tvselector_sec(const struct timeval
*tv
)
162 static long tvselector_usec(const struct timeval
*tv
)
166 #define TVCALC_ELEM(result, expr, sec, adj) \
168 const long TVADJUST = adj; \
169 long (*const TVELEM)(const struct timeval *) = \
171 (result).tv_##sec = (expr); \
173 #define TVCALC(result, expr) \
175 TVCALC_ELEM(result, expr, sec, (-1)); \
176 TVCALC_ELEM(result, expr, usec, (+1000000)); \
177 (result).tv_sec += (result).tv_usec / 1000000; \
178 (result).tv_usec %= 1000000; \
182 static void fatal(const char *format
, ...)
186 fprintf(stderr
, "%s: ", progname
);
187 va_start(arglist
, format
);
188 vfprintf(stderr
, format
, arglist
);
195 static void *xmalloc(int size
)
202 fatal("malloc(%d) failed", size
);
205 static void xgettimeofday(struct timeval
*tv
)
207 if (gettimeofday(tv
, 0) != 0)
208 fatal("gettimeofday failed: %s", strerror(errno
));
211 static void push(struct pid_list
**list
, pid_t pid
)
215 p
= xmalloc(sizeof(*p
));
221 static void clear(struct pid_list
**list
)
223 struct pid_list
*here
, *next
;
225 for (here
= *list
; here
!= NULL
; here
= next
) {
234 static const char *next_dirname(const char *s
)
241 for (; *cur
!= '/'; ++cur
)
245 for (; *cur
== '/'; ++cur
)
252 static void add_namespace(const char *path
)
255 const char *nsdirname
, *nsname
, *cur
;
256 struct namespace *namespace;
259 nsdirname
= nsname
= "";
261 while ((cur
= next_dirname(cur
))[0] != '\0') {
266 if (!strncmp(nsdirname
, "ipcns/", strlen("ipcns/")))
267 nstype
= CLONE_NEWIPC
;
268 else if (!strncmp(nsdirname
, "netns/", strlen("netns/")))
269 nstype
= CLONE_NEWNET
;
270 else if (!strncmp(nsdirname
, "utcns/", strlen("utcns/")))
271 nstype
= CLONE_NEWUTS
;
273 badusage("invalid namepspace path");
275 namespace = xmalloc(sizeof(*namespace));
276 namespace->path
= path
;
277 namespace->nstype
= nstype
;
278 LIST_INSERT_HEAD(&namespace_head
, namespace, list
);
283 static void set_namespaces(void)
285 struct namespace *namespace;
288 LIST_FOREACH (namespace, &namespace_head
, list
) {
289 if ((fd
= open(namespace->path
, O_RDONLY
)) == -1)
290 fatal("open namespace %s: %s", namespace->path
,
292 if (setns(fd
, namespace->nstype
) == -1)
293 fatal("setns %s: %s", namespace->path
, strerror(errno
));
297 static void set_namespaces(void)
299 if (!LIST_EMPTY(&namespace_head
))
300 fatal("LCX namespaces not supported");
304 static void do_help(void)
306 printf("start-stop-daemon " VERSION
307 " for Debian - small and fast C version written by\n"
308 "Marek Michalkiewicz <marekm@i17linuxb.ists.pwr.wroc.pl>, public domain.\n"
311 " start-stop-daemon -S|--start options ... -- arguments ...\n"
312 " start-stop-daemon -K|--stop options ...\n"
313 " start-stop-daemon -H|--help\n"
314 " start-stop-daemon -V|--version\n"
316 "Options (at least one of --exec|--pidfile|--user is required):\n"
317 " -x|--exec <executable> program to start/check if it is running\n"
318 " -p|--pidfile <pid-file> pid file to check\n"
319 " -c|--chuid <name|uid[:group|gid]>\n"
320 " change to this user/group before starting process\n"
321 " -u|--user <username>|<uid> stop processes owned by this user\n"
322 " -n|--name <process-name> stop processes with this name\n"
323 " -s|--signal <signal> signal to send (default TERM)\n"
324 " -a|--startas <pathname> program to start (default is <executable>)\n"
325 " -N|--nicelevel <incr> add incr to the process's nice level\n"
326 " -b|--background force the process to detach\n"
327 " -m|--make-pidfile create the pidfile before starting\n"
328 " -R|--retry <schedule> check whether processes die, and retry\n"
329 " -t|--test test mode, don't do anything\n"
330 " -o|--oknodo exit status 0 (not 1) if nothing done\n"
331 " -q|--quiet be more quiet\n"
332 " -v|--verbose be more verbose\n"
333 "Retry <schedule> is <item>|/<item>/... where <item> is one of\n"
334 " -<signal-num>|[-]<signal-name> send that signal\n"
335 " <timeout> wait that many seconds\n"
336 " forever repeat remainder forever\n"
337 "or <schedule> may be just <timeout>, meaning <signal>/<timeout>/KILL/<timeout>\n"
339 "Exit status: 0 = done 1 = nothing done (=> 0 if --oknodo)\n"
340 " 3 = trouble 2 = with --retry, processes wouldn't die\n");
344 static void badusage(const char *msg
)
347 fprintf(stderr
, "%s: %s\n", progname
, msg
);
348 fprintf(stderr
, "Try `%s --help' for more information.\n", progname
);
357 const struct sigpair siglist
[] = {
358 {"ABRT", SIGABRT
}, {"ALRM", SIGALRM
}, {"FPE", SIGFPE
},
359 {"HUP", SIGHUP
}, {"ILL", SIGILL
}, {"INT", SIGINT
},
360 {"KILL", SIGKILL
}, {"PIPE", SIGPIPE
}, {"QUIT", SIGQUIT
},
361 {"SEGV", SIGSEGV
}, {"TERM", SIGTERM
}, {"USR1", SIGUSR1
},
362 {"USR2", SIGUSR2
}, {"CHLD", SIGCHLD
}, {"CONT", SIGCONT
},
363 {"STOP", SIGSTOP
}, {"TSTP", SIGTSTP
}, {"TTIN", SIGTTIN
},
366 static int parse_integer(const char *string
, int *value_r
)
374 ul
= strtoul(string
, &ep
, 10);
375 if (ul
> INT_MAX
|| *ep
!= '\0')
382 static int parse_signal(const char *signal_str
, int *signal_nr
)
386 if (parse_integer(signal_str
, signal_nr
) == 0)
389 for (i
= 0; i
< sizeof(siglist
) / sizeof(siglist
[0]); i
++) {
390 if (strcmp(signal_str
, siglist
[i
].name
) == 0) {
391 *signal_nr
= siglist
[i
].signal
;
398 static void parse_schedule_item(const char *string
, struct schedule_item
*item
)
400 const char *after_hyph
;
402 if (!strcmp(string
, "forever")) {
403 item
->type
= sched_forever
;
404 } else if (isdigit((unsigned char)string
[0])) {
405 item
->type
= sched_timeout
;
406 if (parse_integer(string
, &item
->value
) != 0)
407 badusage("invalid timeout value in schedule");
408 } else if ((after_hyph
= string
+ (string
[0] == '-'))
409 && parse_signal(after_hyph
, &item
->value
) == 0) {
410 item
->type
= sched_signal
;
413 "invalid schedule item (must be [-]<signal-name>, "
414 "-<signal-number>, <timeout> or `forever'");
418 static void parse_schedule(const char *schedule_str
)
426 for (slash
= schedule_str
; *slash
; slash
++)
430 schedule_length
= (count
== 0) ? 4 : count
+ 1;
431 schedule
= xmalloc(sizeof(*schedule
) * schedule_length
);
434 schedule
[0].type
= sched_signal
;
435 schedule
[0].value
= signal_nr
;
436 parse_schedule_item(schedule_str
, &schedule
[1]);
437 if (schedule
[1].type
!= sched_timeout
) {
439 "--retry takes timeout, or schedule list"
440 " of at least two items");
442 schedule
[2].type
= sched_signal
;
443 schedule
[2].value
= SIGKILL
;
444 schedule
[3] = schedule
[1];
448 while (schedule_str
!= NULL
) {
449 slash
= strchr(schedule_str
, '/');
450 str_len
= slash
? slash
- schedule_str
451 : (ptrdiff_t)strlen(schedule_str
);
452 if (str_len
>= (ptrdiff_t)sizeof(item_buf
))
454 "invalid schedule item: far too long"
455 " (you must delimit items with slashes)");
456 memcpy(item_buf
, schedule_str
, str_len
);
457 item_buf
[str_len
] = 0;
458 schedule_str
= slash
? slash
+ 1 : NULL
;
460 parse_schedule_item(item_buf
, &schedule
[count
]);
461 if (schedule
[count
].type
== sched_forever
) {
464 "invalid schedule: `forever'"
465 " appears more than once");
472 schedule
[count
].type
= sched_goto
;
473 schedule
[count
].value
= repeatat
;
476 assert(count
== schedule_length
);
480 static void parse_options(int argc
, char *const *argv
)
482 static struct option longopts
[] = {
483 {"help", 0, NULL
, 'H'}, {"stop", 0, NULL
, 'K'},
484 {"start", 0, NULL
, 'S'}, {"version", 0, NULL
, 'V'},
485 {"startas", 1, NULL
, 'a'}, {"name", 1, NULL
, 'n'},
486 {"oknodo", 0, NULL
, 'o'}, {"pidfile", 1, NULL
, 'p'},
487 {"quiet", 0, NULL
, 'q'}, {"signal", 1, NULL
, 's'},
488 {"test", 0, NULL
, 't'}, {"user", 1, NULL
, 'u'},
489 {"chroot", 1, NULL
, 'r'}, {"namespace", 1, NULL
, 'd'},
490 {"verbose", 0, NULL
, 'v'}, {"exec", 1, NULL
, 'x'},
491 {"chuid", 1, NULL
, 'c'}, {"nicelevel", 1, NULL
, 'N'},
492 {"background", 0, NULL
, 'b'}, {"make-pidfile", 0, NULL
, 'm'},
493 {"retry", 1, NULL
, 'R'}, {NULL
, 0, NULL
, 0}};
497 c
= getopt_long(argc
, argv
,
498 "HKSVa:n:op:qr:d:s:tu:vx:c:N:bmR:", longopts
,
503 case 'H': /* --help */
506 case 'K': /* --stop */
509 case 'S': /* --start */
512 case 'V': /* --version */
513 printf("start-stop-daemon " VERSION
"\n");
515 case 'a': /* --startas <pathname> */
518 case 'n': /* --name <process-name> */
521 case 'o': /* --oknodo */
524 case 'p': /* --pidfile <pid-file> */
527 case 'q': /* --quiet */
530 case 's': /* --signal <signal> */
533 case 't': /* --test */
536 case 'u': /* --user <username>|<uid> */
539 case 'v': /* --verbose */
542 case 'x': /* --exec <executable> */
545 case 'c': /* --chuid <username>|<uid> */
546 changeuser
= strtok(optarg
, ":");
547 changegroup
= strtok(NULL
, ":");
549 case 'r': /* --chroot /new/root */
552 case 'd': /* --namespace /.../<ipcns>|<netns>|<utsns>/name */
554 add_namespace(optarg
);
557 case 'N': /* --nice */
558 nicelevel
= atoi(optarg
);
560 case 'b': /* --background */
563 case 'm': /* --make-pidfile */
566 case 'R': /* --retry <schedule>|<timeout> */
567 schedule_str
= optarg
;
570 badusage(NULL
); /* message printed by getopt */
574 if (signal_str
!= NULL
) {
575 if (parse_signal(signal_str
, &signal_nr
) != 0)
577 "signal value must be numeric or name"
578 " of signal (KILL, INTR, ...)");
581 if (schedule_str
!= NULL
) {
582 parse_schedule(schedule_str
);
586 badusage("need one of --start or --stop");
588 if (!execname
&& !pidfile
&& !userspec
&& !cmdname
)
590 "need at least one of --exec, --pidfile, --user or --name");
595 if (start
&& !startas
)
596 badusage("--start needs --exec or --startas");
598 if (mpidfile
&& pidfile
== NULL
)
599 badusage("--make-pidfile is only relevant with --pidfile");
601 if (background
&& !start
)
602 badusage("--background is only relevant with --start");
605 static int pid_is_exec(pid_t pid
, const struct stat
*esb
)
610 snprintf(buf
, sizeof(buf
), "/proc/%ld/exe", (long)pid
);
611 if (stat(buf
, &sb
) != 0)
613 return (sb
.st_dev
== esb
->st_dev
&& sb
.st_ino
== esb
->st_ino
);
617 static int pid_is_user(pid_t pid
, uid_t uid
)
622 snprintf(buf
, sizeof(buf
), "/proc/%ld", (long)pid
);
623 if (stat(buf
, &sb
) != 0)
625 return (sb
.st_uid
== uid
);
629 static int pid_is_cmd(pid_t pid
, const char *name
)
635 snprintf(buf
, sizeof(buf
), "/proc/%ld/stat", (long)pid
);
639 while ((c
= getc(f
)) != EOF
&& c
!= '(')
645 /* this hopefully handles command names containing ')' */
646 while ((c
= getc(f
)) != EOF
&& c
== *name
)
649 return (c
== ')' && *name
== '\0');
653 static void check(pid_t pid
)
655 if (execname
&& !pid_is_exec(pid
, &exec_stat
))
657 if (userspec
&& !pid_is_user(pid
, user_id
))
659 if (cmdname
&& !pid_is_cmd(pid
, cmdname
))
664 static void do_pidfile(const char *name
)
669 f
= fopen(name
, "r");
671 if (fscanf(f
, "%ld", &pid
) == 1)
674 } else if (errno
!= ENOENT
)
675 fatal("open pidfile %s: %s", name
, strerror(errno
));
678 /* WTA: this needs to be an autoconf check for /proc/pid existance.
680 static void do_procinit(void)
683 struct dirent
*entry
;
687 procdir
= opendir("/proc");
689 fatal("opendir /proc: %s", strerror(errno
));
692 while ((entry
= readdir(procdir
)) != NULL
) {
693 if (sscanf(entry
->d_name
, "%ld", &pid
) != 1)
700 fatal("nothing in /proc - not mounted?");
703 static void do_findprocs(void)
713 /* return 1 on failure */
714 static void do_stop(int signal_nr
, int quietmode
, int *n_killed
,
715 int *n_notkilled
, int retry_nr
)
729 for (p
= found
; p
; p
= p
->next
) {
731 printf("Would send signal %d to %ld.\n", signal_nr
,
733 else if (kill(p
->pid
, signal_nr
) == 0) {
734 push(&killed
, p
->pid
);
737 printf("%s: warning: failed to kill %ld: %s\n",
738 progname
, (long)p
->pid
, strerror(errno
));
742 if (quietmode
< 0 && killed
) {
743 printf("Stopped %s (pid", what_stop
);
744 for (p
= killed
; p
; p
= p
->next
)
745 printf(" %ld", (long)p
->pid
);
748 printf(", retry #%d", retry_nr
);
754 static void set_what_stop(const char *str
)
756 strncpy(what_stop
, str
, sizeof(what_stop
));
757 what_stop
[sizeof(what_stop
) - 1] = '\0';
760 static int run_stop_schedule(void)
762 int r
, position
, n_killed
, n_notkilled
, value
, ratio
, anykilled
,
764 struct timeval stopat
, before
, after
, interval
, maxinterval
;
767 if (schedule
!= NULL
) {
768 printf("Ignoring --retry in test mode\n");
774 set_what_stop(cmdname
);
776 set_what_stop(execname
);
778 sprintf(what_stop
, "process in pidfile `%.200s'", pidfile
);
780 sprintf(what_stop
, "process(es) owned by `%.200s'", userspec
);
782 fatal("internal error, please report");
788 if (schedule
== NULL
) {
789 do_stop(signal_nr
, quietmode
, &n_killed
, &n_notkilled
, 0);
790 if (n_notkilled
> 0 && quietmode
<= 0)
791 printf("%d pids were not killed\n", n_notkilled
);
797 for (position
= 0; position
< schedule_length
;) {
798 value
= schedule
[position
].value
;
801 switch (schedule
[position
].type
) {
808 do_stop(value
, quietmode
, &n_killed
, &n_notkilled
,
817 /* We want to keep polling for the processes, to see if
819 * or until the timeout expires.
821 * This is a somewhat complicated algorithm to try to
823 * notice reasonably quickly when all the processes have
825 * don't spend too much CPU time polling. In
826 * particular, on a fast
827 * machine with quick-exiting daemons we don't want to
829 * shutdown too much, whereas on a slow one, or where
831 * taking some time to exit, we want to increase the
835 * The algorithm is as follows: we measure the elapsed
837 * to do one poll(), and wait a multiple of this time
839 * poll. However, if that would put us past the end of
841 * period we wait only as long as the timeout period,
843 * we always wait at least MIN_POLL_INTERVAL (20ms).
845 * (`ratio') starts out as 2, and increases by 1 for
847 * maximum of 10; so we use up to between 30% and 10% of
849 * machine's resources (assuming a few reasonable things
853 xgettimeofday(&stopat
);
854 stopat
.tv_sec
+= value
;
857 xgettimeofday(&before
);
858 if (timercmp(&before
, &stopat
, >))
861 do_stop(0, 1, &n_killed
, &n_notkilled
, 0);
865 xgettimeofday(&after
);
867 if (!timercmp(&after
, &stopat
, <))
874 ratio
* (TVELEM(&after
) - TVELEM(&before
)
877 TVELEM(&stopat
) - TVELEM(&after
)
880 if (timercmp(&interval
, &maxinterval
, >))
881 interval
= maxinterval
;
883 if (interval
.tv_sec
== 0
884 && interval
.tv_usec
<= MIN_POLL_INTERVAL
)
885 interval
.tv_usec
= MIN_POLL_INTERVAL
;
887 r
= select(0, 0, 0, 0, &interval
);
888 if (r
< 0 && errno
!= EINTR
)
889 fatal("select() failed for pause: %s",
894 assert(!"schedule[].type value must be valid");
902 printf("Program %s, %d process(es), refused to die.\n",
903 what_stop
, n_killed
);
910 printf("No %s found running; none killed.\n",
919 int main(int argc, char **argv) NONRETURNING;
922 int main(int argc
, char **argv
)
926 LIST_INIT(&namespace_head
);
928 parse_options(argc
, argv
);
932 if (execname
&& stat(execname
, &exec_stat
))
933 fatal("stat %s: %s", execname
, strerror(errno
));
935 if (userspec
&& sscanf(userspec
, "%d", &user_id
) != 1) {
938 pw
= getpwnam(userspec
);
940 fatal("user `%s' not found\n", userspec
);
942 user_id
= pw
->pw_uid
;
945 if (changegroup
&& sscanf(changegroup
, "%d", &runas_gid
) != 1) {
946 struct group
*gr
= getgrnam(changegroup
);
948 fatal("group `%s' not found\n", changegroup
);
949 runas_gid
= gr
->gr_gid
;
951 if (changeuser
&& sscanf(changeuser
, "%d", &runas_uid
) != 1) {
952 struct passwd
*pw
= getpwnam(changeuser
);
954 fatal("user `%s' not found\n", changeuser
);
955 runas_uid
= pw
->pw_uid
;
957 == NULL
) { /* pass the default group of this user */
958 changegroup
= ""; /* just empty */
959 runas_gid
= pw
->pw_gid
;
964 int i
= run_stop_schedule();
972 printf("%s already running.\n", execname
);
976 printf("Would start %s ", startas
);
978 printf("%s ", *argv
++);
979 if (changeuser
!= NULL
) {
980 printf(" (as user %s[%d]", changeuser
, runas_uid
);
981 if (changegroup
!= NULL
)
982 printf(", and group %s[%d])", changegroup
,
987 if (changeroot
!= NULL
)
988 printf(" in directory %s", changeroot
);
990 printf(", and add %i to the priority", nicelevel
);
995 printf("Starting %s...\n", startas
);
997 if (changeroot
!= NULL
) {
998 if (chdir(changeroot
) < 0)
999 fatal("Unable to chdir() to %s", changeroot
);
1000 if (chroot(changeroot
) < 0)
1001 fatal("Unable to chroot() to %s", changeroot
);
1003 if (changeuser
!= NULL
) {
1004 if (setgid(runas_gid
))
1005 fatal("Unable to set gid to %d", runas_gid
);
1006 if (initgroups(changeuser
, runas_gid
))
1007 fatal("Unable to set initgroups() with gid %d",
1009 if (setuid(runas_uid
))
1010 fatal("Unable to set uid to %s", changeuser
);
1013 if (background
) { /* ok, we need to detach this process */
1016 printf("Detaching to start %s...", startas
);
1019 fatal("Unable to fork.\n");
1021 if (i
) { /* parent */
1026 /* child continues here */
1027 /* now close all extra fds */
1028 for (i
= getdtablesize() - 1; i
>= 0; --i
)
1031 fd
= open("/dev/tty", O_RDWR
);
1033 if (ioctl(fd
, TIOCNOTTY
, 0) < 0)
1034 printf("ioctl TIOCNOTTY failed: %s\n",
1039 umask(022); /* set a default for dumb programs */
1040 setpgid(0, 0); /* set the process group */
1041 fd
= open("/dev/null", O_RDWR
); /* stdin */
1043 dup(fd
); /* stdout */
1044 dup(fd
); /* stderr */
1049 if (nice(nicelevel
) < 0 && errno
)
1050 fatal("Unable to alter nice level by %i: %s", nicelevel
,
1054 && pidfile
!= NULL
) { /* user wants _us_ to make the pidfile :) */
1055 FILE *pidf
= fopen(pidfile
, "w");
1056 pid_t pidt
= getpid();
1058 fatal("Unable to open pidfile `%s' for writing: %s",
1059 pidfile
, strerror(errno
));
1060 fprintf(pidf
, "%ld\n", (long)pidt
);
1064 execv(startas
, argv
);
1065 fatal("Unable to start %s: %s", startas
, strerror(errno
));