1 1.7 rmind /* $NetBSD: subr_callback.c,v 1.7 2009/11/24 20:11:50 rmind Exp $ */ 2 1.1 yamt 3 1.1 yamt /*- 4 1.1 yamt * Copyright (c)2006 YAMAMOTO Takashi, 5 1.1 yamt * All rights reserved. 6 1.1 yamt * 7 1.1 yamt * Redistribution and use in source and binary forms, with or without 8 1.1 yamt * modification, are permitted provided that the following conditions 9 1.1 yamt * are met: 10 1.1 yamt * 1. Redistributions of source code must retain the above copyright 11 1.1 yamt * notice, this list of conditions and the following disclaimer. 12 1.1 yamt * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 yamt * notice, this list of conditions and the following disclaimer in the 14 1.1 yamt * documentation and/or other materials provided with the distribution. 15 1.1 yamt * 16 1.1 yamt * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 1.1 yamt * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 1.1 yamt * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 1.1 yamt * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20 1.1 yamt * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 1.1 yamt * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 1.1 yamt * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 1.1 yamt * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 1.1 yamt * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 1.1 yamt * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 1.1 yamt * SUCH DAMAGE. 27 1.1 yamt */ 28 1.1 yamt 29 1.1 yamt #include <sys/cdefs.h> 30 1.7 rmind __KERNEL_RCSID(0, "$NetBSD: subr_callback.c,v 1.7 2009/11/24 20:11:50 rmind Exp $"); 31 1.1 yamt 32 1.1 yamt #include <sys/param.h> 33 1.1 yamt #include <sys/systm.h> 34 1.1 yamt #include <sys/proc.h> 35 1.1 yamt #include <sys/callback.h> 36 1.1 yamt 37 1.1 yamt void 38 1.5 ad callback_head_init(struct callback_head *ch, int ipl) 39 1.1 yamt { 40 1.1 yamt 41 1.3 drochner memset(ch, 0, sizeof(struct callback_head)); 42 1.6 ad mutex_init(&ch->ch_lock, MUTEX_DEFAULT, ipl); 43 1.5 ad cv_init(&ch->ch_cv, "callback"); 44 1.1 yamt TAILQ_INIT(&ch->ch_q); 45 1.4 drochner #if 0 /* already zero-initialized */ 46 1.4 drochner ch->ch_next = NULL; 47 1.4 drochner ch->ch_nentries = 0; 48 1.4 drochner ch->ch_running = 0; 49 1.4 drochner ch->ch_flags = 0; 50 1.4 drochner #endif 51 1.1 yamt } 52 1.1 yamt 53 1.1 yamt void 54 1.5 ad callback_head_destroy(struct callback_head *ch) 55 1.5 ad { 56 1.5 ad 57 1.5 ad mutex_destroy(&ch->ch_lock); 58 1.5 ad cv_destroy(&ch->ch_cv); 59 1.5 ad } 60 1.5 ad 61 1.5 ad void 62 1.1 yamt callback_register(struct callback_head *ch, struct callback_entry *ce, 63 1.1 yamt void *obj, int (*fn)(struct callback_entry *, void *, void *)) 64 1.1 yamt { 65 1.1 yamt 66 1.7 rmind KASSERT(fn != NULL); 67 1.1 yamt ce->ce_func = fn; 68 1.1 yamt ce->ce_obj = obj; 69 1.5 ad mutex_enter(&ch->ch_lock); 70 1.1 yamt TAILQ_INSERT_TAIL(&ch->ch_q, ce, ce_q); 71 1.1 yamt ch->ch_nentries++; 72 1.5 ad mutex_exit(&ch->ch_lock); 73 1.1 yamt } 74 1.1 yamt 75 1.1 yamt void 76 1.1 yamt callback_unregister(struct callback_head *ch, struct callback_entry *ce) 77 1.1 yamt { 78 1.1 yamt 79 1.5 ad mutex_enter(&ch->ch_lock); 80 1.5 ad while (ch->ch_running > 0) 81 1.5 ad cv_wait(&ch->ch_cv, &ch->ch_lock); 82 1.1 yamt if (__predict_false(ch->ch_next == ce)) { 83 1.1 yamt ch->ch_next = TAILQ_NEXT(ce, ce_q); 84 1.1 yamt } 85 1.1 yamt TAILQ_REMOVE(&ch->ch_q, ce, ce_q); 86 1.1 yamt ch->ch_nentries--; 87 1.5 ad mutex_exit(&ch->ch_lock); 88 1.1 yamt } 89 1.1 yamt 90 1.1 yamt static int 91 1.1 yamt callback_runone(struct callback_head *ch, void *arg) 92 1.1 yamt { 93 1.1 yamt struct callback_entry *ce; 94 1.1 yamt int result; 95 1.1 yamt 96 1.1 yamt KASSERT(ch->ch_nentries > 0); 97 1.1 yamt KASSERT(ch->ch_running > 0); 98 1.1 yamt 99 1.1 yamt ce = ch->ch_next; 100 1.1 yamt if (ce == NULL) { 101 1.1 yamt ce = TAILQ_FIRST(&ch->ch_q); 102 1.1 yamt } 103 1.1 yamt KASSERT(ce != NULL); 104 1.7 rmind KASSERT(ce->ce_func != NULL); 105 1.1 yamt result = (*ce->ce_func)(ce, ce->ce_obj, arg); 106 1.1 yamt ch->ch_next = TAILQ_NEXT(ce, ce_q); 107 1.1 yamt return result; 108 1.1 yamt } 109 1.1 yamt 110 1.1 yamt static void 111 1.1 yamt callback_run_enter(struct callback_head *ch) 112 1.1 yamt { 113 1.1 yamt 114 1.5 ad mutex_enter(&ch->ch_lock); 115 1.1 yamt ch->ch_running++; 116 1.5 ad mutex_exit(&ch->ch_lock); 117 1.1 yamt } 118 1.1 yamt 119 1.1 yamt static void 120 1.1 yamt callback_run_leave(struct callback_head *ch) 121 1.1 yamt { 122 1.1 yamt 123 1.5 ad mutex_enter(&ch->ch_lock); 124 1.1 yamt KASSERT(ch->ch_running > 0); 125 1.1 yamt ch->ch_running--; 126 1.5 ad if (ch->ch_running == 0) 127 1.5 ad cv_broadcast(&ch->ch_cv); 128 1.5 ad mutex_exit(&ch->ch_lock); 129 1.1 yamt } 130 1.1 yamt 131 1.1 yamt int 132 1.1 yamt callback_run_roundrobin(struct callback_head *ch, void *arg) 133 1.1 yamt { 134 1.1 yamt int i; 135 1.1 yamt int n; 136 1.1 yamt int result = 0; 137 1.1 yamt 138 1.1 yamt callback_run_enter(ch); 139 1.1 yamt n = ch->ch_nentries; 140 1.1 yamt for (i = 0; i < n; i++) { 141 1.1 yamt result = callback_runone(ch, arg); 142 1.1 yamt if (result != CALLBACK_CHAIN_CONTINUE) { 143 1.1 yamt break; 144 1.1 yamt } 145 1.1 yamt } 146 1.1 yamt callback_run_leave(ch); 147 1.1 yamt 148 1.1 yamt return result; 149 1.1 yamt } 150