Annotation of embedaddon/dhcp/omapip/dispatch.c, revision 1.1
1.1 ! misho 1: /* dispatch.c
! 2:
! 3: I/O dispatcher. */
! 4:
! 5: /*
! 6: * Copyright (c) 2004,2007-2009 by Internet Systems Consortium, Inc. ("ISC")
! 7: * Copyright (c) 1999-2003 by Internet Software Consortium
! 8: *
! 9: * Permission to use, copy, modify, and distribute this software for any
! 10: * purpose with or without fee is hereby granted, provided that the above
! 11: * copyright notice and this permission notice appear in all copies.
! 12: *
! 13: * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES
! 14: * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
! 15: * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR
! 16: * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
! 17: * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
! 18: * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT
! 19: * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
! 20: *
! 21: * Internet Systems Consortium, Inc.
! 22: * 950 Charter Street
! 23: * Redwood City, CA 94063
! 24: * <info@isc.org>
! 25: * https://www.isc.org/
! 26: *
! 27: * This software has been written for Internet Systems Consortium
! 28: * by Ted Lemon in cooperation with Vixie Enterprises and Nominum, Inc.
! 29: * To learn more about Internet Systems Consortium, see
! 30: * ``https://www.isc.org/''. To learn more about Vixie Enterprises,
! 31: * see ``http://www.vix.com''. To learn more about Nominum, Inc., see
! 32: * ``http://www.nominum.com''.
! 33: */
! 34:
! 35: #include "dhcpd.h"
! 36:
! 37: #include <omapip/omapip_p.h>
! 38: #include <sys/time.h>
! 39:
! 40: static omapi_io_object_t omapi_io_states;
! 41: struct timeval cur_tv;
! 42:
! 43: struct eventqueue *rw_queue_empty;
! 44:
! 45: OMAPI_OBJECT_ALLOC (omapi_io,
! 46: omapi_io_object_t, omapi_type_io_object)
! 47: OMAPI_OBJECT_ALLOC (omapi_waiter,
! 48: omapi_waiter_object_t, omapi_type_waiter)
! 49:
! 50: void
! 51: register_eventhandler(struct eventqueue **queue, void (*handler)(void *))
! 52: {
! 53: struct eventqueue *t, *q;
! 54:
! 55: /* traverse to end of list */
! 56: t = NULL;
! 57: for (q = *queue ; q ; q = q->next) {
! 58: if (q->handler == handler)
! 59: return; /* handler already registered */
! 60: t = q;
! 61: }
! 62:
! 63: q = ((struct eventqueue *)dmalloc(sizeof(struct eventqueue), MDL));
! 64: if (!q)
! 65: log_fatal("register_eventhandler: no memory!");
! 66: memset(q, 0, sizeof *q);
! 67: if (t)
! 68: t->next = q;
! 69: else
! 70: *queue = q;
! 71: q->handler = handler;
! 72: return;
! 73: }
! 74:
! 75: void
! 76: unregister_eventhandler(struct eventqueue **queue, void (*handler)(void *))
! 77: {
! 78: struct eventqueue *t, *q;
! 79:
! 80: /* traverse to end of list */
! 81: t= NULL;
! 82: for (q = *queue ; q ; q = q->next) {
! 83: if (q->handler == handler) {
! 84: if (t)
! 85: t->next = q->next;
! 86: else
! 87: *queue = q->next;
! 88: dfree(q, MDL); /* Don't access q after this!*/
! 89: break;
! 90: }
! 91: t = q;
! 92: }
! 93: return;
! 94: }
! 95:
! 96: void
! 97: trigger_event(struct eventqueue **queue)
! 98: {
! 99: struct eventqueue *q;
! 100:
! 101: for (q=*queue ; q ; q=q->next) {
! 102: if (q->handler)
! 103: (*q->handler)(NULL);
! 104: }
! 105: }
! 106:
! 107:
! 108: /* Register an I/O handle so that we can do asynchronous I/O on it. */
! 109:
! 110: isc_result_t omapi_register_io_object (omapi_object_t *h,
! 111: int (*readfd) (omapi_object_t *),
! 112: int (*writefd) (omapi_object_t *),
! 113: isc_result_t (*reader)
! 114: (omapi_object_t *),
! 115: isc_result_t (*writer)
! 116: (omapi_object_t *),
! 117: isc_result_t (*reaper)
! 118: (omapi_object_t *))
! 119: {
! 120: isc_result_t status;
! 121: omapi_io_object_t *obj, *p;
! 122:
! 123: /* omapi_io_states is a static object. If its reference count
! 124: is zero, this is the first I/O handle to be registered, so
! 125: we need to initialize it. Because there is no inner or outer
! 126: pointer on this object, and we're setting its refcnt to 1, it
! 127: will never be freed. */
! 128: if (!omapi_io_states.refcnt) {
! 129: omapi_io_states.refcnt = 1;
! 130: omapi_io_states.type = omapi_type_io_object;
! 131: }
! 132:
! 133: obj = (omapi_io_object_t *)0;
! 134: status = omapi_io_allocate (&obj, MDL);
! 135: if (status != ISC_R_SUCCESS)
! 136: return status;
! 137:
! 138: status = omapi_object_reference (&obj -> inner, h, MDL);
! 139: if (status != ISC_R_SUCCESS) {
! 140: omapi_io_dereference (&obj, MDL);
! 141: return status;
! 142: }
! 143:
! 144: status = omapi_object_reference (&h -> outer,
! 145: (omapi_object_t *)obj, MDL);
! 146: if (status != ISC_R_SUCCESS) {
! 147: omapi_io_dereference (&obj, MDL);
! 148: return status;
! 149: }
! 150:
! 151: /* Find the last I/O state, if there are any. */
! 152: for (p = omapi_io_states.next;
! 153: p && p -> next; p = p -> next)
! 154: ;
! 155: if (p)
! 156: omapi_io_reference (&p -> next, obj, MDL);
! 157: else
! 158: omapi_io_reference (&omapi_io_states.next, obj, MDL);
! 159:
! 160: obj -> readfd = readfd;
! 161: obj -> writefd = writefd;
! 162: obj -> reader = reader;
! 163: obj -> writer = writer;
! 164: obj -> reaper = reaper;
! 165:
! 166: omapi_io_dereference(&obj, MDL);
! 167: return ISC_R_SUCCESS;
! 168: }
! 169:
! 170: /* ReRegister an I/O handle so that we can do asynchronous I/O on it.
! 171: * If the handle doesn't exist we call the register routine to build it.
! 172: * if it does exist we change the functions associated with it, and
! 173: * repoke the fd code to make it happy. Neither the objects nor the
! 174: * fd are allowed to have changed. */
! 175:
! 176: isc_result_t omapi_reregister_io_object (omapi_object_t *h,
! 177: int (*readfd) (omapi_object_t *),
! 178: int (*writefd) (omapi_object_t *),
! 179: isc_result_t (*reader)
! 180: (omapi_object_t *),
! 181: isc_result_t (*writer)
! 182: (omapi_object_t *),
! 183: isc_result_t (*reaper)
! 184: (omapi_object_t *))
! 185: {
! 186: omapi_io_object_t *obj;
! 187:
! 188: if ((!h -> outer) || (h -> outer -> type != omapi_type_io_object)) {
! 189: /* If we don't have an object or if the type isn't what
! 190: * we expect do the normal registration (which will overwrite
! 191: * an incorrect type, that's what we did historically, may
! 192: * want to change that)
! 193: */
! 194: return (omapi_register_io_object (h, readfd, writefd,
! 195: reader, writer, reaper));
! 196: }
! 197:
! 198: /* We have an io object of the correct type, try to update it */
! 199: /*sar*/
! 200: /* Should we validate that the fd matches the previous one?
! 201: * It's suppossed to, that's a requirement, don't bother yet */
! 202:
! 203: obj = (omapi_io_object_t *)h->outer;
! 204:
! 205: obj -> readfd = readfd;
! 206: obj -> writefd = writefd;
! 207: obj -> reader = reader;
! 208: obj -> writer = writer;
! 209: obj -> reaper = reaper;
! 210:
! 211: return (ISC_R_SUCCESS);
! 212: }
! 213:
! 214: isc_result_t omapi_unregister_io_object (omapi_object_t *h)
! 215: {
! 216: omapi_io_object_t *p, *obj, *last, *ph;
! 217:
! 218: if (!h -> outer || h -> outer -> type != omapi_type_io_object)
! 219: return ISC_R_INVALIDARG;
! 220: obj = (omapi_io_object_t *)h -> outer;
! 221: ph = (omapi_io_object_t *)0;
! 222: omapi_io_reference (&ph, obj, MDL);
! 223:
! 224: /* remove from the list of I/O states */
! 225: last = &omapi_io_states;
! 226: for (p = omapi_io_states.next; p; p = p -> next) {
! 227: if (p == obj) {
! 228: omapi_io_dereference (&last -> next, MDL);
! 229: omapi_io_reference (&last -> next, p -> next, MDL);
! 230: break;
! 231: }
! 232: last = p;
! 233: }
! 234: if (obj -> next)
! 235: omapi_io_dereference (&obj -> next, MDL);
! 236:
! 237: if (obj -> outer) {
! 238: if (obj -> outer -> inner == (omapi_object_t *)obj)
! 239: omapi_object_dereference (&obj -> outer -> inner,
! 240: MDL);
! 241: omapi_object_dereference (&obj -> outer, MDL);
! 242: }
! 243: omapi_object_dereference (&obj -> inner, MDL);
! 244: omapi_object_dereference (&h -> outer, MDL);
! 245: omapi_io_dereference (&ph, MDL);
! 246: return ISC_R_SUCCESS;
! 247: }
! 248:
! 249: isc_result_t omapi_dispatch (struct timeval *t)
! 250: {
! 251: return omapi_wait_for_completion ((omapi_object_t *)&omapi_io_states,
! 252: t);
! 253: }
! 254:
! 255: isc_result_t omapi_wait_for_completion (omapi_object_t *object,
! 256: struct timeval *t)
! 257: {
! 258: isc_result_t status;
! 259: omapi_waiter_object_t *waiter;
! 260: omapi_object_t *inner;
! 261:
! 262: if (object) {
! 263: waiter = (omapi_waiter_object_t *)0;
! 264: status = omapi_waiter_allocate (&waiter, MDL);
! 265: if (status != ISC_R_SUCCESS)
! 266: return status;
! 267:
! 268: /* Paste the waiter object onto the inner object we're
! 269: waiting on. */
! 270: for (inner = object; inner -> inner; inner = inner -> inner)
! 271: ;
! 272:
! 273: status = omapi_object_reference (&waiter -> outer, inner, MDL);
! 274: if (status != ISC_R_SUCCESS) {
! 275: omapi_waiter_dereference (&waiter, MDL);
! 276: return status;
! 277: }
! 278:
! 279: status = omapi_object_reference (&inner -> inner,
! 280: (omapi_object_t *)waiter,
! 281: MDL);
! 282: if (status != ISC_R_SUCCESS) {
! 283: omapi_waiter_dereference (&waiter, MDL);
! 284: return status;
! 285: }
! 286: } else
! 287: waiter = (omapi_waiter_object_t *)0;
! 288:
! 289: do {
! 290: status = omapi_one_dispatch ((omapi_object_t *)waiter, t);
! 291: if (status != ISC_R_SUCCESS)
! 292: return status;
! 293: } while (!waiter || !waiter -> ready);
! 294:
! 295: if (waiter -> outer) {
! 296: if (waiter -> outer -> inner) {
! 297: omapi_object_dereference (&waiter -> outer -> inner,
! 298: MDL);
! 299: if (waiter -> inner)
! 300: omapi_object_reference
! 301: (&waiter -> outer -> inner,
! 302: waiter -> inner, MDL);
! 303: }
! 304: omapi_object_dereference (&waiter -> outer, MDL);
! 305: }
! 306: if (waiter -> inner)
! 307: omapi_object_dereference (&waiter -> inner, MDL);
! 308:
! 309: status = waiter -> waitstatus;
! 310: omapi_waiter_dereference (&waiter, MDL);
! 311: return status;
! 312: }
! 313:
! 314: isc_result_t omapi_one_dispatch (omapi_object_t *wo,
! 315: struct timeval *t)
! 316: {
! 317: fd_set r, w, x, rr, ww, xx;
! 318: int max = 0;
! 319: int count;
! 320: int desc;
! 321: struct timeval now, to;
! 322: omapi_io_object_t *io, *prev, *next;
! 323: omapi_waiter_object_t *waiter;
! 324: omapi_object_t *tmp = (omapi_object_t *)0;
! 325:
! 326: if (!wo || wo -> type != omapi_type_waiter)
! 327: waiter = (omapi_waiter_object_t *)0;
! 328: else
! 329: waiter = (omapi_waiter_object_t *)wo;
! 330:
! 331: FD_ZERO (&x);
! 332:
! 333: /* First, see if the timeout has expired, and if so return. */
! 334: if (t) {
! 335: gettimeofday (&now, (struct timezone *)0);
! 336: cur_tv.tv_sec = now.tv_sec;
! 337: cur_tv.tv_usec = now.tv_usec;
! 338: if (now.tv_sec > t -> tv_sec ||
! 339: (now.tv_sec == t -> tv_sec && now.tv_usec >= t -> tv_usec))
! 340: return ISC_R_TIMEDOUT;
! 341:
! 342: /* We didn't time out, so figure out how long until
! 343: we do. */
! 344: to.tv_sec = t -> tv_sec - now.tv_sec;
! 345: to.tv_usec = t -> tv_usec - now.tv_usec;
! 346: if (to.tv_usec < 0) {
! 347: to.tv_usec += 1000000;
! 348: to.tv_sec--;
! 349: }
! 350:
! 351: /* It is possible for the timeout to get set larger than
! 352: the largest time select() is willing to accept.
! 353: Restricting the timeout to a maximum of one day should
! 354: work around this. -DPN. (Ref: Bug #416) */
! 355: if (to.tv_sec > (60 * 60 * 24))
! 356: to.tv_sec = 60 * 60 * 24;
! 357: }
! 358:
! 359: /* If the object we're waiting on has reached completion,
! 360: return now. */
! 361: if (waiter && waiter -> ready)
! 362: return ISC_R_SUCCESS;
! 363:
! 364: again:
! 365: /* If we have no I/O state, we can't proceed. */
! 366: if (!(io = omapi_io_states.next))
! 367: return ISC_R_NOMORE;
! 368:
! 369: /* Set up the read and write masks. */
! 370: FD_ZERO (&r);
! 371: FD_ZERO (&w);
! 372:
! 373: for (; io; io = io -> next) {
! 374: /* Check for a read socket. If we shouldn't be
! 375: trying to read for this I/O object, either there
! 376: won't be a readfd function, or it'll return -1. */
! 377: if (io -> readfd && io -> inner &&
! 378: (desc = (*(io -> readfd)) (io -> inner)) >= 0) {
! 379: FD_SET (desc, &r);
! 380: if (desc > max)
! 381: max = desc;
! 382: }
! 383:
! 384: /* Same deal for write fdets. */
! 385: if (io -> writefd && io -> inner &&
! 386: (desc = (*(io -> writefd)) (io -> inner)) >= 0) {
! 387: FD_SET (desc, &w);
! 388: if (desc > max)
! 389: max = desc;
! 390: }
! 391: }
! 392:
! 393: /* poll if all reader are dry */
! 394: now.tv_sec = 0;
! 395: now.tv_usec = 0;
! 396: rr=r;
! 397: ww=w;
! 398: xx=x;
! 399:
! 400: /* poll once */
! 401: count = select(max + 1, &r, &w, &x, &now);
! 402: if (!count) {
! 403: /* We are dry now */
! 404: trigger_event(&rw_queue_empty);
! 405: /* Wait for a packet or a timeout... XXX */
! 406: r = rr;
! 407: w = ww;
! 408: x = xx;
! 409: count = select(max + 1, &r, &w, &x, t ? &to : NULL);
! 410: }
! 411:
! 412: /* Get the current time... */
! 413: gettimeofday (&cur_tv, (struct timezone *)0);
! 414:
! 415: /* We probably have a bad file descriptor. Figure out which one.
! 416: When we find it, call the reaper function on it, which will
! 417: maybe make it go away, and then try again. */
! 418: if (count < 0) {
! 419: struct timeval t0;
! 420: omapi_io_object_t *prev = (omapi_io_object_t *)0;
! 421: io = (omapi_io_object_t *)0;
! 422: if (omapi_io_states.next)
! 423: omapi_io_reference (&io, omapi_io_states.next, MDL);
! 424:
! 425: while (io) {
! 426: omapi_object_t *obj;
! 427: FD_ZERO (&r);
! 428: FD_ZERO (&w);
! 429: t0.tv_sec = t0.tv_usec = 0;
! 430:
! 431: if (io -> readfd && io -> inner &&
! 432: (desc = (*(io -> readfd)) (io -> inner)) >= 0) {
! 433: FD_SET (desc, &r);
! 434: count = select (desc + 1, &r, &w, &x, &t0);
! 435: bogon:
! 436: if (count < 0) {
! 437: log_error ("Bad descriptor %d.", desc);
! 438: for (obj = (omapi_object_t *)io;
! 439: obj -> outer;
! 440: obj = obj -> outer)
! 441: ;
! 442: for (; obj; obj = obj -> inner) {
! 443: omapi_value_t *ov;
! 444: int len;
! 445: const char *s;
! 446: ov = (omapi_value_t *)0;
! 447: omapi_get_value_str (obj,
! 448: (omapi_object_t *)0,
! 449: "name", &ov);
! 450: if (ov && ov -> value &&
! 451: (ov -> value -> type ==
! 452: omapi_datatype_string)) {
! 453: s = (char *)
! 454: ov -> value -> u.buffer.value;
! 455: len = ov -> value -> u.buffer.len;
! 456: } else {
! 457: s = "";
! 458: len = 0;
! 459: }
! 460: log_error ("Object %lx %s%s%.*s",
! 461: (unsigned long)obj,
! 462: obj -> type -> name,
! 463: len ? " " : "",
! 464: len, s);
! 465: if (len)
! 466: omapi_value_dereference (&ov, MDL);
! 467: }
! 468: (*(io -> reaper)) (io -> inner);
! 469: if (prev) {
! 470: omapi_io_dereference (&prev -> next, MDL);
! 471: if (io -> next)
! 472: omapi_io_reference (&prev -> next,
! 473: io -> next, MDL);
! 474: } else {
! 475: omapi_io_dereference
! 476: (&omapi_io_states.next, MDL);
! 477: if (io -> next)
! 478: omapi_io_reference
! 479: (&omapi_io_states.next,
! 480: io -> next, MDL);
! 481: }
! 482: omapi_io_dereference (&io, MDL);
! 483: goto again;
! 484: }
! 485: }
! 486:
! 487: FD_ZERO (&r);
! 488: FD_ZERO (&w);
! 489: t0.tv_sec = t0.tv_usec = 0;
! 490:
! 491: /* Same deal for write fdets. */
! 492: if (io -> writefd && io -> inner &&
! 493: (desc = (*(io -> writefd)) (io -> inner)) >= 0) {
! 494: FD_SET (desc, &w);
! 495: count = select (desc + 1, &r, &w, &x, &t0);
! 496: if (count < 0)
! 497: goto bogon;
! 498: }
! 499: if (prev)
! 500: omapi_io_dereference (&prev, MDL);
! 501: omapi_io_reference (&prev, io, MDL);
! 502: omapi_io_dereference (&io, MDL);
! 503: if (prev -> next)
! 504: omapi_io_reference (&io, prev -> next, MDL);
! 505: }
! 506: if (prev)
! 507: omapi_io_dereference (&prev, MDL);
! 508:
! 509: }
! 510:
! 511: for (io = omapi_io_states.next; io; io = io -> next) {
! 512: if (!io -> inner)
! 513: continue;
! 514: omapi_object_reference (&tmp, io -> inner, MDL);
! 515: /* Check for a read descriptor, and if there is one,
! 516: see if we got input on that socket. */
! 517: if (io -> readfd &&
! 518: (desc = (*(io -> readfd)) (tmp)) >= 0) {
! 519: if (FD_ISSET (desc, &r))
! 520: ((*(io -> reader)) (tmp));
! 521: }
! 522:
! 523: /* Same deal for write descriptors. */
! 524: if (io -> writefd &&
! 525: (desc = (*(io -> writefd)) (tmp)) >= 0)
! 526: {
! 527: if (FD_ISSET (desc, &w))
! 528: ((*(io -> writer)) (tmp));
! 529: }
! 530: omapi_object_dereference (&tmp, MDL);
! 531: }
! 532:
! 533: /* Now check for I/O handles that are no longer valid,
! 534: and remove them from the list. */
! 535: prev = NULL;
! 536: io = NULL;
! 537: if (omapi_io_states.next != NULL) {
! 538: omapi_io_reference(&io, omapi_io_states.next, MDL);
! 539: }
! 540: while (io != NULL) {
! 541: if ((io->inner == NULL) ||
! 542: ((io->reaper != NULL) &&
! 543: ((io->reaper)(io->inner) != ISC_R_SUCCESS)))
! 544: {
! 545:
! 546: omapi_io_object_t *tmp = NULL;
! 547: /* Save a reference to the next
! 548: pointer, if there is one. */
! 549: if (io->next != NULL) {
! 550: omapi_io_reference(&tmp, io->next, MDL);
! 551: omapi_io_dereference(&io->next, MDL);
! 552: }
! 553: if (prev != NULL) {
! 554: omapi_io_dereference(&prev->next, MDL);
! 555: if (tmp != NULL)
! 556: omapi_io_reference(&prev->next,
! 557: tmp, MDL);
! 558: } else {
! 559: omapi_io_dereference(&omapi_io_states.next,
! 560: MDL);
! 561: if (tmp != NULL)
! 562: omapi_io_reference
! 563: (&omapi_io_states.next,
! 564: tmp, MDL);
! 565: else
! 566: omapi_signal_in(
! 567: (omapi_object_t *)
! 568: &omapi_io_states,
! 569: "ready");
! 570: }
! 571: if (tmp != NULL)
! 572: omapi_io_dereference(&tmp, MDL);
! 573:
! 574: } else {
! 575:
! 576: if (prev != NULL) {
! 577: omapi_io_dereference(&prev, MDL);
! 578: }
! 579: omapi_io_reference(&prev, io, MDL);
! 580:
! 581: }
! 582:
! 583: /*
! 584: * Equivalent to:
! 585: * io = io->next
! 586: * But using our reference counting voodoo.
! 587: */
! 588: next = NULL;
! 589: if (io->next != NULL) {
! 590: omapi_io_reference(&next, io->next, MDL);
! 591: }
! 592: omapi_io_dereference(&io, MDL);
! 593: if (next != NULL) {
! 594: omapi_io_reference(&io, next, MDL);
! 595: omapi_io_dereference(&next, MDL);
! 596: }
! 597: }
! 598: if (prev != NULL) {
! 599: omapi_io_dereference(&prev, MDL);
! 600: }
! 601:
! 602: return ISC_R_SUCCESS;
! 603: }
! 604:
! 605: isc_result_t omapi_io_set_value (omapi_object_t *h,
! 606: omapi_object_t *id,
! 607: omapi_data_string_t *name,
! 608: omapi_typed_data_t *value)
! 609: {
! 610: if (h -> type != omapi_type_io_object)
! 611: return ISC_R_INVALIDARG;
! 612:
! 613: if (h -> inner && h -> inner -> type -> set_value)
! 614: return (*(h -> inner -> type -> set_value))
! 615: (h -> inner, id, name, value);
! 616: return ISC_R_NOTFOUND;
! 617: }
! 618:
! 619: isc_result_t omapi_io_get_value (omapi_object_t *h,
! 620: omapi_object_t *id,
! 621: omapi_data_string_t *name,
! 622: omapi_value_t **value)
! 623: {
! 624: if (h -> type != omapi_type_io_object)
! 625: return ISC_R_INVALIDARG;
! 626:
! 627: if (h -> inner && h -> inner -> type -> get_value)
! 628: return (*(h -> inner -> type -> get_value))
! 629: (h -> inner, id, name, value);
! 630: return ISC_R_NOTFOUND;
! 631: }
! 632:
! 633: /* omapi_io_destroy (object, MDL);
! 634: *
! 635: * Find the requested IO [object] and remove it from the list of io
! 636: * states, causing the cleanup functions to destroy it. Note that we must
! 637: * hold a reference on the object while moving its ->next reference and
! 638: * removing the reference in the chain to the target object...otherwise it
! 639: * may be cleaned up from under us.
! 640: */
! 641: isc_result_t omapi_io_destroy (omapi_object_t *h, const char *file, int line)
! 642: {
! 643: omapi_io_object_t *obj = NULL, *p, *last = NULL, **holder;
! 644:
! 645: if (h -> type != omapi_type_io_object)
! 646: return ISC_R_INVALIDARG;
! 647:
! 648: /* remove from the list of I/O states */
! 649: for (p = omapi_io_states.next; p; p = p -> next) {
! 650: if (p == (omapi_io_object_t *)h) {
! 651: omapi_io_reference (&obj, p, MDL);
! 652:
! 653: if (last)
! 654: holder = &last -> next;
! 655: else
! 656: holder = &omapi_io_states.next;
! 657:
! 658: omapi_io_dereference (holder, MDL);
! 659:
! 660: if (obj -> next) {
! 661: omapi_io_reference (holder, obj -> next, MDL);
! 662: omapi_io_dereference (&obj -> next, MDL);
! 663: }
! 664:
! 665: return omapi_io_dereference (&obj, MDL);
! 666: }
! 667: last = p;
! 668: }
! 669:
! 670: return ISC_R_NOTFOUND;
! 671: }
! 672:
! 673: isc_result_t omapi_io_signal_handler (omapi_object_t *h,
! 674: const char *name, va_list ap)
! 675: {
! 676: if (h -> type != omapi_type_io_object)
! 677: return ISC_R_INVALIDARG;
! 678:
! 679: if (h -> inner && h -> inner -> type -> signal_handler)
! 680: return (*(h -> inner -> type -> signal_handler)) (h -> inner,
! 681: name, ap);
! 682: return ISC_R_NOTFOUND;
! 683: }
! 684:
! 685: isc_result_t omapi_io_stuff_values (omapi_object_t *c,
! 686: omapi_object_t *id,
! 687: omapi_object_t *i)
! 688: {
! 689: if (i -> type != omapi_type_io_object)
! 690: return ISC_R_INVALIDARG;
! 691:
! 692: if (i -> inner && i -> inner -> type -> stuff_values)
! 693: return (*(i -> inner -> type -> stuff_values)) (c, id,
! 694: i -> inner);
! 695: return ISC_R_SUCCESS;
! 696: }
! 697:
! 698: isc_result_t omapi_waiter_signal_handler (omapi_object_t *h,
! 699: const char *name, va_list ap)
! 700: {
! 701: omapi_waiter_object_t *waiter;
! 702:
! 703: if (h -> type != omapi_type_waiter)
! 704: return ISC_R_INVALIDARG;
! 705:
! 706: if (!strcmp (name, "ready")) {
! 707: waiter = (omapi_waiter_object_t *)h;
! 708: waiter -> ready = 1;
! 709: waiter -> waitstatus = ISC_R_SUCCESS;
! 710: return ISC_R_SUCCESS;
! 711: }
! 712:
! 713: if (!strcmp (name, "status")) {
! 714: waiter = (omapi_waiter_object_t *)h;
! 715: waiter -> ready = 1;
! 716: waiter -> waitstatus = va_arg (ap, isc_result_t);
! 717: return ISC_R_SUCCESS;
! 718: }
! 719:
! 720: if (!strcmp (name, "disconnect")) {
! 721: waiter = (omapi_waiter_object_t *)h;
! 722: waiter -> ready = 1;
! 723: waiter -> waitstatus = ISC_R_CONNRESET;
! 724: return ISC_R_SUCCESS;
! 725: }
! 726:
! 727: if (h -> inner && h -> inner -> type -> signal_handler)
! 728: return (*(h -> inner -> type -> signal_handler)) (h -> inner,
! 729: name, ap);
! 730: return ISC_R_NOTFOUND;
! 731: }
! 732:
! 733: isc_result_t omapi_io_state_foreach (isc_result_t (*func) (omapi_object_t *,
! 734: void *),
! 735: void *p)
! 736: {
! 737: omapi_io_object_t *io;
! 738: isc_result_t status;
! 739:
! 740: for (io = omapi_io_states.next; io; io = io -> next) {
! 741: if (io -> inner) {
! 742: status = (*func) (io -> inner, p);
! 743: if (status != ISC_R_SUCCESS)
! 744: return status;
! 745: }
! 746: }
! 747: return ISC_R_SUCCESS;
! 748: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>