mirror of
				https://git.proxmox.com/git/mirror_zfs
				synced 2025-10-26 05:28:04 +00:00 
			
		
		
		
	 f272960d52
			
		
	
	
		f272960d52
		
			
		
	
	
	
	
		
			
			Coverity complained about the format specifiers not matching variables. In one case, the variable is a constant, so we fix it. In another, we were missing an argument (about which coverity also complained). Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Reviewed-by: Tony Hutter <hutter2@llnl.gov> Signed-off-by: Richard Yao <richard.yao@alumni.stonybrook.edu> Closes #13888
		
			
				
	
	
		
			721 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			721 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * This file is part of the ZFS Event Daemon (ZED).
 | |
|  *
 | |
|  * Developed at Lawrence Livermore National Laboratory (LLNL-CODE-403049).
 | |
|  * Copyright (C) 2013-2014 Lawrence Livermore National Security, LLC.
 | |
|  * Refer to the OpenZFS git commit log for authoritative copyright attribution.
 | |
|  *
 | |
|  * The contents of this file are subject to the terms of the
 | |
|  * Common Development and Distribution License Version 1.0 (CDDL-1.0).
 | |
|  * You can obtain a copy of the license from the top-level file
 | |
|  * "OPENSOLARIS.LICENSE" or at <http://opensource.org/licenses/CDDL-1.0>.
 | |
|  * You may not use this file except in compliance with the license.
 | |
|  */
 | |
| 
 | |
| #include <assert.h>
 | |
| #include <ctype.h>
 | |
| #include <dirent.h>
 | |
| #include <errno.h>
 | |
| #include <fcntl.h>
 | |
| #include <libgen.h>
 | |
| #include <limits.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/uio.h>
 | |
| #include <unistd.h>
 | |
| #include "zed.h"
 | |
| #include "zed_conf.h"
 | |
| #include "zed_file.h"
 | |
| #include "zed_log.h"
 | |
| #include "zed_strings.h"
 | |
| 
 | |
| /*
 | |
|  * Initialise the configuration with default values.
 | |
|  */
 | |
| void
 | |
| zed_conf_init(struct zed_conf *zcp)
 | |
| {
 | |
| 	memset(zcp, 0, sizeof (*zcp));
 | |
| 
 | |
| 	/* zcp->zfs_hdl opened in zed_event_init() */
 | |
| 	/* zcp->zedlets created in zed_conf_scan_dir() */
 | |
| 
 | |
| 	zcp->pid_fd = -1;		/* opened in zed_conf_write_pid() */
 | |
| 	zcp->state_fd = -1;		/* opened in zed_conf_open_state() */
 | |
| 	zcp->zevent_fd = -1;		/* opened in zed_event_init() */
 | |
| 
 | |
| 	zcp->max_jobs = 16;
 | |
| 	zcp->max_zevent_buf_len = 1 << 20;
 | |
| 
 | |
| 	if (!(zcp->pid_file = strdup(ZED_PID_FILE)) ||
 | |
| 	    !(zcp->zedlet_dir = strdup(ZED_ZEDLET_DIR)) ||
 | |
| 	    !(zcp->state_file = strdup(ZED_STATE_FILE)))
 | |
| 		zed_log_die("Failed to create conf: %s", strerror(errno));
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Destroy the configuration [zcp].
 | |
|  *
 | |
|  * Note: zfs_hdl & zevent_fd are destroyed via zed_event_fini().
 | |
|  */
 | |
| void
 | |
| zed_conf_destroy(struct zed_conf *zcp)
 | |
| {
 | |
| 	if (zcp->state_fd >= 0) {
 | |
| 		if (close(zcp->state_fd) < 0)
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to close state file \"%s\": %s",
 | |
| 			    zcp->state_file, strerror(errno));
 | |
| 		zcp->state_fd = -1;
 | |
| 	}
 | |
| 	if (zcp->pid_file) {
 | |
| 		if ((unlink(zcp->pid_file) < 0) && (errno != ENOENT))
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to remove PID file \"%s\": %s",
 | |
| 			    zcp->pid_file, strerror(errno));
 | |
| 	}
 | |
| 	if (zcp->pid_fd >= 0) {
 | |
| 		if (close(zcp->pid_fd) < 0)
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to close PID file \"%s\": %s",
 | |
| 			    zcp->pid_file, strerror(errno));
 | |
