ntpd: print result of hostname resolution
[oweals/busybox.git] / util-linux / acpid.c
index 5a3bd5a6af14f385cd90bc0acd12aea330283947..3e68b61abf65a9fcaca46ee936e26e88b759fd34 100644 (file)
@@ -6,19 +6,87 @@
  *
  * Licensed under GPLv2, see file LICENSE in this source tree.
  */
+//config:config ACPID
+//config:      bool "acpid"
+//config:      default y
+//config:      select PLATFORM_LINUX
+//config:      help
+//config:        acpid listens to ACPI events coming either in textual form from
+//config:        /proc/acpi/event (though it is marked deprecated it is still widely
+//config:        used and _is_ a standard) or in binary form from specified evdevs
+//config:        (just use /dev/input/event*).
+//config:
+//config:        It parses the event to retrieve ACTION and a possible PARAMETER.
+//config:        It then spawns /etc/acpi/<ACTION>[/<PARAMETER>] either via run-parts
+//config:        (if the resulting path is a directory) or directly as an executable.
+//config:
+//config:        N.B. acpid relies on run-parts so have the latter installed.
+//config:
+//config:config FEATURE_ACPID_COMPAT
+//config:      bool "Accept and ignore redundant options"
+//config:      default y
+//config:      depends on ACPID
+//config:      help
+//config:        Accept and ignore compatibility options -g -m -s -S -v.
+
+//applet:IF_ACPID(APPLET(acpid, BB_DIR_SBIN, BB_SUID_DROP))
+
+//kbuild:lib-$(CONFIG_ACPID) += acpid.o
+
+//usage:#define acpid_trivial_usage
+//usage:       "[-df] [-c CONFDIR] [-l LOGFILE] [-a ACTIONFILE] [-M MAPFILE] [-e PROC_EVENT_FILE] [-p PIDFILE]"
+//usage:#define acpid_full_usage "\n\n"
+//usage:       "Listen to ACPI events and spawn specific helpers on event arrival\n"
+//usage:     "\n       -d      Log to stderr, not log file (implies -f)"
+//usage:     "\n       -f      Run in foreground"
+//usage:     "\n       -c DIR  Config directory [/etc/acpi]"
+//usage:     "\n       -e FILE /proc event file [/proc/acpi/event]"
+//usage:     "\n       -l FILE Log file [/var/log/acpid.log]"
+//usage:     "\n       -p FILE Pid file [/var/run/acpid.pid]"
+//usage:     "\n       -a FILE Action file [/etc/acpid.conf]"
+//usage:     "\n       -M FILE Map file [/etc/acpi.map]"
+//usage:       IF_FEATURE_ACPID_COMPAT(
+//usage:     "\n\nAccept and ignore compatibility options -g -m -s -S -v"
+//usage:       )
+//usage:
+//usage:#define acpid_example_usage
+//usage:       "Without -e option, acpid uses all /dev/input/event* files\n"
+//usage:       "# acpid\n"
+//usage:       "# acpid -l /var/log/my-acpi-log\n"
+//usage:       "# acpid -e /proc/acpi/event\n"
+
 #include "libbb.h"
 #include <syslog.h>
 #include <linux/input.h>
 
