Sync usage with man page.
[netbsd-mini2440.git] / external / bsd / bind / dist / bin / tests / shutdown_test.c
blobba2e3822e9d7f9be0f64c84613c3089b72dca8f8
1 /* $NetBSD$ */
3 /*
4 * Copyright (C) 2004, 2007 Internet Systems Consortium, Inc. ("ISC")
5 * Copyright (C) 1998-2001 Internet Software Consortium.
7 * Permission to use, copy, modify, and/or distribute this software for any
8 * purpose with or without fee is hereby granted, provided that the above
9 * copyright notice and this permission notice appear in all copies.
11 * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13 * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16 * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17 * PERFORMANCE OF THIS SOFTWARE.
20 /* Id: shutdown_test.c,v 1.23 2007/06/19 23:46:59 tbox Exp */
22 #include <config.h>
24 #include <stdlib.h>
25 #include <string.h>
27 #include <isc/app.h>
28 #include <isc/mem.h>
29 #include <isc/print.h>
30 #include <isc/task.h>
31 #include <isc/time.h>
32 #include <isc/timer.h>
33 #include <isc/util.h>
35 typedef struct {
36 isc_mem_t * mctx;
37 isc_task_t * task;
38 isc_timer_t * timer;
39 unsigned int ticks;
40 char name[16];
41 isc_boolean_t exiting;
42 isc_task_t * peer;
43 } t_info;
45 #define MAX_TASKS 3
46 #define T2_SHUTDOWNOK (ISC_EVENTCLASS(1024) + 0)
47 #define T2_SHUTDOWNDONE (ISC_EVENTCLASS(1024) + 1)
48 #define FOO_EVENT (ISC_EVENTCLASS(1024) + 2)
50 static t_info tasks[MAX_TASKS];
51 static unsigned int task_count;
52 static isc_taskmgr_t * task_manager;
53 static isc_timermgr_t * timer_manager;
55 static void
56 t1_shutdown(isc_task_t *task, isc_event_t *event) {
57 t_info *info = event->ev_arg;
59 printf("task %s (%p) t1_shutdown\n", info->name, task);
60 isc_task_detach(&info->task);
61 isc_event_free(&event);
64 static void
65 t2_shutdown(isc_task_t *task, isc_event_t *event) {
66 t_info *info = event->ev_arg;
68 printf("task %s (%p) t2_shutdown\n", info->name, task);
69 info->exiting = ISC_TRUE;
70 isc_event_free(&event);
73 static void
74 shutdown_action(isc_task_t *task, isc_event_t *event) {
75 t_info *info = event->ev_arg;
76 isc_event_t *nevent;
78 INSIST(event->ev_type == ISC_TASKEVENT_SHUTDOWN);
80 printf("task %s (%p) shutdown\n", info->name, task);
81 if (strcmp(info->name, "0") == 0) {
82 isc_timer_detach(&info->timer);
83 nevent = isc_event_allocate(info->mctx, info, T2_SHUTDOWNOK,
84 t2_shutdown, &tasks[1],
85 sizeof(*event));
86 RUNTIME_CHECK(nevent != NULL);
87 info->exiting = ISC_TRUE;
88 isc_task_sendanddetach(&info->peer, &nevent);
90 isc_event_free(&event);
93 static void
94 foo_event(isc_task_t *task, isc_event_t *event) {
95 printf("task(%p) foo\n", task);
96 isc_event_free(&event);
99 static void
100 tick(isc_task_t *task, isc_event_t *event) {
101 t_info *info = event->ev_arg;
102 isc_event_t *nevent;
104 INSIST(event->ev_type == ISC_TIMEREVENT_TICK);
106 printf("task %s (%p) tick\n", info->name, task);
108 info->ticks++;
109 if (strcmp(info->name, "1") == 0) {
110 if (info->ticks == 10) {
111 RUNTIME_CHECK(isc_app_shutdown() == ISC_R_SUCCESS);
112 } else if (info->ticks >= 15 && info->exiting) {
113 isc_timer_detach(&info->timer);
114 isc_task_detach(&info->task);
115 nevent = isc_event_allocate(info->mctx, info,
116 T2_SHUTDOWNDONE,
117 t1_shutdown, &tasks[0],
118 sizeof(*event));
119 RUNTIME_CHECK(nevent != NULL);
120 isc_task_send(info->peer, &nevent);
121 isc_task_detach(&info->peer);
123 } else if (strcmp(info->name, "foo") == 0) {
124 isc_timer_detach(&info->timer);
125 nevent = isc_event_allocate(info->mctx, info,
126 FOO_EVENT,
127 foo_event, task,
128 sizeof(*event));
129 RUNTIME_CHECK(nevent != NULL);
130 isc_task_sendanddetach(&task, &nevent);
133 isc_event_free(&event);
136 static t_info *
137 new_task(isc_mem_t *mctx, const char *name) {
138 t_info *ti;
139 isc_time_t expires;
140 isc_interval_t interval;
142 RUNTIME_CHECK(task_count < MAX_TASKS);
143 ti = &tasks[task_count];
144 ti->mctx = mctx;
145 ti->task = NULL;
146 ti->timer = NULL;
147 ti->ticks = 0;
148 if (name != NULL) {
149 INSIST(strlen(name) < sizeof(ti->name));
150 strcpy(ti->name, name);
151 } else
152 sprintf(ti->name, "%d", task_count);
153 RUNTIME_CHECK(isc_task_create(task_manager, 0, &ti->task) ==
154 ISC_R_SUCCESS);
155 RUNTIME_CHECK(isc_task_onshutdown(ti->task, shutdown_action, ti) ==
156 ISC_R_SUCCESS);
158 isc_time_settoepoch(&expires);
159 isc_interval_set(&interval, 1, 0);
160 RUNTIME_CHECK(isc_timer_create(timer_manager, isc_timertype_ticker,
161 &expires, &interval, ti->task,
162 tick, ti, &ti->timer) ==
163 ISC_R_SUCCESS);
165 task_count++;
167 return (ti);
171 main(int argc, char *argv[]) {
172 unsigned int workers;
173 t_info *t1, *t2, *t3;
174 isc_task_t *task;
175 isc_mem_t *mctx, *mctx2;
177 RUNTIME_CHECK(isc_app_start() == ISC_R_SUCCESS);
179 if (argc > 1)
180 workers = atoi(argv[1]);
181 else
182 workers = 2;
183 printf("%d workers\n", workers);
185 mctx = NULL;
186 RUNTIME_CHECK(isc_mem_create(0, 0, &mctx) == ISC_R_SUCCESS);
187 mctx2 = NULL;
188 RUNTIME_CHECK(isc_mem_create(0, 0, &mctx2) == ISC_R_SUCCESS);
189 RUNTIME_CHECK(isc_taskmgr_create(mctx, workers, 0, &task_manager) ==
190 ISC_R_SUCCESS);
191 RUNTIME_CHECK(isc_timermgr_create(mctx, &timer_manager) ==
192 ISC_R_SUCCESS);
194 t1 = new_task(mctx, NULL);
195 t2 = new_task(mctx2, NULL);
196 isc_task_attach(t2->task, &t1->peer);
197 isc_task_attach(t1->task, &t2->peer);
200 * Test run-triggered shutdown.
202 t3 = new_task(mctx2, "foo");
205 * Test implicit shutdown.
207 task = NULL;
208 RUNTIME_CHECK(isc_task_create(task_manager, 0, &task) ==
209 ISC_R_SUCCESS);
210 isc_task_detach(&task);
213 * Test anti-zombie code.
215 RUNTIME_CHECK(isc_task_create(task_manager, 0, &task) ==
216 ISC_R_SUCCESS);
217 isc_task_detach(&task);
219 RUNTIME_CHECK(isc_app_run() == ISC_R_SUCCESS);
221 isc_taskmgr_destroy(&task_manager);
222 isc_timermgr_destroy(&timer_manager);
224 printf("Statistics for mctx:\n");
225 isc_mem_stats(mctx, stdout);
226 isc_mem_destroy(&mctx);
227 printf("Statistics for mctx2:\n");
228 isc_mem_stats(mctx2, stdout);
229 isc_mem_destroy(&mctx2);
231 isc_app_finish();
233 return (0);