1*61145dc2SMartin Matuska // SPDX-License-Identifier: CDDL-1.0
2eda14cbcSMatt Macy /*
3180f8225SMatt Macy * This file is part of the ZFS Event Daemon (ZED).
4180f8225SMatt Macy *
5eda14cbcSMatt Macy * Developed at Lawrence Livermore National Laboratory (LLNL-CODE-403049).
6eda14cbcSMatt Macy * Copyright (C) 2013-2014 Lawrence Livermore National Security, LLC.
716038816SMartin Matuska * Refer to the OpenZFS git commit log for authoritative copyright attribution.
8eda14cbcSMatt Macy *
9eda14cbcSMatt Macy * The contents of this file are subject to the terms of the
10eda14cbcSMatt Macy * Common Development and Distribution License Version 1.0 (CDDL-1.0).
11eda14cbcSMatt Macy * You can obtain a copy of the license from the top-level file
12eda14cbcSMatt Macy * "OPENSOLARIS.LICENSE" or at <http://opensource.org/licenses/CDDL-1.0>.
13eda14cbcSMatt Macy * You may not use this file except in compliance with the license.
14eda14cbcSMatt Macy */
15eda14cbcSMatt Macy
16eda14cbcSMatt Macy #include <assert.h>
17eda14cbcSMatt Macy #include <ctype.h>
18eda14cbcSMatt Macy #include <dirent.h>
19eda14cbcSMatt Macy #include <errno.h>
20eda14cbcSMatt Macy #include <fcntl.h>
21eda14cbcSMatt Macy #include <libgen.h>
22eda14cbcSMatt Macy #include <limits.h>
23eda14cbcSMatt Macy #include <stdio.h>
24eda14cbcSMatt Macy #include <stdlib.h>
25eda14cbcSMatt Macy #include <string.h>
2653b70c86SMartin Matuska #include <sys/types.h>
27eda14cbcSMatt Macy #include <sys/stat.h>
28eda14cbcSMatt Macy #include <sys/uio.h>
29eda14cbcSMatt Macy #include <unistd.h>
30eda14cbcSMatt Macy #include "zed.h"
31eda14cbcSMatt Macy #include "zed_conf.h"
32eda14cbcSMatt Macy #include "zed_file.h"
33eda14cbcSMatt Macy #include "zed_log.h"
34eda14cbcSMatt Macy #include "zed_strings.h"
35eda14cbcSMatt Macy
36eda14cbcSMatt Macy /*
3716038816SMartin Matuska * Initialise the configuration with default values.
38eda14cbcSMatt Macy */
3916038816SMartin Matuska void
zed_conf_init(struct zed_conf * zcp)4016038816SMartin Matuska zed_conf_init(struct zed_conf *zcp)
41eda14cbcSMatt Macy {
4216038816SMartin Matuska memset(zcp, 0, sizeof (*zcp));
43eda14cbcSMatt Macy
4416038816SMartin Matuska /* zcp->zfs_hdl opened in zed_event_init() */
4516038816SMartin Matuska /* zcp->zedlets created in zed_conf_scan_dir() */
46eda14cbcSMatt Macy
4716038816SMartin Matuska zcp->pid_fd = -1; /* opened in zed_conf_write_pid() */
4816038816SMartin Matuska zcp->state_fd = -1; /* opened in zed_conf_open_state() */
4916038816SMartin Matuska zcp->zevent_fd = -1; /* opened in zed_event_init() */
50eda14cbcSMatt Macy
5116038816SMartin Matuska zcp->max_jobs = 16;
5208aba0aeSMartin Matuska zcp->max_zevent_buf_len = 1 << 20;
53eda14cbcSMatt Macy
5416038816SMartin Matuska if (!(zcp->pid_file = strdup(ZED_PID_FILE)) ||
5516038816SMartin Matuska !(zcp->zedlet_dir = strdup(ZED_ZEDLET_DIR)) ||
5616038816SMartin Matuska !(zcp->state_file = strdup(ZED_STATE_FILE)))
57eda14cbcSMatt Macy zed_log_die("Failed to create conf: %s", strerror(errno));
58eda14cbcSMatt Macy }
59eda14cbcSMatt Macy
60eda14cbcSMatt Macy /*
61eda14cbcSMatt Macy * Destroy the configuration [zcp].
62eda14cbcSMatt Macy *
63eda14cbcSMatt Macy * Note: zfs_hdl & zevent_fd are destroyed via zed_event_fini().
64eda14cbcSMatt Macy */
65eda14cbcSMatt Macy void
zed_conf_destroy(struct zed_conf * zcp)66eda14cbcSMatt Macy zed_conf_destroy(struct zed_conf *zcp)
67eda14cbcSMatt Macy {
68eda14cbcSMatt Macy if (zcp->state_fd >= 0) {
69eda14cbcSMatt Macy if (close(zcp->state_fd) < 0)
70eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
71eda14cbcSMatt Macy "Failed to close state file \"%s\": %s",
72eda14cbcSMatt Macy zcp->state_file, strerror(errno));
73eda14cbcSMatt Macy zcp->state_fd = -1;
74eda14cbcSMatt Macy }
75eda14cbcSMatt Macy if (zcp->pid_file) {
76eda14cbcSMatt Macy if ((unlink(zcp->pid_file) < 0) && (errno != ENOENT))
77eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
78eda14cbcSMatt Macy "Failed to remove PID file \"%s\": %s",
79eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
80eda14cbcSMatt Macy }
81eda14cbcSMatt Macy if (zcp->pid_fd >= 0) {
82eda14cbcSMatt Macy if (close(zcp->pid_fd) < 0)
83eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
84eda14cbcSMatt Macy "Failed to close PID file \"%s\": %s",
85eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
86eda14cbcSMatt Macy zcp->pid_fd = -1;
87eda14cbcSMatt Macy }
88eda14cbcSMatt Macy if (zcp->pid_file) {
89eda14cbcSMatt Macy free(zcp->pid_file);
90eda14cbcSMatt Macy zcp->pid_file = NULL;
91eda14cbcSMatt Macy }
92eda14cbcSMatt Macy if (zcp->zedlet_dir) {
93eda14cbcSMatt Macy free(zcp->zedlet_dir);
94eda14cbcSMatt Macy zcp->zedlet_dir = NULL;
95eda14cbcSMatt Macy }
96eda14cbcSMatt Macy if (zcp->state_file) {
97eda14cbcSMatt Macy free(zcp->state_file);
98eda14cbcSMatt Macy zcp->state_file = NULL;
99eda14cbcSMatt Macy }
100eda14cbcSMatt Macy if (zcp->zedlets) {
101eda14cbcSMatt Macy zed_strings_destroy(zcp->zedlets);
102eda14cbcSMatt Macy zcp->zedlets = NULL;
103eda14cbcSMatt Macy }
104eda14cbcSMatt Macy }
105eda14cbcSMatt Macy
106eda14cbcSMatt Macy /*
107eda14cbcSMatt Macy * Display command-line help and exit.
108eda14cbcSMatt Macy *
109eda14cbcSMatt Macy * If [got_err] is 0, output to stdout and exit normally;
110eda14cbcSMatt Macy * otherwise, output to stderr and exit with a failure status.
111eda14cbcSMatt Macy */
112eda14cbcSMatt Macy static void
_zed_conf_display_help(const char * prog,boolean_t got_err)11316038816SMartin Matuska _zed_conf_display_help(const char *prog, boolean_t got_err)
114eda14cbcSMatt Macy {
11516038816SMartin Matuska struct opt { const char *o, *d, *v; };
11616038816SMartin Matuska
117eda14cbcSMatt Macy FILE *fp = got_err ? stderr : stdout;
11816038816SMartin Matuska
11916038816SMartin Matuska struct opt *oo;
12016038816SMartin Matuska struct opt iopts[] = {
12116038816SMartin Matuska { .o = "-h", .d = "Display help" },
12216038816SMartin Matuska { .o = "-L", .d = "Display license information" },
12316038816SMartin Matuska { .o = "-V", .d = "Display version information" },
12416038816SMartin Matuska {},
12516038816SMartin Matuska };
12616038816SMartin Matuska struct opt nopts[] = {
12716038816SMartin Matuska { .o = "-v", .d = "Be verbose" },
12816038816SMartin Matuska { .o = "-f", .d = "Force daemon to run" },
12916038816SMartin Matuska { .o = "-F", .d = "Run daemon in the foreground" },
13016038816SMartin Matuska { .o = "-I",
13116038816SMartin Matuska .d = "Idle daemon until kernel module is (re)loaded" },
13216038816SMartin Matuska { .o = "-M", .d = "Lock all pages in memory" },
13316038816SMartin Matuska { .o = "-P", .d = "$PATH for ZED to use (only used by ZTS)" },
13416038816SMartin Matuska { .o = "-Z", .d = "Zero state file" },
13516038816SMartin Matuska {},
13616038816SMartin Matuska };
13716038816SMartin Matuska struct opt vopts[] = {
13816038816SMartin Matuska { .o = "-d DIR", .d = "Read enabled ZEDLETs from DIR.",
13916038816SMartin Matuska .v = ZED_ZEDLET_DIR },
14016038816SMartin Matuska { .o = "-p FILE", .d = "Write daemon's PID to FILE.",
14116038816SMartin Matuska .v = ZED_PID_FILE },
14216038816SMartin Matuska { .o = "-s FILE", .d = "Write daemon's state to FILE.",
14316038816SMartin Matuska .v = ZED_STATE_FILE },
14416038816SMartin Matuska { .o = "-j JOBS", .d = "Start at most JOBS at once.",
14516038816SMartin Matuska .v = "16" },
14608aba0aeSMartin Matuska { .o = "-b LEN", .d = "Cap kernel event buffer at LEN entries.",
14708aba0aeSMartin Matuska .v = "1048576" },
14816038816SMartin Matuska {},
14916038816SMartin Matuska };
150eda14cbcSMatt Macy
151eda14cbcSMatt Macy fprintf(fp, "Usage: %s [OPTION]...\n", (prog ? prog : "zed"));
152eda14cbcSMatt Macy fprintf(fp, "\n");
15316038816SMartin Matuska for (oo = iopts; oo->o; ++oo)
15416038816SMartin Matuska fprintf(fp, " %*s %s\n", -8, oo->o, oo->d);
155eda14cbcSMatt Macy fprintf(fp, "\n");
15616038816SMartin Matuska for (oo = nopts; oo->o; ++oo)
15716038816SMartin Matuska fprintf(fp, " %*s %s\n", -8, oo->o, oo->d);
158eda14cbcSMatt Macy fprintf(fp, "\n");
15916038816SMartin Matuska for (oo = vopts; oo->o; ++oo)
16016038816SMartin Matuska fprintf(fp, " %*s %s [%s]\n", -8, oo->o, oo->d, oo->v);
161eda14cbcSMatt Macy fprintf(fp, "\n");
162eda14cbcSMatt Macy
163eda14cbcSMatt Macy exit(got_err ? EXIT_FAILURE : EXIT_SUCCESS);
164eda14cbcSMatt Macy }
165eda14cbcSMatt Macy
166eda14cbcSMatt Macy /*
167eda14cbcSMatt Macy * Display license information to stdout and exit.
168eda14cbcSMatt Macy */
169eda14cbcSMatt Macy static void
_zed_conf_display_license(void)170eda14cbcSMatt Macy _zed_conf_display_license(void)
171eda14cbcSMatt Macy {
17216038816SMartin Matuska printf(
17316038816SMartin Matuska "The ZFS Event Daemon (ZED) is distributed under the terms of the\n"
17416038816SMartin Matuska " Common Development and Distribution License (CDDL-1.0)\n"
17516038816SMartin Matuska " <http://opensource.org/licenses/CDDL-1.0>.\n"
17616038816SMartin Matuska "\n"
177eda14cbcSMatt Macy "Developed at Lawrence Livermore National Laboratory"
17816038816SMartin Matuska " (LLNL-CODE-403049).\n"
17916038816SMartin Matuska "\n");
180eda14cbcSMatt Macy
181eda14cbcSMatt Macy exit(EXIT_SUCCESS);
182eda14cbcSMatt Macy }
183eda14cbcSMatt Macy
184eda14cbcSMatt Macy /*
185eda14cbcSMatt Macy * Display version information to stdout and exit.
186eda14cbcSMatt Macy */
187eda14cbcSMatt Macy static void
_zed_conf_display_version(void)188eda14cbcSMatt Macy _zed_conf_display_version(void)
189eda14cbcSMatt Macy {
190eda14cbcSMatt Macy printf("%s-%s-%s\n",
191eda14cbcSMatt Macy ZFS_META_NAME, ZFS_META_VERSION, ZFS_META_RELEASE);
192eda14cbcSMatt Macy
193eda14cbcSMatt Macy exit(EXIT_SUCCESS);
194eda14cbcSMatt Macy }
195eda14cbcSMatt Macy
196eda14cbcSMatt Macy /*
197eda14cbcSMatt Macy * Copy the [path] string to the [resultp] ptr.
198eda14cbcSMatt Macy * If [path] is not an absolute path, prefix it with the current working dir.
199eda14cbcSMatt Macy * If [resultp] is non-null, free its existing string before assignment.
200eda14cbcSMatt Macy */
201eda14cbcSMatt Macy static void
_zed_conf_parse_path(char ** resultp,const char * path)202eda14cbcSMatt Macy _zed_conf_parse_path(char **resultp, const char *path)
203eda14cbcSMatt Macy {
204eda14cbcSMatt Macy char buf[PATH_MAX];
205eda14cbcSMatt Macy
206eda14cbcSMatt Macy assert(resultp != NULL);
207eda14cbcSMatt Macy assert(path != NULL);
208eda14cbcSMatt Macy
209eda14cbcSMatt Macy if (*resultp)
210eda14cbcSMatt Macy free(*resultp);
211eda14cbcSMatt Macy
212eda14cbcSMatt Macy if (path[0] == '/') {
213eda14cbcSMatt Macy *resultp = strdup(path);
21416038816SMartin Matuska } else {
21516038816SMartin Matuska if (!getcwd(buf, sizeof (buf)))
216eda14cbcSMatt Macy zed_log_die("Failed to get current working dir: %s",
217eda14cbcSMatt Macy strerror(errno));
21816038816SMartin Matuska
21916038816SMartin Matuska if (strlcat(buf, "/", sizeof (buf)) >= sizeof (buf) ||
22016038816SMartin Matuska strlcat(buf, path, sizeof (buf)) >= sizeof (buf))
22116038816SMartin Matuska zed_log_die("Failed to copy path: %s",
22216038816SMartin Matuska strerror(ENAMETOOLONG));
22316038816SMartin Matuska
224eda14cbcSMatt Macy *resultp = strdup(buf);
225eda14cbcSMatt Macy }
22616038816SMartin Matuska
227eda14cbcSMatt Macy if (!*resultp)
228eda14cbcSMatt Macy zed_log_die("Failed to copy path: %s", strerror(ENOMEM));
229eda14cbcSMatt Macy }
230eda14cbcSMatt Macy
231eda14cbcSMatt Macy /*
232eda14cbcSMatt Macy * Parse the command-line options into the configuration [zcp].
233eda14cbcSMatt Macy */
234eda14cbcSMatt Macy void
zed_conf_parse_opts(struct zed_conf * zcp,int argc,char ** argv)235eda14cbcSMatt Macy zed_conf_parse_opts(struct zed_conf *zcp, int argc, char **argv)
236eda14cbcSMatt Macy {
23708aba0aeSMartin Matuska const char * const opts = ":hLVd:p:P:s:vfFMZIj:b:";
238eda14cbcSMatt Macy int opt;
23916038816SMartin Matuska unsigned long raw;
240eda14cbcSMatt Macy
241eda14cbcSMatt Macy if (!zcp || !argv || !argv[0])
242eda14cbcSMatt Macy zed_log_die("Failed to parse options: Internal error");
243eda14cbcSMatt Macy
244eda14cbcSMatt Macy opterr = 0; /* suppress default getopt err msgs */
245eda14cbcSMatt Macy
246eda14cbcSMatt Macy while ((opt = getopt(argc, argv, opts)) != -1) {
247eda14cbcSMatt Macy switch (opt) {
248eda14cbcSMatt Macy case 'h':
24916038816SMartin Matuska _zed_conf_display_help(argv[0], B_FALSE);
250eda14cbcSMatt Macy break;
251eda14cbcSMatt Macy case 'L':
252eda14cbcSMatt Macy _zed_conf_display_license();
253eda14cbcSMatt Macy break;
254eda14cbcSMatt Macy case 'V':
255eda14cbcSMatt Macy _zed_conf_display_version();
256eda14cbcSMatt Macy break;
257eda14cbcSMatt Macy case 'd':
258eda14cbcSMatt Macy _zed_conf_parse_path(&zcp->zedlet_dir, optarg);
259eda14cbcSMatt Macy break;
260eda14cbcSMatt Macy case 'I':
261eda14cbcSMatt Macy zcp->do_idle = 1;
262eda14cbcSMatt Macy break;
263eda14cbcSMatt Macy case 'p':
264eda14cbcSMatt Macy _zed_conf_parse_path(&zcp->pid_file, optarg);
265eda14cbcSMatt Macy break;
266eda14cbcSMatt Macy case 'P':
267eda14cbcSMatt Macy _zed_conf_parse_path(&zcp->path, optarg);
268eda14cbcSMatt Macy break;
269eda14cbcSMatt Macy case 's':
270eda14cbcSMatt Macy _zed_conf_parse_path(&zcp->state_file, optarg);
271eda14cbcSMatt Macy break;
272eda14cbcSMatt Macy case 'v':
273eda14cbcSMatt Macy zcp->do_verbose = 1;
274eda14cbcSMatt Macy break;
275eda14cbcSMatt Macy case 'f':
276eda14cbcSMatt Macy zcp->do_force = 1;
277eda14cbcSMatt Macy break;
278eda14cbcSMatt Macy case 'F':
279eda14cbcSMatt Macy zcp->do_foreground = 1;
280eda14cbcSMatt Macy break;
281eda14cbcSMatt Macy case 'M':
282eda14cbcSMatt Macy zcp->do_memlock = 1;
283eda14cbcSMatt Macy break;
284eda14cbcSMatt Macy case 'Z':
285eda14cbcSMatt Macy zcp->do_zero = 1;
286eda14cbcSMatt Macy break;
28716038816SMartin Matuska case 'j':
28816038816SMartin Matuska errno = 0;
28916038816SMartin Matuska raw = strtoul(optarg, NULL, 0);
29016038816SMartin Matuska if (errno == ERANGE || raw > INT16_MAX) {
29116038816SMartin Matuska zed_log_die("%lu is too many jobs", raw);
29216038816SMartin Matuska } if (raw == 0) {
29316038816SMartin Matuska zed_log_die("0 jobs makes no sense");
29416038816SMartin Matuska } else {
29516038816SMartin Matuska zcp->max_jobs = raw;
29616038816SMartin Matuska }
29716038816SMartin Matuska break;
29808aba0aeSMartin Matuska case 'b':
29908aba0aeSMartin Matuska errno = 0;
30008aba0aeSMartin Matuska raw = strtoul(optarg, NULL, 0);
30108aba0aeSMartin Matuska if (errno == ERANGE || raw > INT32_MAX) {
30208aba0aeSMartin Matuska zed_log_die("%lu is too large", raw);
30308aba0aeSMartin Matuska } if (raw == 0) {
30408aba0aeSMartin Matuska zcp->max_zevent_buf_len = INT32_MAX;
30508aba0aeSMartin Matuska } else {
30608aba0aeSMartin Matuska zcp->max_zevent_buf_len = raw;
30708aba0aeSMartin Matuska }
30808aba0aeSMartin Matuska break;
309eda14cbcSMatt Macy case '?':
310eda14cbcSMatt Macy default:
311eda14cbcSMatt Macy if (optopt == '?')
31216038816SMartin Matuska _zed_conf_display_help(argv[0], B_FALSE);
313eda14cbcSMatt Macy
31416038816SMartin Matuska fprintf(stderr, "%s: Invalid option '-%c'\n\n",
31516038816SMartin Matuska argv[0], optopt);
31616038816SMartin Matuska _zed_conf_display_help(argv[0], B_TRUE);
317eda14cbcSMatt Macy break;
318eda14cbcSMatt Macy }
319eda14cbcSMatt Macy }
320eda14cbcSMatt Macy }
321eda14cbcSMatt Macy
322eda14cbcSMatt Macy /*
323eda14cbcSMatt Macy * Scan the [zcp] zedlet_dir for files to exec based on the event class.
324eda14cbcSMatt Macy * Files must be executable by user, but not writable by group or other.
325eda14cbcSMatt Macy * Dotfiles are ignored.
326eda14cbcSMatt Macy *
327eda14cbcSMatt Macy * Return 0 on success with an updated set of zedlets,
328eda14cbcSMatt Macy * or -1 on error with errno set.
329eda14cbcSMatt Macy */
330eda14cbcSMatt Macy int
zed_conf_scan_dir(struct zed_conf * zcp)331eda14cbcSMatt Macy zed_conf_scan_dir(struct zed_conf *zcp)
332eda14cbcSMatt Macy {
333eda14cbcSMatt Macy zed_strings_t *zedlets;
334eda14cbcSMatt Macy DIR *dirp;
335eda14cbcSMatt Macy struct dirent *direntp;
336eda14cbcSMatt Macy char pathname[PATH_MAX];
337eda14cbcSMatt Macy struct stat st;
338eda14cbcSMatt Macy int n;
339eda14cbcSMatt Macy
340eda14cbcSMatt Macy if (!zcp) {
341eda14cbcSMatt Macy errno = EINVAL;
342eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to scan zedlet dir: %s",
343eda14cbcSMatt Macy strerror(errno));
344eda14cbcSMatt Macy return (-1);
345eda14cbcSMatt Macy }
346eda14cbcSMatt Macy zedlets = zed_strings_create();
347eda14cbcSMatt Macy if (!zedlets) {
348eda14cbcSMatt Macy errno = ENOMEM;
349eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to scan dir \"%s\": %s",
350eda14cbcSMatt Macy zcp->zedlet_dir, strerror(errno));
351eda14cbcSMatt Macy return (-1);
352eda14cbcSMatt Macy }
353eda14cbcSMatt Macy dirp = opendir(zcp->zedlet_dir);
354eda14cbcSMatt Macy if (!dirp) {
355eda14cbcSMatt Macy int errno_bak = errno;
356eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to open dir \"%s\": %s",
357eda14cbcSMatt Macy zcp->zedlet_dir, strerror(errno));
358eda14cbcSMatt Macy zed_strings_destroy(zedlets);
359eda14cbcSMatt Macy errno = errno_bak;
360eda14cbcSMatt Macy return (-1);
361eda14cbcSMatt Macy }
362eda14cbcSMatt Macy while ((direntp = readdir(dirp))) {
363eda14cbcSMatt Macy if (direntp->d_name[0] == '.')
364eda14cbcSMatt Macy continue;
365eda14cbcSMatt Macy
366eda14cbcSMatt Macy n = snprintf(pathname, sizeof (pathname),
367eda14cbcSMatt Macy "%s/%s", zcp->zedlet_dir, direntp->d_name);
368eda14cbcSMatt Macy if ((n < 0) || (n >= sizeof (pathname))) {
369eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to stat \"%s\": %s",
370eda14cbcSMatt Macy direntp->d_name, strerror(ENAMETOOLONG));
371eda14cbcSMatt Macy continue;
372eda14cbcSMatt Macy }
373eda14cbcSMatt Macy if (stat(pathname, &st) < 0) {
374eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to stat \"%s\": %s",
375eda14cbcSMatt Macy pathname, strerror(errno));
376eda14cbcSMatt Macy continue;
377eda14cbcSMatt Macy }
378eda14cbcSMatt Macy if (!S_ISREG(st.st_mode)) {
379eda14cbcSMatt Macy zed_log_msg(LOG_INFO,
380eda14cbcSMatt Macy "Ignoring \"%s\": not a regular file",
381eda14cbcSMatt Macy direntp->d_name);
382eda14cbcSMatt Macy continue;
383eda14cbcSMatt Macy }
384eda14cbcSMatt Macy if ((st.st_uid != 0) && !zcp->do_force) {
385eda14cbcSMatt Macy zed_log_msg(LOG_NOTICE,
386eda14cbcSMatt Macy "Ignoring \"%s\": not owned by root",
387eda14cbcSMatt Macy direntp->d_name);
388eda14cbcSMatt Macy continue;
389eda14cbcSMatt Macy }
390eda14cbcSMatt Macy if (!(st.st_mode & S_IXUSR)) {
391eda14cbcSMatt Macy zed_log_msg(LOG_INFO,
392eda14cbcSMatt Macy "Ignoring \"%s\": not executable by user",
393eda14cbcSMatt Macy direntp->d_name);
394eda14cbcSMatt Macy continue;
395eda14cbcSMatt Macy }
396eda14cbcSMatt Macy if ((st.st_mode & S_IWGRP) && !zcp->do_force) {
397eda14cbcSMatt Macy zed_log_msg(LOG_NOTICE,
398eda14cbcSMatt Macy "Ignoring \"%s\": writable by group",
399eda14cbcSMatt Macy direntp->d_name);
400eda14cbcSMatt Macy continue;
401eda14cbcSMatt Macy }
402eda14cbcSMatt Macy if ((st.st_mode & S_IWOTH) && !zcp->do_force) {
403eda14cbcSMatt Macy zed_log_msg(LOG_NOTICE,
404eda14cbcSMatt Macy "Ignoring \"%s\": writable by other",
405eda14cbcSMatt Macy direntp->d_name);
406eda14cbcSMatt Macy continue;
407eda14cbcSMatt Macy }
408eda14cbcSMatt Macy if (zed_strings_add(zedlets, NULL, direntp->d_name) < 0) {
409eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
410eda14cbcSMatt Macy "Failed to register \"%s\": %s",
411eda14cbcSMatt Macy direntp->d_name, strerror(errno));
412eda14cbcSMatt Macy continue;
413eda14cbcSMatt Macy }
414eda14cbcSMatt Macy if (zcp->do_verbose)
415eda14cbcSMatt Macy zed_log_msg(LOG_INFO,
416eda14cbcSMatt Macy "Registered zedlet \"%s\"", direntp->d_name);
417eda14cbcSMatt Macy }
418eda14cbcSMatt Macy if (closedir(dirp) < 0) {
419eda14cbcSMatt Macy int errno_bak = errno;
420eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to close dir \"%s\": %s",
421eda14cbcSMatt Macy zcp->zedlet_dir, strerror(errno));
422eda14cbcSMatt Macy zed_strings_destroy(zedlets);
423eda14cbcSMatt Macy errno = errno_bak;
424eda14cbcSMatt Macy return (-1);
425eda14cbcSMatt Macy }
426eda14cbcSMatt Macy if (zcp->zedlets)
427eda14cbcSMatt Macy zed_strings_destroy(zcp->zedlets);
428eda14cbcSMatt Macy
429eda14cbcSMatt Macy zcp->zedlets = zedlets;
430eda14cbcSMatt Macy return (0);
431eda14cbcSMatt Macy }
432eda14cbcSMatt Macy
433eda14cbcSMatt Macy /*
434eda14cbcSMatt Macy * Write the PID file specified in [zcp].
435eda14cbcSMatt Macy * Return 0 on success, -1 on error.
436eda14cbcSMatt Macy *
437eda14cbcSMatt Macy * This must be called after fork()ing to become a daemon (so the correct PID
438eda14cbcSMatt Macy * is recorded), but before daemonization is complete and the parent process
439eda14cbcSMatt Macy * exits (for synchronization with systemd).
440eda14cbcSMatt Macy */
441eda14cbcSMatt Macy int
zed_conf_write_pid(struct zed_conf * zcp)442eda14cbcSMatt Macy zed_conf_write_pid(struct zed_conf *zcp)
443eda14cbcSMatt Macy {
444eda14cbcSMatt Macy char buf[PATH_MAX];
445eda14cbcSMatt Macy int n;
446eda14cbcSMatt Macy char *p;
447eda14cbcSMatt Macy mode_t mask;
448eda14cbcSMatt Macy int rv;
449eda14cbcSMatt Macy
450eda14cbcSMatt Macy if (!zcp || !zcp->pid_file) {
451eda14cbcSMatt Macy errno = EINVAL;
452eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to create PID file: %s",
453eda14cbcSMatt Macy strerror(errno));
454eda14cbcSMatt Macy return (-1);
455eda14cbcSMatt Macy }
456eda14cbcSMatt Macy assert(zcp->pid_fd == -1);
457eda14cbcSMatt Macy /*
458eda14cbcSMatt Macy * Create PID file directory if needed.
459eda14cbcSMatt Macy */
460eda14cbcSMatt Macy n = strlcpy(buf, zcp->pid_file, sizeof (buf));
461eda14cbcSMatt Macy if (n >= sizeof (buf)) {
462eda14cbcSMatt Macy errno = ENAMETOOLONG;
463eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to create PID file: %s",
464eda14cbcSMatt Macy strerror(errno));
465eda14cbcSMatt Macy goto err;
466eda14cbcSMatt Macy }
467eda14cbcSMatt Macy p = strrchr(buf, '/');
468eda14cbcSMatt Macy if (p)
469eda14cbcSMatt Macy *p = '\0';
470eda14cbcSMatt Macy
47116038816SMartin Matuska if ((mkdirp(buf, 0755) < 0) && (errno != EEXIST)) {
472eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to create directory \"%s\": %s",
473eda14cbcSMatt Macy buf, strerror(errno));
474eda14cbcSMatt Macy goto err;
475eda14cbcSMatt Macy }
476eda14cbcSMatt Macy /*
477eda14cbcSMatt Macy * Obtain PID file lock.
478eda14cbcSMatt Macy */
479eda14cbcSMatt Macy mask = umask(0);
480eda14cbcSMatt Macy umask(mask | 022);
48116038816SMartin Matuska zcp->pid_fd = open(zcp->pid_file, O_RDWR | O_CREAT | O_CLOEXEC, 0644);
482eda14cbcSMatt Macy umask(mask);
483eda14cbcSMatt Macy if (zcp->pid_fd < 0) {
484eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to open PID file \"%s\": %s",
485eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
486eda14cbcSMatt Macy goto err;
487eda14cbcSMatt Macy }
488eda14cbcSMatt Macy rv = zed_file_lock(zcp->pid_fd);
489eda14cbcSMatt Macy if (rv < 0) {
490eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to lock PID file \"%s\": %s",
491eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
492eda14cbcSMatt Macy goto err;
493eda14cbcSMatt Macy } else if (rv > 0) {
494eda14cbcSMatt Macy pid_t pid = zed_file_is_locked(zcp->pid_fd);
495eda14cbcSMatt Macy if (pid < 0) {
496eda14cbcSMatt Macy zed_log_msg(LOG_ERR,
497eda14cbcSMatt Macy "Failed to test lock on PID file \"%s\"",
498eda14cbcSMatt Macy zcp->pid_file);
499eda14cbcSMatt Macy } else if (pid > 0) {
500eda14cbcSMatt Macy zed_log_msg(LOG_ERR,
501eda14cbcSMatt Macy "Found PID %d bound to PID file \"%s\"",
502eda14cbcSMatt Macy pid, zcp->pid_file);
503eda14cbcSMatt Macy } else {
504eda14cbcSMatt Macy zed_log_msg(LOG_ERR,
505eda14cbcSMatt Macy "Inconsistent lock state on PID file \"%s\"",
506eda14cbcSMatt Macy zcp->pid_file);
507eda14cbcSMatt Macy }
508eda14cbcSMatt Macy goto err;
509eda14cbcSMatt Macy }
510eda14cbcSMatt Macy /*
511eda14cbcSMatt Macy * Write PID file.
512eda14cbcSMatt Macy */
513eda14cbcSMatt Macy n = snprintf(buf, sizeof (buf), "%d\n", (int)getpid());
514eda14cbcSMatt Macy if ((n < 0) || (n >= sizeof (buf))) {
515eda14cbcSMatt Macy errno = ERANGE;
516eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to write PID file \"%s\": %s",
517eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
51816038816SMartin Matuska } else if (write(zcp->pid_fd, buf, n) != n) {
519eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to write PID file \"%s\": %s",
520eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
521eda14cbcSMatt Macy } else if (fdatasync(zcp->pid_fd) < 0) {
522eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to sync PID file \"%s\": %s",
523eda14cbcSMatt Macy zcp->pid_file, strerror(errno));
524eda14cbcSMatt Macy } else {
525eda14cbcSMatt Macy return (0);
526eda14cbcSMatt Macy }
527eda14cbcSMatt Macy
528eda14cbcSMatt Macy err:
529eda14cbcSMatt Macy if (zcp->pid_fd >= 0) {
530eda14cbcSMatt Macy (void) close(zcp->pid_fd);
531eda14cbcSMatt Macy zcp->pid_fd = -1;
532eda14cbcSMatt Macy }
533eda14cbcSMatt Macy return (-1);
534eda14cbcSMatt Macy }
535eda14cbcSMatt Macy
536eda14cbcSMatt Macy /*
537eda14cbcSMatt Macy * Open and lock the [zcp] state_file.
538eda14cbcSMatt Macy * Return 0 on success, -1 on error.
539eda14cbcSMatt Macy *
540eda14cbcSMatt Macy * FIXME: Move state information into kernel.
541eda14cbcSMatt Macy */
542eda14cbcSMatt Macy int
zed_conf_open_state(struct zed_conf * zcp)543eda14cbcSMatt Macy zed_conf_open_state(struct zed_conf *zcp)
544eda14cbcSMatt Macy {
545eda14cbcSMatt Macy char dirbuf[PATH_MAX];
546eda14cbcSMatt Macy int n;
547eda14cbcSMatt Macy char *p;
548eda14cbcSMatt Macy int rv;
549eda14cbcSMatt Macy
550eda14cbcSMatt Macy if (!zcp || !zcp->state_file) {
551eda14cbcSMatt Macy errno = EINVAL;
552eda14cbcSMatt Macy zed_log_msg(LOG_ERR, "Failed to open state file: %s",
553eda14cbcSMatt Macy strerror(errno));
554eda14cbcSMatt Macy return (-1);
555eda14cbcSMatt Macy }
556eda14cbcSMatt Macy n = strlcpy(dirbuf, zcp->state_file, sizeof (dirbuf));
557eda14cbcSMatt Macy if (n >= sizeof (dirbuf)) {
558eda14cbcSMatt Macy errno = ENAMETOOLONG;
559eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to open state file: %s",
560eda14cbcSMatt Macy strerror(errno));
561eda14cbcSMatt Macy return (-1);
562eda14cbcSMatt Macy }
563eda14cbcSMatt Macy p = strrchr(dirbuf, '/');
564eda14cbcSMatt Macy if (p)
565eda14cbcSMatt Macy *p = '\0';
566eda14cbcSMatt Macy
56716038816SMartin Matuska if ((mkdirp(dirbuf, 0755) < 0) && (errno != EEXIST)) {
568eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
569eda14cbcSMatt Macy "Failed to create directory \"%s\": %s",
570eda14cbcSMatt Macy dirbuf, strerror(errno));
571eda14cbcSMatt Macy return (-1);
572eda14cbcSMatt Macy }
573eda14cbcSMatt Macy if (zcp->state_fd >= 0) {
574eda14cbcSMatt Macy if (close(zcp->state_fd) < 0) {
575eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
576eda14cbcSMatt Macy "Failed to close state file \"%s\": %s",
577eda14cbcSMatt Macy zcp->state_file, strerror(errno));
578eda14cbcSMatt Macy return (-1);
579eda14cbcSMatt Macy }
580eda14cbcSMatt Macy }
581eda14cbcSMatt Macy if (zcp->do_zero)
582eda14cbcSMatt Macy (void) unlink(zcp->state_file);
583eda14cbcSMatt Macy
584eda14cbcSMatt Macy zcp->state_fd = open(zcp->state_file,
58516038816SMartin Matuska O_RDWR | O_CREAT | O_CLOEXEC, 0644);
586eda14cbcSMatt Macy if (zcp->state_fd < 0) {
587eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to open state file \"%s\": %s",
588eda14cbcSMatt Macy zcp->state_file, strerror(errno));
589eda14cbcSMatt Macy return (-1);
590eda14cbcSMatt Macy }
591eda14cbcSMatt Macy rv = zed_file_lock(zcp->state_fd);
592eda14cbcSMatt Macy if (rv < 0) {
593eda14cbcSMatt Macy zed_log_msg(LOG_WARNING, "Failed to lock state file \"%s\": %s",
594eda14cbcSMatt Macy zcp->state_file, strerror(errno));
595eda14cbcSMatt Macy return (-1);
596eda14cbcSMatt Macy }
597eda14cbcSMatt Macy if (rv > 0) {
598eda14cbcSMatt Macy pid_t pid = zed_file_is_locked(zcp->state_fd);
599eda14cbcSMatt Macy if (pid < 0) {
600eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
601eda14cbcSMatt Macy "Failed to test lock on state file \"%s\"",
602eda14cbcSMatt Macy zcp->state_file);
603eda14cbcSMatt Macy } else if (pid > 0) {
604eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
605eda14cbcSMatt Macy "Found PID %d bound to state file \"%s\"",
606eda14cbcSMatt Macy pid, zcp->state_file);
607eda14cbcSMatt Macy } else {
608eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
609eda14cbcSMatt Macy "Inconsistent lock state on state file \"%s\"",
610eda14cbcSMatt Macy zcp->state_file);
611eda14cbcSMatt Macy }
612eda14cbcSMatt Macy return (-1);
613eda14cbcSMatt Macy }
614eda14cbcSMatt Macy return (0);
615eda14cbcSMatt Macy }
616eda14cbcSMatt Macy
617eda14cbcSMatt Macy /*
618eda14cbcSMatt Macy * Read the opened [zcp] state_file to obtain the eid & etime of the last event
619eda14cbcSMatt Macy * processed. Write the state from the last event to the [eidp] & [etime] args
620eda14cbcSMatt Macy * passed by reference. Note that etime[] is an array of size 2.
621eda14cbcSMatt Macy * Return 0 on success, -1 on error.
622eda14cbcSMatt Macy */
623eda14cbcSMatt Macy int
zed_conf_read_state(struct zed_conf * zcp,uint64_t * eidp,int64_t etime[])624eda14cbcSMatt Macy zed_conf_read_state(struct zed_conf *zcp, uint64_t *eidp, int64_t etime[])
625eda14cbcSMatt Macy {
626eda14cbcSMatt Macy ssize_t len;
627eda14cbcSMatt Macy struct iovec iov[3];
628eda14cbcSMatt Macy ssize_t n;
629eda14cbcSMatt Macy
630eda14cbcSMatt Macy if (!zcp || !eidp || !etime) {
631eda14cbcSMatt Macy errno = EINVAL;
632eda14cbcSMatt Macy zed_log_msg(LOG_ERR,
633eda14cbcSMatt Macy "Failed to read state file: %s", strerror(errno));
634eda14cbcSMatt Macy return (-1);
635eda14cbcSMatt Macy }
636eda14cbcSMatt Macy if (lseek(zcp->state_fd, 0, SEEK_SET) == (off_t)-1) {
637eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
638eda14cbcSMatt Macy "Failed to reposition state file offset: %s",
639eda14cbcSMatt Macy strerror(errno));
640eda14cbcSMatt Macy return (-1);
641eda14cbcSMatt Macy }
642eda14cbcSMatt Macy len = 0;
643eda14cbcSMatt Macy iov[0].iov_base = eidp;
644eda14cbcSMatt Macy len += iov[0].iov_len = sizeof (*eidp);
645eda14cbcSMatt Macy iov[1].iov_base = &etime[0];
646eda14cbcSMatt Macy len += iov[1].iov_len = sizeof (etime[0]);
647eda14cbcSMatt Macy iov[2].iov_base = &etime[1];
648eda14cbcSMatt Macy len += iov[2].iov_len = sizeof (etime[1]);
649eda14cbcSMatt Macy
650eda14cbcSMatt Macy n = readv(zcp->state_fd, iov, 3);
651eda14cbcSMatt Macy if (n == 0) {
652eda14cbcSMatt Macy *eidp = 0;
653eda14cbcSMatt Macy } else if (n < 0) {
654eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
655eda14cbcSMatt Macy "Failed to read state file \"%s\": %s",
656eda14cbcSMatt Macy zcp->state_file, strerror(errno));
657eda14cbcSMatt Macy return (-1);
658eda14cbcSMatt Macy } else if (n != len) {
659eda14cbcSMatt Macy errno = EIO;
660eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
661c7046f76SMartin Matuska "Failed to read state file \"%s\": Read %zd of %zd bytes",
662eda14cbcSMatt Macy zcp->state_file, n, len);
663eda14cbcSMatt Macy return (-1);
664eda14cbcSMatt Macy }
665eda14cbcSMatt Macy return (0);
666eda14cbcSMatt Macy }
667eda14cbcSMatt Macy
668eda14cbcSMatt Macy /*
669eda14cbcSMatt Macy * Write the [eid] & [etime] of the last processed event to the opened
670eda14cbcSMatt Macy * [zcp] state_file. Note that etime[] is an array of size 2.
671eda14cbcSMatt Macy * Return 0 on success, -1 on error.
672eda14cbcSMatt Macy */
673eda14cbcSMatt Macy int
zed_conf_write_state(struct zed_conf * zcp,uint64_t eid,int64_t etime[])674eda14cbcSMatt Macy zed_conf_write_state(struct zed_conf *zcp, uint64_t eid, int64_t etime[])
675eda14cbcSMatt Macy {
676eda14cbcSMatt Macy ssize_t len;
677eda14cbcSMatt Macy struct iovec iov[3];
678eda14cbcSMatt Macy ssize_t n;
679eda14cbcSMatt Macy
680eda14cbcSMatt Macy if (!zcp) {
681eda14cbcSMatt Macy errno = EINVAL;
682eda14cbcSMatt Macy zed_log_msg(LOG_ERR,
683eda14cbcSMatt Macy "Failed to write state file: %s", strerror(errno));
684eda14cbcSMatt Macy return (-1);
685eda14cbcSMatt Macy }
686eda14cbcSMatt Macy if (lseek(zcp->state_fd, 0, SEEK_SET) == (off_t)-1) {
687eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
688eda14cbcSMatt Macy "Failed to reposition state file offset: %s",
689eda14cbcSMatt Macy strerror(errno));
690eda14cbcSMatt Macy return (-1);
691eda14cbcSMatt Macy }
692eda14cbcSMatt Macy len = 0;
693eda14cbcSMatt Macy iov[0].iov_base = &eid;
694eda14cbcSMatt Macy len += iov[0].iov_len = sizeof (eid);
695eda14cbcSMatt Macy iov[1].iov_base = &etime[0];
696eda14cbcSMatt Macy len += iov[1].iov_len = sizeof (etime[0]);
697eda14cbcSMatt Macy iov[2].iov_base = &etime[1];
698eda14cbcSMatt Macy len += iov[2].iov_len = sizeof (etime[1]);
699eda14cbcSMatt Macy
700eda14cbcSMatt Macy n = writev(zcp->state_fd, iov, 3);
701eda14cbcSMatt Macy if (n < 0) {
702eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
703eda14cbcSMatt Macy "Failed to write state file \"%s\": %s",
704eda14cbcSMatt Macy zcp->state_file, strerror(errno));
705eda14cbcSMatt Macy return (-1);
706eda14cbcSMatt Macy }
707eda14cbcSMatt Macy if (n != len) {
708eda14cbcSMatt Macy errno = EIO;
709eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
710c7046f76SMartin Matuska "Failed to write state file \"%s\": Wrote %zd of %zd bytes",
711eda14cbcSMatt Macy zcp->state_file, n, len);
712eda14cbcSMatt Macy return (-1);
713eda14cbcSMatt Macy }
714eda14cbcSMatt Macy if (fdatasync(zcp->state_fd) < 0) {
715eda14cbcSMatt Macy zed_log_msg(LOG_WARNING,
716eda14cbcSMatt Macy "Failed to sync state file \"%s\": %s",
717eda14cbcSMatt Macy zcp->state_file, strerror(errno));
718eda14cbcSMatt Macy return (-1);
719eda14cbcSMatt Macy }
720eda14cbcSMatt Macy return (0);
721eda14cbcSMatt Macy }
722