/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2009 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* * datalink syseventd module. * * The purpose of this module is to identify all datalink related events, * and react accordingly. */ #include #include #include #include #include #include extern void syseventd_err_print(char *, ...); struct event_list { nvlist_t *ev; struct event_list *next; }; static rcm_handle_t *rcm_hdl = NULL; static boolean_t dl_exiting; static thread_t dl_notify_tid; static mutex_t dl_mx; static cond_t dl_cv; static struct event_list *dl_events; /* ARGSUSED */ static void * datalink_notify_thread(void *arg) { struct event_list *tmp_events, *ep; (void) mutex_lock(&dl_mx); while (! dl_exiting || dl_events != NULL) { if (dl_events == NULL) { (void) cond_wait(&dl_cv, &dl_mx); continue; } tmp_events = dl_events; dl_events = NULL; (void) mutex_unlock(&dl_mx); while (tmp_events != NULL) { /* * Send the PHYSLINK_NEW event to network_rcm to update * the network devices cache accordingly. */ if (rcm_notify_event(rcm_hdl, RCM_RESOURCE_PHYSLINK_NEW, 0, tmp_events->ev, NULL) != RCM_SUCCESS) syseventd_err_print("datalink_mod: Can not" "notify event: %s\n", strerror(errno)); ep = tmp_events; tmp_events = tmp_events->next; nvlist_free(ep->ev); free(ep); } (void) mutex_lock(&dl_mx); } (void) mutex_unlock(&dl_mx); return (NULL); } /*ARGSUSED*/ static int datalink_deliver_event(sysevent_t *ev, int unused) { const char *class = sysevent_get_class_name(ev); const char *subclass = sysevent_get_subclass_name(ev); nvlist_t *nvl; struct event_list *newp, **elpp; if (strcmp(class, EC_DATALINK) != 0 || strcmp(subclass, ESC_DATALINK_PHYS_ADD) != 0) { return (0); } if (sysevent_get_attr_list(ev, &nvl) != 0) return (EINVAL); /* * rcm_notify_event() needs to be called asynchronously otherwise when * sysevent queue is full, deadlock will happen. */ if ((newp = malloc(sizeof (struct event_list))) == NULL) return (ENOMEM); newp->ev = nvl; newp->next = NULL; /* * queue up at the end of the event list and signal notify_thread to * process it. */ (void) mutex_lock(&dl_mx); elpp = &dl_events; while (*elpp != NULL) elpp = &(*elpp)->next; *elpp = newp; (void) cond_signal(&dl_cv); (void) mutex_unlock(&dl_mx); return (0); } static struct slm_mod_ops datalink_mod_ops = { SE_MAJOR_VERSION, SE_MINOR_VERSION, SE_MAX_RETRY_LIMIT, datalink_deliver_event }; struct slm_mod_ops * slm_init() { dl_events = NULL; dl_exiting = B_FALSE; if (rcm_alloc_handle(NULL, 0, NULL, &rcm_hdl) != RCM_SUCCESS) return (NULL); if (thr_create(NULL, 0, datalink_notify_thread, NULL, 0, &dl_notify_tid) != 0) { (void) rcm_free_handle(rcm_hdl); return (NULL); } (void) mutex_init(&dl_mx, USYNC_THREAD, NULL); (void) cond_init(&dl_cv, USYNC_THREAD, NULL); return (&datalink_mod_ops); } void slm_fini() { (void) mutex_lock(&dl_mx); dl_exiting = B_TRUE; (void) cond_signal(&dl_cv); (void) mutex_unlock(&dl_mx); (void) thr_join(dl_notify_tid, NULL, NULL); (void) mutex_destroy(&dl_mx); (void) cond_destroy(&dl_cv); (void) rcm_free_handle(rcm_hdl); rcm_hdl = NULL; }