8354 sync regcomp(3C) with upstream (fix make catalog)
[unleashed/tickless.git] / usr / src / lib / libdiskmgt / common / events.c
blobe6dc869d2a259360f03a953f0034dbc3a3502b30
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License, Version 1.0 only
6 * (the "License"). You may not use this file except in compliance
7 * with the License.
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
20 * CDDL HEADER END
23 * Copyright 2004 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
27 #pragma ident "%Z%%M% %I% %E% SMI"
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <stropts.h>
32 #include <synch.h>
33 #include <thread.h>
34 #include <libsysevent.h>
35 #include <sys/sysevent/eventdefs.h>
36 #include <sys/sysevent/dev.h>
37 #include <errno.h>
38 #include <libgen.h>
39 #include <unistd.h>
41 #include "libdiskmgt.h"
42 #include "disks_private.h"
44 struct event_list {
45 struct event_list *next;
46 nvlist_t *event;
49 static struct event_list *events = NULL;
50 static int event_error = 0;
51 static int event_break = 0;
52 static mutex_t queue_lock;
53 static sema_t semaphore;
56 * When we add a controller we get an add event for each drive on the
57 * controller. We don't want to walk the devtree for each drive since
58 * we will get the same information each time. So, the solution is to
59 * wait for a few seconds for all of the add events to come in and then
60 * do a single walk. If an add event comes in after we start the walk, we
61 * need to do another walk since we might have missed that drive.
63 * State: 0 - no walker; 1 - walker waiting; 2 - walker running
64 * 0 -> 1; wait a few seconds
65 * 1 -> 2; walking the devtree
66 * 2 -> either 0 or 1 (see below)
67 * While running (state 2), if event comes in, go back to waiting (state 1)
68 * after the walk otherwise go back to none (state 0).
70 * walker_lock protects walker_state & events_pending
72 #define WALK_NONE 0
73 #define WALK_WAITING 1
74 #define WALK_RUNNING 2
75 #define WALK_WAIT_TIME 60 /* wait 60 seconds */
77 static mutex_t walker_lock;
78 static int walker_state = WALK_NONE;
79 static int events_pending = 0;
81 static int sendevents = 0;
83 static void add_event_to_queue(nvlist_t *event);
84 static void cb_watch_events();
85 static void event_handler(sysevent_t *ev);
86 static void print_nvlist(char *prefix, nvlist_t *list);
87 static void walk_devtree();
88 static void walker(void *arg);
90 static void(*callback)(nvlist_t *, int) = NULL;
92 nvlist_t *
93 dm_get_event(int *errp)
95 nvlist_t *event = NULL;
97 *errp = 0;
99 /* wait until there is an event in the queue */
100 /*CONSTCOND*/
101 while (1) {
102 (void) sema_wait(&semaphore);
104 if (event_break) {
105 event_break = 0;
106 *errp = EINTR;
107 break;
110 (void) mutex_lock(&queue_lock);
112 /* first see if we ran out of memory since the last call */
113 if (event_error != 0) {
114 *errp = event_error;
115 event_error = 0;
117 } else if (events != NULL) {
118 struct event_list *tmpp;
120 event = events->event;
121 tmpp = events->next;
122 free(events);
123 events = tmpp;
126 (void) mutex_unlock(&queue_lock);
128 if (*errp != 0 || event != NULL) {
129 break;
133 return (event);
136 void
137 dm_init_event_queue(void (*cb)(nvlist_t *, int), int *errp)
139 if (sendevents == 1) {
140 /* we were already initialized, see what changes to make */
141 *errp = 0;
142 if (cb != callback) {
144 callback = cb;
145 if (cb == NULL) {
146 /* clearing the cb so shutdown the internal cb thread */
147 event_break = 1;
148 (void) sema_post(&semaphore);
150 } else {
151 /* installing a cb; we didn't have one before */
152 thread_t watch_thread;
154 *errp = thr_create(NULL, NULL,
155 (void *(*)(void *))cb_watch_events, NULL, THR_DAEMON,
156 &watch_thread);
160 } else {
161 /* first time to initialize */
162 sendevents = 1;
164 *errp = sema_init(&semaphore, 0, USYNC_THREAD, NULL);
165 if (*errp != 0) {
166 return;
169 if (cb != NULL) {
170 thread_t watch_thread;
172 callback = cb;
174 *errp = thr_create(NULL, NULL,
175 (void *(*)(void *))cb_watch_events, NULL, THR_DAEMON,
176 &watch_thread);
181 void
182 events_new_event(char *name, int dtype, char *etype)
184 nvlist_t *event = NULL;
186 if (!sendevents) {
187 return;
190 if (nvlist_alloc(&event, NVATTRS, 0) != 0) {
191 event = NULL;
193 } else {
194 int error = 0;
196 if (name != NULL &&
197 nvlist_add_string(event, DM_EV_NAME, name) != 0) {
198 error = ENOMEM;
201 if (dtype != -1 &&
202 nvlist_add_uint32(event, DM_EV_DTYPE, dtype) != 0) {
203 error = ENOMEM;
206 if (nvlist_add_string(event, DM_EV_TYPE, etype) != 0) {
207 error = ENOMEM;
210 if (error != 0) {
211 nvlist_free(event);
212 event = NULL;
216 add_event_to_queue(event);
219 void
220 events_new_slice_event(char *dev, char *type)
222 events_new_event(basename(dev), DM_SLICE, type);
226 events_start_event_watcher()
228 sysevent_handle_t *shp;
229 const char *subclass_list[1];
231 /* Bind event handler and create subscriber handle */
232 shp = sysevent_bind_handle(event_handler);
233 if (shp == NULL) {
234 if (dm_debug) {
235 /* keep going when we're debugging */
236 (void) fprintf(stderr, "ERROR: bind failed %d\n", errno);
237 return (0);
239 return (errno);
242 subclass_list[0] = ESC_DISK;
243 if (sysevent_subscribe_event(shp, EC_DEV_ADD, subclass_list, 1) != 0) {
244 if (dm_debug) {
245 /* keep going when we're debugging */
246 (void) fprintf(stderr, "ERROR: subscribe failed\n");
247 return (0);
249 return (errno);
251 if (sysevent_subscribe_event(shp, EC_DEV_REMOVE, subclass_list, 1)
252 != 0) {
253 if (dm_debug) {
254 /* keep going when we're debugging */
255 (void) fprintf(stderr, "ERROR: subscribe failed\n");
256 return (0);
258 return (errno);
261 return (0);
264 static void
265 add_event_to_queue(nvlist_t *event)
267 (void) mutex_lock(&queue_lock);
269 if (event == NULL) {
270 event_error = ENOMEM;
271 (void) mutex_unlock(&queue_lock);
272 return;
275 if (events == NULL) {
277 events = (struct event_list *)malloc(sizeof (struct event_list));
278 if (events == NULL) {
279 event_error = ENOMEM;
280 nvlist_free(event);
281 } else {
282 events->next = NULL;
283 events->event = event;
286 } else {
287 /* already have events in the queue */
288 struct event_list *ep;
289 struct event_list *new_event;
291 /* find the last element in the list */
292 for (ep = events; ep->next != NULL; ep = ep->next);
294 new_event = (struct event_list *)malloc(sizeof (struct event_list));
295 if (new_event == NULL) {
296 event_error = ENOMEM;
297 nvlist_free(event);
298 } else {
299 new_event->next = NULL;
300 new_event->event = event;
301 ep->next = new_event;
305 (void) mutex_unlock(&queue_lock);
307 (void) sema_post(&semaphore);
310 static void
311 cb_watch_events()
313 nvlist_t *event;
314 int error;
316 /*CONSTCOND*/
317 while (1) {
318 event = dm_get_event(&error);
319 if (callback == NULL) {
320 /* end the thread */
321 return;
323 callback(event, error);
327 static void
328 event_handler(sysevent_t *ev)
330 char *class_name;
331 char *pub;
333 class_name = sysevent_get_class_name(ev);
334 if (dm_debug) {
335 (void) fprintf(stderr, "****EVENT: %s %s ", class_name,
336 sysevent_get_subclass_name(ev));
337 if ((pub = sysevent_get_pub_name(ev)) != NULL) {
338 (void) fprintf(stderr, "%s\n", pub);
339 free(pub);
340 } else {
341 (void) fprintf(stderr, "\n");
345 if (libdiskmgt_str_eq(class_name, EC_DEV_ADD)) {
346 /* batch up the adds into a single devtree walk */
347 walk_devtree();
349 } else if (libdiskmgt_str_eq(class_name, EC_DEV_REMOVE)) {
350 nvlist_t *nvlist = NULL;
351 char *dev_name = NULL;
353 (void) sysevent_get_attr_list(ev, &nvlist);
354 if (nvlist != NULL) {
355 (void) nvlist_lookup_string(nvlist, DEV_NAME, &dev_name);
357 if (dm_debug) {
358 print_nvlist("**** ", nvlist);
362 if (dev_name != NULL) {
363 cache_update(DM_EV_DISK_DELETE, dev_name);
366 if (nvlist != NULL) {
367 nvlist_free(nvlist);
373 * This is a debugging function only.
375 static void
376 print_nvlist(char *prefix, nvlist_t *list)
378 nvpair_t *nvp;
380 nvp = nvlist_next_nvpair(list, NULL);
381 while (nvp != NULL) {
382 char *attrname;
383 char *str;
384 uint32_t ui32;
385 uint64_t ui64;
386 char **str_array;
387 uint_t cnt;
388 int i;
390 attrname = nvpair_name(nvp);
391 switch (nvpair_type(nvp)) {
392 case DATA_TYPE_STRING:
393 (void) nvpair_value_string(nvp, &str);
394 (void) fprintf(stderr, "%s%s: %s\n", prefix, attrname, str);
395 break;
397 case DATA_TYPE_STRING_ARRAY:
398 (void) nvpair_value_string_array(nvp, &str_array, &cnt);
399 (void) fprintf(stderr, "%s%s:\n", prefix, attrname);
400 for (i = 0; i < cnt; i++) {
401 (void) fprintf(stderr, "%s %s\n", prefix, str_array[i]);
403 break;
405 case DATA_TYPE_UINT32:
406 (void) nvpair_value_uint32(nvp, &ui32);
407 (void) fprintf(stderr, "%s%s: %u\n", prefix, attrname, ui32);
408 break;
410 case DATA_TYPE_UINT64:
411 (void) nvpair_value_uint64(nvp, &ui64);
412 #ifdef _LP64
413 (void) fprintf(stderr, "%s%s: %lu\n", prefix, attrname, ui64);
414 #else
415 (void) fprintf(stderr, "%s%s: %llu\n", prefix, attrname, ui64);
416 #endif
417 break;
420 case DATA_TYPE_BOOLEAN:
421 (void) fprintf(stderr, "%s%s: true\n", prefix, attrname);
422 break;
424 default:
425 (void) fprintf(stderr, "%s%s: UNSUPPORTED TYPE\n", prefix,
426 attrname);
427 break;
430 nvp = nvlist_next_nvpair(list, nvp);
435 * Batch up the adds into a single devtree walk. We can get a bunch of
436 * adds when we add a controller since we will get an add event for each
437 * drive.
439 static void
440 walk_devtree()
442 thread_t walk_thread;
444 (void) mutex_lock(&walker_lock);
446 switch (walker_state) {
447 case WALK_NONE:
448 if (thr_create(NULL, NULL, (void *(*)(void *))walker, NULL,
449 THR_DAEMON, &walk_thread) == 0) {
450 walker_state = WALK_WAITING;
452 break;
454 case WALK_WAITING:
455 /* absorb the event and do nothing */
456 break;
458 case WALK_RUNNING:
459 events_pending = 1;
460 break;
463 (void) mutex_unlock(&walker_lock);
466 /*ARGSUSED*/
467 static void
468 walker(void *arg)
470 int walk_again = 0;
472 do {
473 /* start by wating for a few seconds to absorb extra events */
474 (void) sleep(WALK_WAIT_TIME);
476 (void) mutex_lock(&walker_lock);
477 walker_state = WALK_RUNNING;
478 (void) mutex_unlock(&walker_lock);
480 cache_update(DM_EV_DISK_ADD, NULL);
482 (void) mutex_lock(&walker_lock);
484 if (events_pending) {
485 events_pending = 0;
486 walker_state = WALK_WAITING;
487 walk_again = 1;
488 } else {
489 walker_state = WALK_NONE;
490 walk_again = 0;
493 (void) mutex_unlock(&walker_lock);
495 } while (walk_again);