148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * CDDL HEADER START
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * The contents of this file are subject to the terms of the
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Common Development and Distribution License (the "License").
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * You may not use this file except in compliance with the License.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * or http://www.opensolaris.org/os/licensing.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * See the License for the specific language governing permissions
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * and limitations under the License.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * When distributing Covered Code, include this CDDL HEADER in each
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * If applicable, add the following below this CDDL HEADER, with the
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * fields enclosed by brackets "[]" replaced with your own identifying
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * information: Portions Copyright [yyyy] [name of copyright owner]
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * CDDL HEADER END
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * smbd NIC monitor.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <sys/types.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <stdlib.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <errno.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <string.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <unistd.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <signal.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <stdio.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <net/if.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <net/route.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <sys/sockio.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <sys/socket.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <netinet/in.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <arpa/inet.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <fcntl.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <pthread.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <syslog.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include <smbsrv/libsmb.h>
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#include "smbd.h"
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#define SMBD_NICMON_ENABLE "nicmon_enable"
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#define SMBD_NICMON_THROTTLE 100
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright#define SMBD_NICMON_DEBOUNCE 2
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightextern smbd_t smbd;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic boolean_t smbd_nicmon_enabled = B_TRUE;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/* Use this to stop monitoring */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int eventpipe_write = -1;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/* Use this to refresh service instance */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic char *smbd_nicmon_caller_fmri = NULL;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic void smbd_nicmon_run_check(void);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int smbd_nicmon_setup_rtsock(int);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int smbd_nicmon_needscan(int);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int smbd_nicmon_setup_eventpipe(int *, int *);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic void *smbd_nicmon_daemon(void *);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Start the nic monitor thread.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightint
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_start(const char *svc_fmri)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright pthread_t smbd_nicmon_tid;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int rc;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (smb_nic_init() != SMB_NIC_SUCCESS)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright rc = pthread_create(&smbd_nicmon_tid, NULL, smbd_nicmon_daemon, NULL);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (rc != 0)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (svc_fmri)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_caller_fmri = (char *)svc_fmri;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_run_check();
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (0);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightvoid
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_stop(void)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright uchar_t buf = 1;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (eventpipe_write < 0)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) write(eventpipe_write, &buf, sizeof (buf));
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_caller_fmri = NULL;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smb_nic_fini();
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightint
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_refresh(void)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (smb_nic_init() != SMB_NIC_SUCCESS)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_run_check();
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (0);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * The monitor is enabled unless it is explicitly
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * disabled by setting smbd/nicmon_enable to false.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * smbd/nicmon_enable is not defined by default.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic void
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_run_check(void)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smb_scfhandle_t *hd;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright uint8_t status;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int rc;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_enabled = B_TRUE;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((hd = smb_smf_scf_init(SMBD_FMRI_PREFIX)) == NULL) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_DEBUG,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: smb_smf_scf_init failed");
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright rc = smb_smf_create_service_pgroup(hd, SMBD_PG_NAME);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (rc != SMBD_SMF_OK) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smb_smf_scf_fini(hd);
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_DEBUG,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: smb_smf_create_service_pgroup failed");
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright rc = smb_smf_get_boolean_property(hd, SMBD_NICMON_ENABLE, &status);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (rc == SMBD_SMF_OK && status == 0)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_enabled = B_FALSE;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smb_smf_scf_fini(hd);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Setup routing socket for getting RTM messages.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_setup_rtsock(int af)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int sd;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int flags;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((sd = socket(PF_ROUTE, SOCK_RAW, af)) == -1) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_ERR,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: routing socket failed: %d", errno);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((flags = fcntl(sd, F_GETFL, 0)) < 0) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_ERR,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: fcntl F_GETFL failed: %d", errno);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) close(sd);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((fcntl(sd, F_SETFL, flags | O_NONBLOCK)) < 0) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_ERR,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: fcntl F_SETFL failed: %d", errno);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) close(sd);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (sd);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_needscan(int sock)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright static uint32_t throttle;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright struct rt_msghdr *rtm;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int64_t msg[2048 / 8];
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int need_if_scan = 0;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int nbytes;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright /* Read as many messages as possible and try to empty the sockets */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright for (;;) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright nbytes = read(sock, msg, sizeof (msg));
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (nbytes <= 0)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright break;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright rtm = (struct rt_msghdr *)msg;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (rtm->rtm_version != RTM_VERSION)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright continue;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (nbytes < rtm->rtm_msglen) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((throttle % SMBD_NICMON_THROTTLE) == 0) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_DEBUG,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: short read: %d of %d",
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright nbytes, rtm->rtm_msglen);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright ++throttle;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright continue;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright switch (rtm->rtm_type) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright case RTM_NEWADDR:
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright case RTM_DELADDR:
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright case RTM_IFINFO:
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright need_if_scan = 1;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright break;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright default:
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright break;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (need_if_scan);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Create pipe for signal delivery and set up signal handlers.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic int
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_setup_eventpipe(int *read_pipe, int *write_pipe)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int fds[2];
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((pipe(fds)) < 0) {
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_ERR,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: event pipe failed: %d", errno);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (-1);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *read_pipe = fds[0];
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *write_pipe = fds[1];
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (0);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Create the global routing socket to monitor changes in NIC interfaces.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * We are only interested in new inerface addition/deletion and changes
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * in UP/DOWN status.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Note: only supports AF_INET routing socket. Need to add AF_INET6 to
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * support IPv6.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright/*ARGSUSED*/
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightstatic void *
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightsmbd_nicmon_daemon(void *arg)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright{
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright static uint32_t throttle;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright static int rtsock_v4;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright static int eventpipe_read = -1;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright struct pollfd pollfds[2];
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int pollfd_num = 2;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int i, nic_changed;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright int rc;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((rtsock_v4 = smbd_nicmon_setup_rtsock(AF_INET)) == -1)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (NULL);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright rc = smbd_nicmon_setup_eventpipe(&eventpipe_read, &eventpipe_write);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (rc != 0)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (NULL);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright /*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * Listen for activity on any of the sockets.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * The delay before checking the rtsock will hopefully
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * smooth things out when there is a lot of activity.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright for (;;) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright errno = 0;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright nic_changed = 0;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright pollfds[0].fd = rtsock_v4;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright pollfds[0].events = POLLIN;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright pollfds[1].fd = eventpipe_read;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright pollfds[1].events = POLLIN;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (poll(pollfds, pollfd_num, -1) < 0) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (errno == EINTR)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright continue;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((throttle % SMBD_NICMON_THROTTLE) == 0)
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_DEBUG,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: poll failed: %d", errno);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright ++throttle;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright break;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright for (i = 0; i < pollfd_num; i++) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if ((pollfds[i].fd < 0) ||
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright !(pollfds[i].revents & POLLIN))
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright continue;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (pollfds[i].fd == rtsock_v4) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) sleep(SMBD_NICMON_DEBOUNCE);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright nic_changed = smbd_nicmon_needscan(rtsock_v4);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (pollfds[i].fd == eventpipe_read)
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright goto done;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright /*
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * If the monitor is enabled and something has changed,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright * refresh the registered SMF service.
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright */
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (smbd_nicmon_enabled && nic_changed &&
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_caller_fmri) {
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright if (smf_refresh_instance(smbd_nicmon_caller_fmri) != 0)
b819cea2f73f98c5662230cc9affc8cc84f77fcfGordon Ross syslog(LOG_ERR,
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright "smbd_nicmon: %s refresh failed",
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright smbd_nicmon_caller_fmri);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright }
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wrightdone:
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) close(rtsock_v4);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) close(eventpipe_read);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright (void) close(eventpipe_write);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright eventpipe_write = -1;
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright return (NULL);
148c5f43199ca0b43fc8e3b643aab11cd66ea327Alan Wright}