+#ifndef EV_SW
+# define EV_SW         0x05
+#endif
+#ifndef EV_KEY
+# define EV_KEY        0x01
+#endif
+#ifndef SW_LID
+# define SW_LID        0x00
+#endif
+#ifndef SW_RFKILL_ALL
+# define SW_RFKILL_ALL 0x03
+#endif
+#ifndef KEY_POWER
+# define KEY_POWER      116     /* SC System Power Down */
+#endif
+#ifndef KEY_SLEEP
+# define KEY_SLEEP      142     /* SC System Sleep */
+#endif
+
 enum {
        OPT_c = (1 << 0),
        OPT_d = (1 << 1),
        OPT_e = (1 << 2),
        OPT_f = (1 << 3),
        OPT_l = (1 << 4),
-       OPT_p = (1 << 5) * ENABLE_FEATURE_PIDFILE,
-       OPT_a = (1 << 6),
-       OPT_M = (1 << 7),
+       OPT_a = (1 << 5),
+       OPT_M = (1 << 6),
+       OPT_p = (1 << 7) * ENABLE_FEATURE_PIDFILE,
 };
 
 struct acpi_event {
@@ -33,6 +101,7 @@ struct acpi_event {
 static const struct acpi_event f_evt_tab[] = {
        { "EV_KEY", 0x01, "KEY_POWER", 116, 1, "button/power PWRF 00000080" },
        { "EV_KEY", 0x01, "KEY_POWER", 116, 1, "button/power PWRB 00000080" },
+       { "EV_SW", 0x05, "SW_LID", 0x00, 1, "button/lid LID0 00000080" },
 };
 
 struct acpi_action {
@@ -52,7 +121,7 @@ struct globals {
        int n_evt;
 } FIX_ALIASING;
 #define G (*ptr_to_globals)
-#define        act_tab         (G.act_tab)
+#define act_tab         (G.act_tab)
 #define n_act           (G.n_act  )
 #define evt_tab         (G.evt_tab)
 #define n_evt           (G.n_evt  )
@@ -78,10 +147,8 @@ static void process_event(const char *event)
        char *handler = xasprintf("./%s", event);
        const char *args[] = { "run-parts", handler, NULL };
 
-       // debug info
-       if (option_mask32 & OPT_d) {
-               bb_error_msg("%s", event);
-       }
+       // log the event
+       bb_error_msg("%s", event);
 
        // spawn handler
        // N.B. run-parts would require scripts to have #!/bin/sh
@@ -110,7 +177,7 @@ static const char *find_action(struct input_event *ev, const char *buf)
                }
 
                if (buf) {
-                       if (strncmp(buf, evt_tab[i].desc, strlen(buf)) == 0) {
+                       if (is_prefixed_with(evt_tab[i].desc, buf)) {
                                action = evt_tab[i].desc;
                                break;
                        }
@@ -183,7 +250,6 @@ static void parse_map_file(const char *filename)
 int acpid_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
 int acpid_main(int argc UNUSED_PARAM, char **argv)
 {
-       struct input_event ev;
        int nfd;
        int opts;
        struct pollfd *pfd;
@@ -193,35 +259,46 @@ int acpid_main(int argc UNUSED_PARAM, char **argv)
        const char *opt_action = "/etc/acpid.conf";
        const char *opt_map = "/etc/acpi.map";
 #if ENABLE_FEATURE_PIDFILE
-       const char *opt_pidfile = "/var/run/acpid.pid";
+       const char *opt_pidfile = CONFIG_PID_FILE_PATH "/acpid.pid";
 #endif
 
        INIT_G();
 
        opt_complementary = "df:e--e";
-       opts = getopt32(argv, "c:de:fl:p:a:M:" IF_FEATURE_ACPID_COMPAT("g:m:s:S:v"),
-               &opt_dir, &opt_input, &opt_logfile, &opt_pidfile, &opt_action, &opt_map
+       opts = getopt32(argv, "c:de:fl:a:M:" IF_FEATURE_PIDFILE("p:") IF_FEATURE_ACPID_COMPAT("g:m:s:S:v"),
+               &opt_dir, &opt_input, &opt_logfile, &opt_action, &opt_map
+               IF_FEATURE_PIDFILE(, &opt_pidfile)
                IF_FEATURE_ACPID_COMPAT(, NULL, NULL, NULL, NULL)
        );
 
        if (!(opts & OPT_f)) {
+               /* No -f "Foreground", we go to background */
                bb_daemonize_or_rexec(DAEMON_CLOSE_EXTRA_FDS, argv);
        }
 
        if (!(opts & OPT_d)) {
+               /* No -d "Debug", we log to log file.
+                * This includes any output from children.
+                */
+               xmove_fd(xopen(opt_logfile, O_WRONLY | O_CREAT | O_APPEND), STDOUT_FILENO);
+               xdup2(STDOUT_FILENO, STDERR_FILENO);
+               /* Also, acpid's messages (but not children) will go to syslog too */
                openlog(applet_name, LOG_PID, LOG_DAEMON);
                logmode = LOGMODE_SYSLOG | LOGMODE_STDIO;
-       } else {
-               xmove_fd(xopen(opt_logfile, O_WRONLY | O_CREAT | O_TRUNC), STDOUT_FILENO);
        }
+       /* else: -d "Debug", log is not redirected */
 
        parse_conf_file(opt_action);
        parse_map_file(opt_map);
 
        xchdir(opt_dir);
 
+       /* We spawn children but don't wait for them. Prevent zombies: */
        bb_signals((1 << SIGCHLD), SIG_IGN);
-       bb_signals(BB_FATAL_SIGS, record_signo);
+       // If you enable this, (1) explain why, (2)
+       // make sure while(poll) loop below is still interruptible
+       // by SIGTERM et al:
+       //bb_signals(BB_FATAL_SIGS, record_signo);
 
        pfd = NULL;
        nfd = 0;
@@ -229,13 +306,14 @@ int acpid_main(int argc UNUSED_PARAM, char **argv)
                int fd;
                char *dev_event;
 
-               dev_event = xasprintf((option_mask32 & OPT_e) ? "%s" : "%s%u", opt_input, nfd);
+               dev_event = xasprintf((opts & OPT_e) ? "%s" : "%s%u", opt_input, nfd);
                fd = open(dev_event, O_RDONLY | O_NONBLOCK);
                if (fd < 0) {
                        if (nfd == 0)
                                bb_simple_perror_msg_and_die(dev_event);
                        break;
                }
+               free(dev_event);
                pfd = xrealloc_vector(pfd, 1, nfd);
                pfd[nfd].fd = fd;
                pfd[nfd].events = POLLIN;
@@ -244,27 +322,40 @@ int acpid_main(int argc UNUSED_PARAM, char **argv)
 
        write_pidfile(opt_pidfile);
 
-       while (poll(pfd, nfd, -1) > 0) {
+       while (safe_poll(pfd, nfd, -1) > 0) {
                int i;
                for (i = 0; i < nfd; i++) {
-                       const char *event = NULL;
-
-                       memset(&ev, 0, sizeof(ev));
-
-                       if (!(pfd[i].revents & POLLIN))
-                               continue;
+                       const char *event;
+
+                       if (!(pfd[i].revents & POLLIN)) {
+                               if (pfd[i].revents == 0)
+                                       continue; /* this fd has nothing */
+
+                               /* Likely POLLERR, POLLHUP, POLLNVAL.
+                                * Do not listen on this fd anymore.
+                                */
+                               close(pfd[i].fd);
+                               nfd--;
+                               for (; i < nfd; i++)
+                                       pfd[i].fd = pfd[i + 1].fd;
+                               break; /* do poll() again */
+                       }
 
+                       event = NULL;
                        if (option_mask32 & OPT_e) {
                                char *buf;
                                int len;
 
-                               buf = xmalloc_reads(pfd[i].fd, NULL, NULL);
+                               buf = xmalloc_reads(pfd[i].fd, NULL);
                                /* buf = "button/power PWRB 00000080 00000000" */
                                len = strlen(buf) - 9;
                                if (len >= 0)
                                        buf[len] = '\0';
                                event = find_action(NULL, buf);
+                               free(buf);
                        } else {
+                               struct input_event ev;
+
                                if (sizeof(ev) != full_read(pfd[i].fd, &ev, sizeof(ev)))
                                        continue;
 
@@ -275,17 +366,14 @@ int acpid_main(int argc UNUSED_PARAM, char **argv)
                        }
                        if (!event)
                                continue;
-                       // spawn event handler
+                       /* spawn event handler */
                        process_event(event);
                }
        }
 
        if (ENABLE_FEATURE_CLEAN_UP) {
-               while (nfd--) {
-                       if (pfd[nfd].fd) {
-                               close(pfd[nfd].fd);
-                       }
-               }
+               while (nfd--)
+                       close(pfd[nfd].fd);
                free(pfd);
        }
        remove_pidfile(opt_pidfile);