| 		zcp->pid_fd = -1;
 | |
| 	}
 | |
| 	if (zcp->pid_file) {
 | |
| 		free(zcp->pid_file);
 | |
| 		zcp->pid_file = NULL;
 | |
| 	}
 | |
| 	if (zcp->zedlet_dir) {
 | |
| 		free(zcp->zedlet_dir);
 | |
| 		zcp->zedlet_dir = NULL;
 | |
| 	}
 | |
| 	if (zcp->state_file) {
 | |
| 		free(zcp->state_file);
 | |
| 		zcp->state_file = NULL;
 | |
| 	}
 | |
| 	if (zcp->zedlets) {
 | |
| 		zed_strings_destroy(zcp->zedlets);
 | |
| 		zcp->zedlets = NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Display command-line help and exit.
 | |
|  *
 | |
|  * If [got_err] is 0, output to stdout and exit normally;
 | |
|  * otherwise, output to stderr and exit with a failure status.
 | |
|  */
 | |
| static void
 | |
| _zed_conf_display_help(const char *prog, boolean_t got_err)
 | |
| {
 | |
| 	struct opt { const char *o, *d, *v; };
 | |
| 
 | |
| 	FILE *fp = got_err ? stderr : stdout;
 | |
| 
 | |
| 	struct opt *oo;
 | |
| 	struct opt iopts[] = {
 | |
| 		{ .o = "-h", .d = "Display help" },
 | |
| 		{ .o = "-L", .d = "Display license information" },
 | |
| 		{ .o = "-V", .d = "Display version information" },
 | |
| 		{},
 | |
| 	};
 | |
| 	struct opt nopts[] = {
 | |
| 		{ .o = "-v", .d = "Be verbose" },
 | |
| 		{ .o = "-f", .d = "Force daemon to run" },
 | |
| 		{ .o = "-F", .d = "Run daemon in the foreground" },
 | |
| 		{ .o = "-I",
 | |
| 		    .d = "Idle daemon until kernel module is (re)loaded" },
 | |
| 		{ .o = "-M", .d = "Lock all pages in memory" },
 | |
| 		{ .o = "-P", .d = "$PATH for ZED to use (only used by ZTS)" },
 | |
| 		{ .o = "-Z", .d = "Zero state file" },
 | |
| 		{},
 | |
| 	};
 | |
| 	struct opt vopts[] = {
 | |
| 		{ .o = "-d DIR", .d = "Read enabled ZEDLETs from DIR.",
 | |
| 		    .v = ZED_ZEDLET_DIR },
 | |
| 		{ .o = "-p FILE", .d = "Write daemon's PID to FILE.",
 | |
| 		    .v = ZED_PID_FILE },
 | |
| 		{ .o = "-s FILE", .d = "Write daemon's state to FILE.",
 | |
| 		    .v = ZED_STATE_FILE },
 | |
| 		{ .o = "-j JOBS", .d = "Start at most JOBS at once.",
 | |
| 		    .v = "16" },
 | |
| 		{ .o = "-b LEN", .d = "Cap kernel event buffer at LEN entries.",
 | |
| 		    .v = "1048576" },
 | |
| 		{},
 | |
| 	};
 | |
| 
 | |
| 	fprintf(fp, "Usage: %s [OPTION]...\n", (prog ? prog : "zed"));
 | |
| 	fprintf(fp, "\n");
 | |
| 	for (oo = iopts; oo->o; ++oo)
 | |
| 		fprintf(fp, "    %*s %s\n", -8, oo->o, oo->d);
 | |
| 	fprintf(fp, "\n");
 | |
| 	for (oo = nopts; oo->o; ++oo)
 | |
| 		fprintf(fp, "    %*s %s\n", -8, oo->o, oo->d);
 | |
| 	fprintf(fp, "\n");
 | |
| 	for (oo = vopts; oo->o; ++oo)
 | |
| 		fprintf(fp, "    %*s %s [%s]\n", -8, oo->o, oo->d, oo->v);
 | |
| 	fprintf(fp, "\n");
 | |
| 
 | |
| 	exit(got_err ? EXIT_FAILURE : EXIT_SUCCESS);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Display license information to stdout and exit.
 | |
|  */
 | |
| static void
 | |
| _zed_conf_display_license(void)
 | |
| {
 | |
| 	printf(
 | |
| 	    "The ZFS Event Daemon (ZED) is distributed under the terms of the\n"
 | |
| 	    "  Common Development and Distribution License (CDDL-1.0)\n"
 | |
| 	    "  <http://opensource.org/licenses/CDDL-1.0>.\n"
 | |
| 	    "\n"
 | |
| 	    "Developed at Lawrence Livermore National Laboratory"
 | |
| 	    " (LLNL-CODE-403049).\n"
 | |
| 	    "\n");
 | |
| 
 | |
| 	exit(EXIT_SUCCESS);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Display version information to stdout and exit.
 | |
|  */
 | |
| static void
 | |
| _zed_conf_display_version(void)
 | |
| {
 | |
| 	printf("%s-%s-%s\n",
 | |
| 	    ZFS_META_NAME, ZFS_META_VERSION, ZFS_META_RELEASE);
 | |
| 
 | |
| 	exit(EXIT_SUCCESS);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Copy the [path] string to the [resultp] ptr.
 | |
|  * If [path] is not an absolute path, prefix it with the current working dir.
 | |
|  * If [resultp] is non-null, free its existing string before assignment.
 | |
|  */
 | |
| static void
 | |
| _zed_conf_parse_path(char **resultp, const char *path)
 | |
| {
 | |
| 	char buf[PATH_MAX];
 | |
| 
 | |
| 	assert(resultp != NULL);
 | |
| 	assert(path != NULL);
 | |
| 
 | |
| 	if (*resultp)
 | |
| 		free(*resultp);
 | |
| 
 | |
| 	if (path[0] == '/') {
 | |
| 		*resultp = strdup(path);
 | |
| 	} else {
 | |
| 		if (!getcwd(buf, sizeof (buf)))
 | |
| 			zed_log_die("Failed to get current working dir: %s",
 | |
| 			    strerror(errno));
 | |
| 
 | |
| 		if (strlcat(buf, "/", sizeof (buf)) >= sizeof (buf) ||
 | |
| 		    strlcat(buf, path, sizeof (buf)) >= sizeof (buf))
 | |
| 			zed_log_die("Failed to copy path: %s",
 | |
| 			    strerror(ENAMETOOLONG));
 | |
| 
 | |
| 		*resultp = strdup(buf);
 | |
| 	}
 | |
| 
 | |
| 	if (!*resultp)
 | |
| 		zed_log_die("Failed to copy path: %s", strerror(ENOMEM));
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Parse the command-line options into the configuration [zcp].
 | |
|  */
 | |
| void
 | |
| zed_conf_parse_opts(struct zed_conf *zcp, int argc, char **argv)
 | |
| {
 | |
| 	const char * const opts = ":hLVd:p:P:s:vfFMZIj:b:";
 | |
| 	int opt;
 | |
| 	unsigned long raw;
 | |
| 
 | |
| 	if (!zcp || !argv || !argv[0])
 | |
| 		zed_log_die("Failed to parse options: Internal error");
 | |
| 
 | |
| 	opterr = 0;			/* suppress default getopt err msgs */
 | |
| 
 | |
| 	while ((opt = getopt(argc, argv, opts)) != -1) {
 | |
| 		switch (opt) {
 | |
| 		case 'h':
 | |
| 			_zed_conf_display_help(argv[0], B_FALSE);
 | |
| 			break;
 | |
| 		case 'L':
 | |
| 			_zed_conf_display_license();
 | |
| 			break;
 | |
| 		case 'V':
 | |
| 			_zed_conf_display_version();
 | |
| 			break;
 | |
| 		case 'd':
 | |
| 			_zed_conf_parse_path(&zcp->zedlet_dir, optarg);
 | |
| 			break;
 | |
| 		case 'I':
 | |
| 			zcp->do_idle = 1;
 | |
| 			break;
 | |
| 		case 'p':
 | |
| 			_zed_conf_parse_path(&zcp->pid_file, optarg);
 | |
| 			break;
 | |
| 		case 'P':
 | |
| 			_zed_conf_parse_path(&zcp->path, optarg);
 | |
| 			break;
 | |
| 		case 's':
 | |
| 			_zed_conf_parse_path(&zcp->state_file, optarg);
 | |
| 			break;
 | |
| 		case 'v':
 | |
| 			zcp->do_verbose = 1;
 | |
| 			break;
 | |
| 		case 'f':
 | |
| 			zcp->do_force = 1;
 | |
| 			break;
 | |
| 		case 'F':
 | |
| 			zcp->do_foreground = 1;
 | |
| 			break;
 | |
| 		case 'M':
 | |
| 			zcp->do_memlock = 1;
 | |
| 			break;
 | |
| 		case 'Z':
 | |
| 			zcp->do_zero = 1;
 | |
| 			break;
 | |
| 		case 'j':
 | |
| 			errno = 0;
 | |
| 			raw = strtoul(optarg, NULL, 0);
 | |
| 			if (errno == ERANGE || raw > INT16_MAX) {
 | |
| 				zed_log_die("%lu is too many jobs", raw);
 | |
| 			} if (raw == 0) {
 | |
| 				zed_log_die("0 jobs makes no sense");
 | |
| 			} else {
 | |
| 				zcp->max_jobs = raw;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'b':
 | |
| 			errno = 0;
 | |
| 			raw = strtoul(optarg, NULL, 0);
 | |
| 			if (errno == ERANGE || raw > INT32_MAX) {
 | |
| 				zed_log_die("%lu is too large", raw);
 | |
| 			} if (raw == 0) {
 | |
| 				zcp->max_zevent_buf_len = INT32_MAX;
 | |
| 			} else {
 | |
| 				zcp->max_zevent_buf_len = raw;
 | |
| 			}
 | |
| 			break;
 | |
| 		case '?':
 | |
| 		default:
 | |
| 			if (optopt == '?')
 | |
| 				_zed_conf_display_help(argv[0], B_FALSE);
 | |
| 
 | |
| 			fprintf(stderr, "%s: Invalid option '-%c'\n\n",
 | |
| 			    argv[0], optopt);
 | |
| 			_zed_conf_display_help(argv[0], B_TRUE);
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Scan the [zcp] zedlet_dir for files to exec based on the event class.
 | |
|  * Files must be executable by user, but not writable by group or other.
 | |
|  * Dotfiles are ignored.
 | |
|  *
 | |
|  * Return 0 on success with an updated set of zedlets,
 | |
|  * or -1 on error with errno set.
 | |
|  */
 | |
| int
 | |
| zed_conf_scan_dir(struct zed_conf *zcp)
 | |
| {
 | |
| 	zed_strings_t *zedlets;
 | |
| 	DIR *dirp;
 | |
| 	struct dirent *direntp;
 | |
| 	char pathname[PATH_MAX];
 | |
| 	struct stat st;
 | |
| 	int n;
 | |
| 
 | |
| 	if (!zcp) {
 | |
| 		errno = EINVAL;
 | |
| 		zed_log_msg(LOG_ERR, "Failed to scan zedlet dir: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	zedlets = zed_strings_create();
 | |
| 	if (!zedlets) {
 | |
| 		errno = ENOMEM;
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to scan dir \"%s\": %s",
 | |
| 		    zcp->zedlet_dir, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	dirp = opendir(zcp->zedlet_dir);
 | |
| 	if (!dirp) {
 | |
| 		int errno_bak = errno;
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to open dir \"%s\": %s",
 | |
| 		    zcp->zedlet_dir, strerror(errno));
 | |
| 		zed_strings_destroy(zedlets);
 | |
| 		errno = errno_bak;
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	while ((direntp = readdir(dirp))) {
 | |
| 		if (direntp->d_name[0] == '.')
 | |
| 			continue;
 | |
| 
 | |
| 		n = snprintf(pathname, sizeof (pathname),
 | |
| 		    "%s/%s", zcp->zedlet_dir, direntp->d_name);
 | |
| 		if ((n < 0) || (n >= sizeof (pathname))) {
 | |
| 			zed_log_msg(LOG_WARNING, "Failed to stat \"%s\": %s",
 | |
| 			    direntp->d_name, strerror(ENAMETOOLONG));
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (stat(pathname, &st) < 0) {
 | |
| 			zed_log_msg(LOG_WARNING, "Failed to stat \"%s\": %s",
 | |
| 			    pathname, strerror(errno));
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (!S_ISREG(st.st_mode)) {
 | |
| 			zed_log_msg(LOG_INFO,
 | |
| 			    "Ignoring \"%s\": not a regular file",
 | |
| 			    direntp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if ((st.st_uid != 0) && !zcp->do_force) {
 | |
| 			zed_log_msg(LOG_NOTICE,
 | |
| 			    "Ignoring \"%s\": not owned by root",
 | |
| 			    direntp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (!(st.st_mode & S_IXUSR)) {
 | |
| 			zed_log_msg(LOG_INFO,
 | |
| 			    "Ignoring \"%s\": not executable by user",
 | |
| 			    direntp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if ((st.st_mode & S_IWGRP) && !zcp->do_force) {
 | |
| 			zed_log_msg(LOG_NOTICE,
 | |
| 			    "Ignoring \"%s\": writable by group",
 | |
| 			    direntp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if ((st.st_mode & S_IWOTH) && !zcp->do_force) {
 | |
| 			zed_log_msg(LOG_NOTICE,
 | |
| 			    "Ignoring \"%s\": writable by other",
 | |
| 			    direntp->d_name);
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (zed_strings_add(zedlets, NULL, direntp->d_name) < 0) {
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to register \"%s\": %s",
 | |
| 			    direntp->d_name, strerror(errno));
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (zcp->do_verbose)
 | |
| 			zed_log_msg(LOG_INFO,
 | |
| 			    "Registered zedlet \"%s\"", direntp->d_name);
 | |
| 	}
 | |
| 	if (closedir(dirp) < 0) {
 | |
| 		int errno_bak = errno;
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to close dir \"%s\": %s",
 | |
| 		    zcp->zedlet_dir, strerror(errno));
 | |
| 		zed_strings_destroy(zedlets);
 | |
| 		errno = errno_bak;
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (zcp->zedlets)
 | |
| 		zed_strings_destroy(zcp->zedlets);
 | |
| 
 | |
| 	zcp->zedlets = zedlets;
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Write the PID file specified in [zcp].
 | |
|  * Return 0 on success, -1 on error.
 | |
|  *
 | |
|  * This must be called after fork()ing to become a daemon (so the correct PID
 | |
|  * is recorded), but before daemonization is complete and the parent process
 | |
|  * exits (for synchronization with systemd).
 | |
|  */
 | |
| int
 | |
| zed_conf_write_pid(struct zed_conf *zcp)
 | |
| {
 | |
| 	char buf[PATH_MAX];
 | |
| 	int n;
 | |
| 	char *p;
 | |
| 	mode_t mask;
 | |
| 	int rv;
 | |
| 
 | |
| 	if (!zcp || !zcp->pid_file) {
 | |
| 		errno = EINVAL;
 | |
| 		zed_log_msg(LOG_ERR, "Failed to create PID file: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	assert(zcp->pid_fd == -1);
 | |
| 	/*
 | |
| 	 * Create PID file directory if needed.
 | |
| 	 */
 | |
| 	n = strlcpy(buf, zcp->pid_file, sizeof (buf));
 | |
| 	if (n >= sizeof (buf)) {
 | |
| 		errno = ENAMETOOLONG;
 | |
| 		zed_log_msg(LOG_ERR, "Failed to create PID file: %s",
 | |
| 		    strerror(errno));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	p = strrchr(buf, '/');
 | |
| 	if (p)
 | |
| 		*p = '\0';
 | |
| 
 | |
| 	if ((mkdirp(buf, 0755) < 0) && (errno != EEXIST)) {
 | |
| 		zed_log_msg(LOG_ERR, "Failed to create directory \"%s\": %s",
 | |
| 		    buf, strerror(errno));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	/*
 | |
| 	 * Obtain PID file lock.
 | |
| 	 */
 | |
| 	mask = umask(0);
 | |
| 	umask(mask | 022);
 | |
| 	zcp->pid_fd = open(zcp->pid_file, O_RDWR | O_CREAT | O_CLOEXEC, 0644);
 | |
| 	umask(mask);
 | |
| 	if (zcp->pid_fd < 0) {
 | |
| 		zed_log_msg(LOG_ERR, "Failed to open PID file \"%s\": %s",
 | |
| 		    zcp->pid_file, strerror(errno));
 | |
| 		goto err;
 | |
| 	}
 | |
| 	rv = zed_file_lock(zcp->pid_fd);
 | |
| 	if (rv < 0) {
 | |
| 		zed_log_msg(LOG_ERR, "Failed to lock PID file \"%s\": %s",
 | |
| 		    zcp->pid_file, strerror(errno));
 | |
| 		goto err;
 | |
| 	} else if (rv > 0) {
 | |
| 		pid_t pid = zed_file_is_locked(zcp->pid_fd);
 | |
| 		if (pid < 0) {
 | |
| 			zed_log_msg(LOG_ERR,
 | |
| 			    "Failed to test lock on PID file \"%s\"",
 | |
| 			    zcp->pid_file);
 | |
| 		} else if (pid > 0) {
 | |
| 			zed_log_msg(LOG_ERR,
 | |
| 			    "Found PID %d bound to PID file \"%s\"",
 | |
| 			    pid, zcp->pid_file);
 | |
| 		} else {
 | |
| 			zed_log_msg(LOG_ERR,
 | |
| 			    "Inconsistent lock state on PID file \"%s\"",
 | |
| 			    zcp->pid_file);
 | |
| 		}
 | |
| 		goto err;
 | |
| 	}
 | |
| 	/*
 | |
| 	 * Write PID file.
 | |
| 	 */
 | |
| 	n = snprintf(buf, sizeof (buf), "%d\n", (int)getpid());
 | |
| 	if ((n < 0) || (n >= sizeof (buf))) {
 | |
| 		errno = ERANGE;
 | |
| 		zed_log_msg(LOG_ERR, "Failed to write PID file \"%s\": %s",
 | |
| 		    zcp->pid_file, strerror(errno));
 | |
| 	} else if (write(zcp->pid_fd, buf, n) != n) {
 | |
| 		zed_log_msg(LOG_ERR, "Failed to write PID file \"%s\": %s",
 | |
| 		    zcp->pid_file, strerror(errno));
 | |
| 	} else if (fdatasync(zcp->pid_fd) < 0) {
 | |
| 		zed_log_msg(LOG_ERR, "Failed to sync PID file \"%s\": %s",
 | |
| 		    zcp->pid_file, strerror(errno));
 | |
| 	} else {
 | |
| 		return (0);
 | |
| 	}
 | |
| 
 | |
| err:
 | |
| 	if (zcp->pid_fd >= 0) {
 | |
| 		(void) close(zcp->pid_fd);
 | |
| 		zcp->pid_fd = -1;
 | |
| 	}
 | |
| 	return (-1);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Open and lock the [zcp] state_file.
 | |
|  * Return 0 on success, -1 on error.
 | |
|  *
 | |
|  * FIXME: Move state information into kernel.
 | |
|  */
 | |
| int
 | |
| zed_conf_open_state(struct zed_conf *zcp)
 | |
| {
 | |
| 	char dirbuf[PATH_MAX];
 | |
| 	int n;
 | |
| 	char *p;
 | |
| 	int rv;
 | |
| 
 | |
| 	if (!zcp || !zcp->state_file) {
 | |
| 		errno = EINVAL;
 | |
| 		zed_log_msg(LOG_ERR, "Failed to open state file: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	n = strlcpy(dirbuf, zcp->state_file, sizeof (dirbuf));
 | |
| 	if (n >= sizeof (dirbuf)) {
 | |
| 		errno = ENAMETOOLONG;
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to open state file: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	p = strrchr(dirbuf, '/');
 | |
| 	if (p)
 | |
| 		*p = '\0';
 | |
| 
 | |
| 	if ((mkdirp(dirbuf, 0755) < 0) && (errno != EEXIST)) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to create directory \"%s\": %s",
 | |
| 		    dirbuf, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (zcp->state_fd >= 0) {
 | |
| 		if (close(zcp->state_fd) < 0) {
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to close state file \"%s\": %s",
 | |
| 			    zcp->state_file, strerror(errno));
 | |
| 			return (-1);
 | |
| 		}
 | |
| 	}
 | |
| 	if (zcp->do_zero)
 | |
| 		(void) unlink(zcp->state_file);
 | |
| 
 | |
| 	zcp->state_fd = open(zcp->state_file,
 | |
| 	    O_RDWR | O_CREAT | O_CLOEXEC, 0644);
 | |
| 	if (zcp->state_fd < 0) {
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to open state file \"%s\": %s",
 | |
| 		    zcp->state_file, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	rv = zed_file_lock(zcp->state_fd);
 | |
| 	if (rv < 0) {
 | |
| 		zed_log_msg(LOG_WARNING, "Failed to lock state file \"%s\": %s",
 | |
| 		    zcp->state_file, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (rv > 0) {
 | |
| 		pid_t pid = zed_file_is_locked(zcp->state_fd);
 | |
| 		if (pid < 0) {
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Failed to test lock on state file \"%s\"",
 | |
| 			    zcp->state_file);
 | |
| 		} else if (pid > 0) {
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Found PID %d bound to state file \"%s\"",
 | |
| 			    pid, zcp->state_file);
 | |
| 		} else {
 | |
| 			zed_log_msg(LOG_WARNING,
 | |
| 			    "Inconsistent lock state on state file \"%s\"",
 | |
| 			    zcp->state_file);
 | |
| 		}
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Read the opened [zcp] state_file to obtain the eid & etime of the last event
 | |
|  * processed.  Write the state from the last event to the [eidp] & [etime] args
 | |
|  * passed by reference.  Note that etime[] is an array of size 2.
 | |
|  * Return 0 on success, -1 on error.
 | |
|  */
 | |
| int
 | |
| zed_conf_read_state(struct zed_conf *zcp, uint64_t *eidp, int64_t etime[])
 | |
| {
 | |
| 	ssize_t len;
 | |
| 	struct iovec iov[3];
 | |
| 	ssize_t n;
 | |
| 
 | |
| 	if (!zcp || !eidp || !etime) {
 | |
| 		errno = EINVAL;
 | |
| 		zed_log_msg(LOG_ERR,
 | |
| 		    "Failed to read state file: %s", strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (lseek(zcp->state_fd, 0, SEEK_SET) == (off_t)-1) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to reposition state file offset: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	len = 0;
 | |
| 	iov[0].iov_base = eidp;
 | |
| 	len += iov[0].iov_len = sizeof (*eidp);
 | |
| 	iov[1].iov_base = &etime[0];
 | |
| 	len += iov[1].iov_len = sizeof (etime[0]);
 | |
| 	iov[2].iov_base = &etime[1];
 | |
| 	len += iov[2].iov_len = sizeof (etime[1]);
 | |
| 
 | |
| 	n = readv(zcp->state_fd, iov, 3);
 | |
| 	if (n == 0) {
 | |
| 		*eidp = 0;
 | |
| 	} else if (n < 0) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to read state file \"%s\": %s",
 | |
| 		    zcp->state_file, strerror(errno));
 | |
| 		return (-1);
 | |
| 	} else if (n != len) {
 | |
| 		errno = EIO;
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to read state file \"%s\": Read %zd of %zd bytes",
 | |
| 		    zcp->state_file, n, len);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Write the [eid] & [etime] of the last processed event to the opened
 | |
|  * [zcp] state_file.  Note that etime[] is an array of size 2.
 | |
|  * Return 0 on success, -1 on error.
 | |
|  */
 | |
| int
 | |
| zed_conf_write_state(struct zed_conf *zcp, uint64_t eid, int64_t etime[])
 | |
| {
 | |
| 	ssize_t len;
 | |
| 	struct iovec iov[3];
 | |
| 	ssize_t n;
 | |
| 
 | |
| 	if (!zcp) {
 | |
| 		errno = EINVAL;
 | |
| 		zed_log_msg(LOG_ERR,
 | |
| 		    "Failed to write state file: %s", strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (lseek(zcp->state_fd, 0, SEEK_SET) == (off_t)-1) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to reposition state file offset: %s",
 | |
| 		    strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	len = 0;
 | |
| 	iov[0].iov_base = &eid;
 | |
| 	len += iov[0].iov_len = sizeof (eid);
 | |
| 	iov[1].iov_base = &etime[0];
 | |
| 	len += iov[1].iov_len = sizeof (etime[0]);
 | |
| 	iov[2].iov_base = &etime[1];
 | |
| 	len += iov[2].iov_len = sizeof (etime[1]);
 | |
| 
 | |
| 	n = writev(zcp->state_fd, iov, 3);
 | |
| 	if (n < 0) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to write state file \"%s\": %s",
 | |
| 		    zcp->state_file, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (n != len) {
 | |
| 		errno = EIO;
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to write state file \"%s\": Wrote %zd of %zd bytes",
 | |
| 		    zcp->state_file, n, len);
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	if (fdatasync(zcp->state_fd) < 0) {
 | |
| 		zed_log_msg(LOG_WARNING,
 | |
| 		    "Failed to sync state file \"%s\": %s",
 | |
| 		    zcp->state_file, strerror(errno));
 | |
| 		return (-1);
 | |
| 	}
 | |
| 	return (0);
 | |
| }
 |