NetBSD/sbin/devpubd/devpubd.c

248 lines
6.2 KiB
C

/* $NetBSD: devpubd.c,v 1.2 2011/09/16 15:42:56 joerg Exp $ */
/*-
* Copyright (c) 2011 Jared D. McNeill <jmcneill@invisible.ca>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by the NetBSD
* Foundation, Inc. and its contributors.
* 4. Neither the name of The NetBSD Foundation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <sys/cdefs.h>
__COPYRIGHT("@(#) Copyright (c) 2011\
Jared D. McNeill <jmcneill@invisible.ca>. All rights reserved.");
__RCSID("$NetBSD: devpubd.c,v 1.2 2011/09/16 15:42:56 joerg Exp $");
#include <sys/types.h>
#include <sys/ioctl.h>
#include <sys/drvctlio.h>
#include <sys/wait.h>
#include <sys/poll.h>
#include <assert.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include <unistd.h>
static int drvctl_fd = -1;
static const char devpubd_script[] = DEVPUBD_RUN_HOOKS;
#define DEVPUBD_ATTACH_EVENT "device-attach"
#define DEVPUBD_DETACH_EVENT "device-detach"
__dead static void
devpubd_exec(const char *path, const char *event, const char *device)
{
int error;
error = execl(path, path, event, device, NULL);
if (error) {
syslog(LOG_ERR, "couldn't exec '%s %s %s': %m",
path, event, device);
exit(EXIT_FAILURE);
}
exit(EXIT_SUCCESS);
}
static void
devpubd_eventhandler(const char *event, const char *device)
{
pid_t pid;
int status;
syslog(LOG_DEBUG, "event = '%s', device = '%s'", event, device);
pid = fork();
switch (pid) {
case -1:
syslog(LOG_ERR, "fork failed: %m");
break;
case 0:
devpubd_exec(devpubd_script, event, device);
/* NOTREACHED */
default:
if (waitpid(pid, &status, 0) == -1) {
syslog(LOG_ERR, "waitpid(%d) failed: %m", pid);
break;
}
if (WIFEXITED(status) && WEXITSTATUS(status)) {
syslog(LOG_WARNING, "%s exited with status %d",
devpubd_script, WEXITSTATUS(status));
} else if (WIFSIGNALED(status)) {
syslog(LOG_WARNING, "%s received signal %d",
devpubd_script, WTERMSIG(status));
}
break;
}
}
__dead static void
devpubd_eventloop(void)
{
const char *event, *device;
prop_dictionary_t ev;
int res;
assert(drvctl_fd != -1);
for (;;) {
res = prop_dictionary_recv_ioctl(drvctl_fd, DRVGETEVENT, &ev);
if (res)
err(EXIT_FAILURE, "DRVGETEVENT failed");
prop_dictionary_get_cstring_nocopy(ev, "event", &event);
prop_dictionary_get_cstring_nocopy(ev, "device", &device);
printf("%s: event='%s', device='%s'\n", __func__,
event, device);
devpubd_eventhandler(event, device);
prop_object_release(ev);
}
}
static void
devpubd_probe(const char *device)
{
struct devlistargs laa;
size_t len, children, n;
void *p;
int error;
assert(drvctl_fd != -1);
memset(&laa, 0, sizeof(laa));
if (device)
strlcpy(laa.l_devname, device, sizeof(laa.l_devname));
/* Get the child device count for this device */
error = ioctl(drvctl_fd, DRVLISTDEV, &laa);
if (error) {
syslog(LOG_ERR, "DRVLISTDEV failed: %m");
return;
}
child_count_changed:
/* If this device has no children, return */
if (laa.l_children == 0)
return;
/* Allocate a buffer large enough to hold the child device names */
p = laa.l_childname;
children = laa.l_children;
len = children * sizeof(laa.l_childname[0]);
laa.l_childname = realloc(laa.l_childname, len);
if (laa.l_childname == NULL) {
syslog(LOG_ERR, "couldn't allocate %zu bytes", len);
laa.l_childname = p;
goto done;
}
/* Get a list of child devices */
error = ioctl(drvctl_fd, DRVLISTDEV, &laa);
if (error) {
syslog(LOG_ERR, "DRVLISTDEV failed: %m");
goto done;
}
/* If the child count changed between DRVLISTDEV calls, retry */
if (children != laa.l_children)
goto child_count_changed;
/*
* For each child device, first post an attach event and
* then scan each one for additional devices.
*/
for (n = 0; n < laa.l_children; n++)
devpubd_eventhandler(DEVPUBD_ATTACH_EVENT, laa.l_childname[n]);
for (n = 0; n < laa.l_children; n++)
devpubd_probe(laa.l_childname[n]);
done:
free(laa.l_childname);
return;
}
__dead static void
usage(void)
{
fprintf(stderr, "usage: %s [-f]\n", getprogname());
exit(EXIT_FAILURE);
}
int
main(int argc, char *argv[])
{
bool fflag = false;
int ch;
setprogname(argv[0]);
while ((ch = getopt(argc, argv, "fh")) != -1) {
switch (ch) {
case 'f':
fflag = true;
break;
case 'h':
default:
usage();
/* NOTREACHED */
}
}
argc -= optind;
argv += optind;
if (argc)
usage();
/* NOTREACHED */
drvctl_fd = open(DRVCTLDEV, O_RDWR);
if (drvctl_fd == -1)
err(EXIT_FAILURE, "couldn't open " DRVCTLDEV);
if (!fflag) {
if (daemon(0, 0) == -1) {
err(EXIT_FAILURE, "couldn't fork");
}
}
/* Look for devices that are already present */
devpubd_probe(NULL);
devpubd_eventloop();
return EXIT_SUCCESS;
}