diff options
| -rw-r--r-- | toys/pending/bootchartd.c | 96 | 
1 files changed, 41 insertions, 55 deletions
| diff --git a/toys/pending/bootchartd.c b/toys/pending/bootchartd.c index 90f37cb5..9b943215 100644 --- a/toys/pending/bootchartd.c +++ b/toys/pending/bootchartd.c @@ -5,21 +5,25 @@   *   * No Standard -USE_BOOTCHARTD(NEWTOY(bootchartd, NULL, TOYFLAG_STAYROOT|TOYFLAG_USR|TOYFLAG_BIN)) +USE_BOOTCHARTD(NEWTOY(bootchartd, 0, TOYFLAG_STAYROOT|TOYFLAG_USR|TOYFLAG_BIN))  config BOOTCHARTD    bool "bootchartd"    default n    help -    usage: bootchartd start [PROG ARGS]|stop|init +    usage: bootchartd {start [PROG ARGS]}|stop|init      Create /var/log/bootlog.tgz with boot chart data -    start: start background logging; with PROG, run PROG, then kill logging with USR1 +    start: start background logging; with PROG, run PROG, +           then kill logging with USR1      stop:  send USR1 to all bootchartd processes -    init:  start background logging; stop when getty/xdm is seen (for init scripts) +    init:  start background logging; stop when getty/xdm is seen +          (for init scripts) +      Under PID 1: as init, then exec $bootchart_init, /init, /sbin/init  */ +  #define FOR_bootchartd  #include "toys.h"  #include <signal.h> @@ -29,32 +33,23 @@ GLOBALS(    long smpl_period_usec;    int proc_accounting;    int is_login; -  int got_signal;  +  int got_signal; + +  void *head;  )  struct pid_list { -  struct pid_list *next; +  struct pid_list *next, *prev;    int pid;  }; -static struct pid_list *head; - -static void insert_pid(struct pid_list **plist, int pid) -{ -  struct pid_list *new_node = xmalloc(sizeof(struct pid_list)); - -  new_node->pid = pid; -  new_node->next = NULL; -  if (!*plist) *plist = new_node; -  else { -    new_node->next = (*plist); -    *plist = new_node; -  } -} -  static int push_pids_in_list(pid_t pid, char *name)  { -  insert_pid(&head, pid); +  struct pid_list *new = xzalloc(sizeof(struct pid_list)); + +  new->pid = pid; +  dlist_add_nomalloc((void *)&TT.head, (void *)new); +    return 0;  } @@ -232,11 +227,11 @@ static void stop_logging(char *tmp_dir, char *prog)    memset(toybuf, 0, sizeof(toybuf));    if ((kcmd_line_fd = open("/proc/cmdline", O_RDONLY)) != -1) {      ssize_t len; +      len = readall(kcmd_line_fd, toybuf, sizeof(toybuf)-1); -    toybuf[len] = '\0'; -    while (--len >= 0 && toybuf[len] == '\0') continue; -    for (; len > 0; len--) -      if ((unsigned char)toybuf[len] < ' ') toybuf[len] = ' '; +    toybuf[len] = 0; +    while (--len >= 0 && !toybuf[len]) continue; +    for (; len > 0; len--) if (toybuf[len] < ' ') toybuf[len] = ' ';    }    fprintf(hdr_fp, "system.kernel.options = %s", toybuf);    close(kcmd_line_fd); @@ -263,52 +258,43 @@ static void signal_handler(int sig)  void bootchartd_main()  {    pid_t lgr_pid, self_pid = getpid(); -  int bchartd_opt;   // 1 -> start, 2 -> stop, 3 -> init, 4 -> PID1 +  int bchartd_opt = 0; // 0=PID1, 1=start, 2=stop, 3=init    TT.smpl_period_usec = 200 * 1000; -  if (toys.optargs[0]) { -    if (!strcmp("start", toys.optargs[0])) bchartd_opt = 1; -    else if (!strcmp("stop", toys.optargs[0])) bchartd_opt = 2; -    else if (!strcmp("init", toys.optargs[0])) bchartd_opt = 3; -    else { -      toys.exithelp++; -      error_exit("Unknown option '%s'", toys.optargs[0]); -    } +  TT.is_login = (self_pid == 1); +  if (*toys.optargs) { +    if (!strcmp("start", *toys.optargs)) bchartd_opt = 1; +    else if (!strcmp("stop", *toys.optargs)) bchartd_opt = 2; +    else if (!strcmp("init", *toys.optargs)) bchartd_opt = 3; +    else error_exit("Unknown option '%s'", *toys.optargs); +      if (bchartd_opt == 2) {        struct pid_list *temp;        char *process_name[] = {"bootchartd", NULL};        names_to_pid(process_name, push_pids_in_list); -      for (temp = head; temp; temp = temp->next)  +      temp = TT.head; +      if (temp) temp->prev->next = 0; +      for (; temp; temp = temp->next)           if (temp->pid != self_pid) kill(temp->pid, SIGUSR1); -      temp = head; -      while (head) { -        temp = head->next; -        free(head); -        head = temp; -      } +      llist_traverse(TT.head, free); +        return;      } -  } else {  -    if (self_pid != 1) { -      toys.exithelp++; -      error_exit("Its Not PID1"); -    } -    bchartd_opt = 4; // PID1 case -  } +  } else if (!TT.is_login) error_exit("not PID 1"); +    // Execute the code below for start or init or PID1     if (!parse_config_file("bootchartd.conf"))      parse_config_file("/etc/bootchartd.conf"); -  TT.is_login = (self_pid == 1);    memset(toybuf, 0, sizeof(toybuf));    if (!(lgr_pid = fork())) {      char *tmp_dir = create_tmp_dir();      sigatexit(signal_handler);      raise(SIGSTOP); -    if (bchartd_opt == 4 && !getenv("PATH"))  -      putenv((char*)"PATH=/sbin:/usr/sbin:/bin:/usr/bin"); +    if (!bchartd_opt && !getenv("PATH"))  +      putenv("PATH=/sbin:/usr/sbin:/bin:/usr/bin");      start_logging();      stop_logging(tmp_dir, bchartd_opt == 1 ? toys.optargs[1] : NULL);      free(tmp_dir); @@ -317,12 +303,12 @@ void bootchartd_main()    waitpid(lgr_pid, NULL, WUNTRACED);    kill(lgr_pid, SIGCONT); -  if (bchartd_opt == 4) {  +  if (!bchartd_opt) {       char *pbchart_init = getenv("bootchart_init");      if (pbchart_init) execl(pbchart_init, pbchart_init, NULL); -    execl("/init", "init", (char *) 0); -    execl("/sbin/init", "init", (char *) 0); +    execl("/init", "init", (void *)0); +    execl("/sbin/init", "init", (void *)0);    }    if (bchartd_opt == 1 && toys.optargs[1]) {       pid_t prog_pid; | 
