xref: /minix3/external/bsd/bind/dist/contrib/idn/idnkit-1.0-src/lib/checker.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: checker.c,v 1.4 2014/12/10 04:37:55 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek #ifndef lint
4*00b67f09SDavid van Moolenbroek static char *rcsid = "Id: checker.c,v 1.1 2003/06/04 00:25:49 marka Exp ";
5*00b67f09SDavid van Moolenbroek #endif
6*00b67f09SDavid van Moolenbroek 
7*00b67f09SDavid van Moolenbroek /*
8*00b67f09SDavid van Moolenbroek  * Copyright (c) 2001,2002 Japan Network Information Center.
9*00b67f09SDavid van Moolenbroek  * All rights reserved.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * By using this file, you agree to the terms and conditions set forth bellow.
12*00b67f09SDavid van Moolenbroek  *
13*00b67f09SDavid van Moolenbroek  * 			LICENSE TERMS AND CONDITIONS
14*00b67f09SDavid van Moolenbroek  *
15*00b67f09SDavid van Moolenbroek  * The following License Terms and Conditions apply, unless a different
16*00b67f09SDavid van Moolenbroek  * license is obtained from Japan Network Information Center ("JPNIC"),
17*00b67f09SDavid van Moolenbroek  * a Japanese association, Kokusai-Kougyou-Kanda Bldg 6F, 2-3-4 Uchi-Kanda,
18*00b67f09SDavid van Moolenbroek  * Chiyoda-ku, Tokyo 101-0047, Japan.
19*00b67f09SDavid van Moolenbroek  *
20*00b67f09SDavid van Moolenbroek  * 1. Use, Modification and Redistribution (including distribution of any
21*00b67f09SDavid van Moolenbroek  *    modified or derived work) in source and/or binary forms is permitted
22*00b67f09SDavid van Moolenbroek  *    under this License Terms and Conditions.
23*00b67f09SDavid van Moolenbroek  *
24*00b67f09SDavid van Moolenbroek  * 2. Redistribution of source code must retain the copyright notices as they
25*00b67f09SDavid van Moolenbroek  *    appear in each source code file, this License Terms and Conditions.
26*00b67f09SDavid van Moolenbroek  *
27*00b67f09SDavid van Moolenbroek  * 3. Redistribution in binary form must reproduce the Copyright Notice,
28*00b67f09SDavid van Moolenbroek  *    this License Terms and Conditions, in the documentation and/or other
29*00b67f09SDavid van Moolenbroek  *    materials provided with the distribution.  For the purposes of binary
30*00b67f09SDavid van Moolenbroek  *    distribution the "Copyright Notice" refers to the following language:
31*00b67f09SDavid van Moolenbroek  *    "Copyright (c) 2000-2002 Japan Network Information Center.  All rights reserved."
32*00b67f09SDavid van Moolenbroek  *
33*00b67f09SDavid van Moolenbroek  * 4. The name of JPNIC may not be used to endorse or promote products
34*00b67f09SDavid van Moolenbroek  *    derived from this Software without specific prior written approval of
35*00b67f09SDavid van Moolenbroek  *    JPNIC.
36*00b67f09SDavid van Moolenbroek  *
37*00b67f09SDavid van Moolenbroek  * 5. Disclaimer/Limitation of Liability: THIS SOFTWARE IS PROVIDED BY JPNIC
38*00b67f09SDavid van Moolenbroek  *    "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
39*00b67f09SDavid van Moolenbroek  *    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
40*00b67f09SDavid van Moolenbroek  *    PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL JPNIC BE LIABLE
41*00b67f09SDavid van Moolenbroek  *    FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
42*00b67f09SDavid van Moolenbroek  *    CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
43*00b67f09SDavid van Moolenbroek  *    SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
44*00b67f09SDavid van Moolenbroek  *    BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
45*00b67f09SDavid van Moolenbroek  *    WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
46*00b67f09SDavid van Moolenbroek  *    OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
47*00b67f09SDavid van Moolenbroek  *    ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
48*00b67f09SDavid van Moolenbroek  */
49*00b67f09SDavid van Moolenbroek 
50*00b67f09SDavid van Moolenbroek #include <config.h>
51*00b67f09SDavid van Moolenbroek 
52*00b67f09SDavid van Moolenbroek #include <stddef.h>
53*00b67f09SDavid van Moolenbroek #include <stdlib.h>
54*00b67f09SDavid van Moolenbroek #include <string.h>
55*00b67f09SDavid van Moolenbroek 
56*00b67f09SDavid van Moolenbroek #include <idn/result.h>
57*00b67f09SDavid van Moolenbroek #include <idn/assert.h>
58*00b67f09SDavid van Moolenbroek #include <idn/logmacro.h>
59*00b67f09SDavid van Moolenbroek #include <idn/checker.h>
60*00b67f09SDavid van Moolenbroek #include <idn/strhash.h>
61*00b67f09SDavid van Moolenbroek #include <idn/debug.h>
62*00b67f09SDavid van Moolenbroek 
63*00b67f09SDavid van Moolenbroek /*
64*00b67f09SDavid van Moolenbroek  * Type for checking scheme.
65*00b67f09SDavid van Moolenbroek  */
66*00b67f09SDavid van Moolenbroek typedef struct {
67*00b67f09SDavid van Moolenbroek 	char *prefix;
68*00b67f09SDavid van Moolenbroek 	char *parameter;
69*00b67f09SDavid van Moolenbroek 	idn_checker_createproc_t create;
70*00b67f09SDavid van Moolenbroek 	idn_checker_destroyproc_t destroy;
71*00b67f09SDavid van Moolenbroek 	idn_checker_lookupproc_t lookup;
72*00b67f09SDavid van Moolenbroek 	void *context;
73*00b67f09SDavid van Moolenbroek } check_scheme_t;
74*00b67f09SDavid van Moolenbroek 
75*00b67f09SDavid van Moolenbroek /*
76*00b67f09SDavid van Moolenbroek  * Standard checking schemes.
77*00b67f09SDavid van Moolenbroek  */
78*00b67f09SDavid van Moolenbroek static const check_scheme_t rfc3491_prohibit_scheme = {
79*00b67f09SDavid van Moolenbroek 	"prohibit#RFC3491",
80*00b67f09SDavid van Moolenbroek 	"RFC3491",
81*00b67f09SDavid van Moolenbroek 	idn_nameprep_createproc,
82*00b67f09SDavid van Moolenbroek 	idn_nameprep_destroyproc,
83*00b67f09SDavid van Moolenbroek 	idn_nameprep_prohibitproc,
84*00b67f09SDavid van Moolenbroek 	NULL,
85*00b67f09SDavid van Moolenbroek };
86*00b67f09SDavid van Moolenbroek 
87*00b67f09SDavid van Moolenbroek static const check_scheme_t rfc3491_unasigned_scheme = {
88*00b67f09SDavid van Moolenbroek 	"unassigned#RFC3491",
89*00b67f09SDavid van Moolenbroek 	"RFC3491",
90*00b67f09SDavid van Moolenbroek 	idn_nameprep_createproc,
91*00b67f09SDavid van Moolenbroek 	idn_nameprep_destroyproc,
92*00b67f09SDavid van Moolenbroek 	idn_nameprep_unassignedproc,
93*00b67f09SDavid van Moolenbroek 	NULL,
94*00b67f09SDavid van Moolenbroek };
95*00b67f09SDavid van Moolenbroek 
96*00b67f09SDavid van Moolenbroek static const check_scheme_t rfc3491_bidi_scheme = {
97*00b67f09SDavid van Moolenbroek 	"bidi#RFC3491",
98*00b67f09SDavid van Moolenbroek 	"RFC3491",
99*00b67f09SDavid van Moolenbroek 	idn_nameprep_createproc,
100*00b67f09SDavid van Moolenbroek 	idn_nameprep_destroyproc,
101*00b67f09SDavid van Moolenbroek 	idn_nameprep_bidiproc,
102*00b67f09SDavid van Moolenbroek 	NULL,
103*00b67f09SDavid van Moolenbroek };
104*00b67f09SDavid van Moolenbroek 
105*00b67f09SDavid van Moolenbroek static const check_scheme_t filecheck_prohibit_scheme = {
106*00b67f09SDavid van Moolenbroek 	"prohibit#fileset",
107*00b67f09SDavid van Moolenbroek 	NULL,
108*00b67f09SDavid van Moolenbroek 	idn__filechecker_createproc,
109*00b67f09SDavid van Moolenbroek 	idn__filechecker_destroyproc,
110*00b67f09SDavid van Moolenbroek 	idn__filechecker_lookupproc,
111*00b67f09SDavid van Moolenbroek 	NULL,
112*00b67f09SDavid van Moolenbroek };
113*00b67f09SDavid van Moolenbroek 
114*00b67f09SDavid van Moolenbroek static const check_scheme_t filecheck_unassigned_scheme = {
115*00b67f09SDavid van Moolenbroek 	"unassigned#fileset",
116*00b67f09SDavid van Moolenbroek 	NULL,
117*00b67f09SDavid van Moolenbroek 	idn__filechecker_createproc,
118*00b67f09SDavid van Moolenbroek 	idn__filechecker_destroyproc,
119*00b67f09SDavid van Moolenbroek 	idn__filechecker_lookupproc,
120*00b67f09SDavid van Moolenbroek 	NULL,
121*00b67f09SDavid van Moolenbroek };
122*00b67f09SDavid van Moolenbroek 
123*00b67f09SDavid van Moolenbroek static const check_scheme_t *standard_check_schemes[] = {
124*00b67f09SDavid van Moolenbroek 	&rfc3491_unasigned_scheme,
125*00b67f09SDavid van Moolenbroek 	&rfc3491_prohibit_scheme,
126*00b67f09SDavid van Moolenbroek 	&rfc3491_bidi_scheme,
127*00b67f09SDavid van Moolenbroek 	&filecheck_prohibit_scheme,
128*00b67f09SDavid van Moolenbroek 	&filecheck_unassigned_scheme,
129*00b67f09SDavid van Moolenbroek 	NULL,
130*00b67f09SDavid van Moolenbroek };
131*00b67f09SDavid van Moolenbroek 
132*00b67f09SDavid van Moolenbroek /*
133*00b67f09SDavid van Moolenbroek  * Hash table for checking schemes.
134*00b67f09SDavid van Moolenbroek  */
135*00b67f09SDavid van Moolenbroek static idn__strhash_t scheme_hash = NULL;
136*00b67f09SDavid van Moolenbroek 
137*00b67f09SDavid van Moolenbroek /*
138*00b67f09SDavid van Moolenbroek  * Mapper object type.
139*00b67f09SDavid van Moolenbroek  */
140*00b67f09SDavid van Moolenbroek struct idn_checker {
141*00b67f09SDavid van Moolenbroek 	int nschemes;
142*00b67f09SDavid van Moolenbroek 	int scheme_size;
143*00b67f09SDavid van Moolenbroek 	check_scheme_t *schemes;
144*00b67f09SDavid van Moolenbroek 	int reference_count;
145*00b67f09SDavid van Moolenbroek };
146*00b67f09SDavid van Moolenbroek 
147*00b67f09SDavid van Moolenbroek #define MAPPER_INITIAL_SCHEME_SIZE	1
148*00b67f09SDavid van Moolenbroek 
149*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_initialize(void)150*00b67f09SDavid van Moolenbroek idn_checker_initialize(void) {
151*00b67f09SDavid van Moolenbroek 	idn_result_t r;
152*00b67f09SDavid van Moolenbroek 	check_scheme_t **scheme;
153*00b67f09SDavid van Moolenbroek 
154*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_initialize()\n"));
155*00b67f09SDavid van Moolenbroek 
156*00b67f09SDavid van Moolenbroek 	if (scheme_hash != NULL) {
157*00b67f09SDavid van Moolenbroek 		r = idn_success;	/* already initialized */
158*00b67f09SDavid van Moolenbroek 		goto ret;
159*00b67f09SDavid van Moolenbroek 	}
160*00b67f09SDavid van Moolenbroek 
161*00b67f09SDavid van Moolenbroek 	r = idn__strhash_create(&scheme_hash);
162*00b67f09SDavid van Moolenbroek 	if (r != idn_success) {
163*00b67f09SDavid van Moolenbroek 		goto ret;
164*00b67f09SDavid van Moolenbroek 	}
165*00b67f09SDavid van Moolenbroek 
166*00b67f09SDavid van Moolenbroek 	for (scheme = (check_scheme_t **)standard_check_schemes;
167*00b67f09SDavid van Moolenbroek 		*scheme != NULL; scheme++) {
168*00b67f09SDavid van Moolenbroek 		r = idn__strhash_put(scheme_hash, (*scheme)->prefix, *scheme);
169*00b67f09SDavid van Moolenbroek 		if (r != idn_success)
170*00b67f09SDavid van Moolenbroek 			goto ret;
171*00b67f09SDavid van Moolenbroek 	}
172*00b67f09SDavid van Moolenbroek 
173*00b67f09SDavid van Moolenbroek 	r = idn_success;
174*00b67f09SDavid van Moolenbroek ret:
175*00b67f09SDavid van Moolenbroek 	if (r != idn_success) {
176*00b67f09SDavid van Moolenbroek 		if (scheme_hash != NULL) {
177*00b67f09SDavid van Moolenbroek 			idn__strhash_destroy(scheme_hash, NULL);
178*00b67f09SDavid van Moolenbroek 			scheme_hash = NULL;
179*00b67f09SDavid van Moolenbroek 		}
180*00b67f09SDavid van Moolenbroek 	}
181*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_initialize(): %s\n", idn_result_tostring(r)));
182*00b67f09SDavid van Moolenbroek 	return (r);
183*00b67f09SDavid van Moolenbroek }
184*00b67f09SDavid van Moolenbroek 
185*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_create(idn_checker_t * ctxp)186*00b67f09SDavid van Moolenbroek idn_checker_create(idn_checker_t *ctxp) {
187*00b67f09SDavid van Moolenbroek 	idn_checker_t ctx = NULL;
188*00b67f09SDavid van Moolenbroek 	idn_result_t r;
189*00b67f09SDavid van Moolenbroek 
190*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
191*00b67f09SDavid van Moolenbroek 	assert(ctxp != NULL);
192*00b67f09SDavid van Moolenbroek 
193*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_create()\n"));
194*00b67f09SDavid van Moolenbroek 
195*00b67f09SDavid van Moolenbroek 	ctx = (idn_checker_t) malloc(sizeof(struct idn_checker));
196*00b67f09SDavid van Moolenbroek 	if (ctx == NULL) {
197*00b67f09SDavid van Moolenbroek 		r = idn_nomemory;
198*00b67f09SDavid van Moolenbroek 		goto ret;
199*00b67f09SDavid van Moolenbroek 	}
200*00b67f09SDavid van Moolenbroek 
201*00b67f09SDavid van Moolenbroek 	ctx->schemes = (check_scheme_t *) malloc(sizeof(check_scheme_t)
202*00b67f09SDavid van Moolenbroek 		 * MAPPER_INITIAL_SCHEME_SIZE);
203*00b67f09SDavid van Moolenbroek 	if (ctx->schemes == NULL) {
204*00b67f09SDavid van Moolenbroek 		r = idn_nomemory;
205*00b67f09SDavid van Moolenbroek 		goto ret;
206*00b67f09SDavid van Moolenbroek 	}
207*00b67f09SDavid van Moolenbroek 
208*00b67f09SDavid van Moolenbroek 	ctx->nschemes = 0;
209*00b67f09SDavid van Moolenbroek 	ctx->scheme_size = MAPPER_INITIAL_SCHEME_SIZE;
210*00b67f09SDavid van Moolenbroek 	ctx->reference_count = 1;
211*00b67f09SDavid van Moolenbroek 	*ctxp = ctx;
212*00b67f09SDavid van Moolenbroek 	r = idn_success;
213*00b67f09SDavid van Moolenbroek ret:
214*00b67f09SDavid van Moolenbroek 	if (r != idn_success) {
215*00b67f09SDavid van Moolenbroek 		if (ctx != NULL)
216*00b67f09SDavid van Moolenbroek 			free(ctx->schemes);
217*00b67f09SDavid van Moolenbroek 		free(ctx);
218*00b67f09SDavid van Moolenbroek 	}
219*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_create(): %s\n", idn_result_tostring(r)));
220*00b67f09SDavid van Moolenbroek 	return (r);
221*00b67f09SDavid van Moolenbroek }
222*00b67f09SDavid van Moolenbroek 
223*00b67f09SDavid van Moolenbroek void
idn_checker_destroy(idn_checker_t ctx)224*00b67f09SDavid van Moolenbroek idn_checker_destroy(idn_checker_t ctx) {
225*00b67f09SDavid van Moolenbroek 	int i;
226*00b67f09SDavid van Moolenbroek 
227*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
228*00b67f09SDavid van Moolenbroek 	assert(ctx != NULL);
229*00b67f09SDavid van Moolenbroek 
230*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_destroy()\n"));
231*00b67f09SDavid van Moolenbroek 
232*00b67f09SDavid van Moolenbroek 	ctx->reference_count--;
233*00b67f09SDavid van Moolenbroek 	if (ctx->reference_count <= 0) {
234*00b67f09SDavid van Moolenbroek 		TRACE(("idn_checker_destroy(): the object is destroyed\n"));
235*00b67f09SDavid van Moolenbroek 		for (i = 0; i < ctx->nschemes; i++)
236*00b67f09SDavid van Moolenbroek 			ctx->schemes[i].destroy(ctx->schemes[i].context);
237*00b67f09SDavid van Moolenbroek 		free(ctx->schemes);
238*00b67f09SDavid van Moolenbroek 		free(ctx);
239*00b67f09SDavid van Moolenbroek 	} else {
240*00b67f09SDavid van Moolenbroek 		TRACE(("idn_checker_destroy(): "
241*00b67f09SDavid van Moolenbroek 		       "update reference count (%d->%d)\n",
242*00b67f09SDavid van Moolenbroek 		       ctx->reference_count + 1, ctx->reference_count));
243*00b67f09SDavid van Moolenbroek 	}
244*00b67f09SDavid van Moolenbroek }
245*00b67f09SDavid van Moolenbroek 
246*00b67f09SDavid van Moolenbroek void
idn_checker_incrref(idn_checker_t ctx)247*00b67f09SDavid van Moolenbroek idn_checker_incrref(idn_checker_t ctx) {
248*00b67f09SDavid van Moolenbroek 	assert(ctx != NULL && scheme_hash != NULL);
249*00b67f09SDavid van Moolenbroek 
250*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_incrref()\n"));
251*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_incrref: update reference count (%d->%d)\n",
252*00b67f09SDavid van Moolenbroek 		ctx->reference_count, ctx->reference_count + 1));
253*00b67f09SDavid van Moolenbroek 
254*00b67f09SDavid van Moolenbroek 	ctx->reference_count++;
255*00b67f09SDavid van Moolenbroek }
256*00b67f09SDavid van Moolenbroek 
257*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_add(idn_checker_t ctx,const char * scheme_name)258*00b67f09SDavid van Moolenbroek idn_checker_add(idn_checker_t ctx, const char *scheme_name) {
259*00b67f09SDavid van Moolenbroek 	idn_result_t r;
260*00b67f09SDavid van Moolenbroek 	check_scheme_t *scheme;
261*00b67f09SDavid van Moolenbroek 	const char *scheme_prefix;
262*00b67f09SDavid van Moolenbroek 	const char *scheme_parameter;
263*00b67f09SDavid van Moolenbroek 	void *scheme_context = NULL;
264*00b67f09SDavid van Moolenbroek 	char *buffer = NULL;
265*00b67f09SDavid van Moolenbroek 
266*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
267*00b67f09SDavid van Moolenbroek 	assert(ctx != NULL);
268*00b67f09SDavid van Moolenbroek 
269*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_add(scheme_name=%s)\n",
270*00b67f09SDavid van Moolenbroek 		idn__debug_xstring(scheme_name, 50)));
271*00b67f09SDavid van Moolenbroek 
272*00b67f09SDavid van Moolenbroek 	/*
273*00b67f09SDavid van Moolenbroek 	 * Split `scheme_name' into `scheme_prefix' and `scheme_parameter'.
274*00b67f09SDavid van Moolenbroek 	 */
275*00b67f09SDavid van Moolenbroek 	scheme_parameter = strchr(scheme_name, ':');
276*00b67f09SDavid van Moolenbroek 	if (scheme_parameter == NULL) {
277*00b67f09SDavid van Moolenbroek 		scheme_prefix = scheme_name;
278*00b67f09SDavid van Moolenbroek 		scheme_parameter = NULL;
279*00b67f09SDavid van Moolenbroek 	} else {
280*00b67f09SDavid van Moolenbroek 		ptrdiff_t scheme_prefixlen;
281*00b67f09SDavid van Moolenbroek 
282*00b67f09SDavid van Moolenbroek 		scheme_prefixlen = scheme_parameter - scheme_name;
283*00b67f09SDavid van Moolenbroek 		buffer = (char *) malloc(scheme_prefixlen + 1);
284*00b67f09SDavid van Moolenbroek 		if (buffer == NULL) {
285*00b67f09SDavid van Moolenbroek 			r = idn_nomemory;
286*00b67f09SDavid van Moolenbroek 			goto ret;
287*00b67f09SDavid van Moolenbroek 		}
288*00b67f09SDavid van Moolenbroek 		memcpy(buffer, scheme_name, scheme_prefixlen);
289*00b67f09SDavid van Moolenbroek 		*(buffer + scheme_prefixlen) = '\0';
290*00b67f09SDavid van Moolenbroek 		scheme_prefix = buffer;
291*00b67f09SDavid van Moolenbroek 		scheme_parameter++;
292*00b67f09SDavid van Moolenbroek 	}
293*00b67f09SDavid van Moolenbroek 
294*00b67f09SDavid van Moolenbroek 	/*
295*00b67f09SDavid van Moolenbroek 	 * Find a scheme.
296*00b67f09SDavid van Moolenbroek 	 */
297*00b67f09SDavid van Moolenbroek 	if (idn__strhash_get(scheme_hash, scheme_prefix, (void **)&scheme)
298*00b67f09SDavid van Moolenbroek 		!= idn_success) {
299*00b67f09SDavid van Moolenbroek 		ERROR(("idn_checker_add(): invalid scheme \"%-.30s\"\n",
300*00b67f09SDavid van Moolenbroek 		       scheme_name));
301*00b67f09SDavid van Moolenbroek 		r = idn_invalid_name;
302*00b67f09SDavid van Moolenbroek 		goto ret;
303*00b67f09SDavid van Moolenbroek 	}
304*00b67f09SDavid van Moolenbroek 	if (scheme_parameter == NULL && scheme->parameter != NULL)
305*00b67f09SDavid van Moolenbroek 		scheme_parameter = scheme->parameter;
306*00b67f09SDavid van Moolenbroek 
307*00b67f09SDavid van Moolenbroek 	/*
308*00b67f09SDavid van Moolenbroek 	 * Add the scheme.
309*00b67f09SDavid van Moolenbroek 	 */
310*00b67f09SDavid van Moolenbroek 	assert(ctx->nschemes <= ctx->scheme_size);
311*00b67f09SDavid van Moolenbroek 
312*00b67f09SDavid van Moolenbroek 	if (ctx->nschemes == ctx->scheme_size) {
313*00b67f09SDavid van Moolenbroek 		check_scheme_t *new_schemes;
314*00b67f09SDavid van Moolenbroek 
315*00b67f09SDavid van Moolenbroek 		new_schemes = (check_scheme_t *) realloc(ctx->schemes,
316*00b67f09SDavid van Moolenbroek 			sizeof(check_scheme_t) * ctx->scheme_size * 2);
317*00b67f09SDavid van Moolenbroek 		if (new_schemes == NULL) {
318*00b67f09SDavid van Moolenbroek 			r = idn_nomemory;
319*00b67f09SDavid van Moolenbroek 			goto ret;
320*00b67f09SDavid van Moolenbroek 		}
321*00b67f09SDavid van Moolenbroek 		ctx->schemes = new_schemes;
322*00b67f09SDavid van Moolenbroek 		ctx->scheme_size *= 2;
323*00b67f09SDavid van Moolenbroek 	}
324*00b67f09SDavid van Moolenbroek 
325*00b67f09SDavid van Moolenbroek 	r = scheme->create(scheme_parameter, &scheme_context);
326*00b67f09SDavid van Moolenbroek 	if (r != idn_success)
327*00b67f09SDavid van Moolenbroek 		goto ret;
328*00b67f09SDavid van Moolenbroek 
329*00b67f09SDavid van Moolenbroek 	memcpy(ctx->schemes + ctx->nschemes, scheme, sizeof(check_scheme_t));
330*00b67f09SDavid van Moolenbroek 	ctx->schemes[ctx->nschemes].context = scheme_context;
331*00b67f09SDavid van Moolenbroek 	ctx->nschemes++;
332*00b67f09SDavid van Moolenbroek 	r = idn_success;
333*00b67f09SDavid van Moolenbroek 
334*00b67f09SDavid van Moolenbroek ret:
335*00b67f09SDavid van Moolenbroek 	free(buffer);
336*00b67f09SDavid van Moolenbroek 	if (r != idn_success)
337*00b67f09SDavid van Moolenbroek 		free(scheme_context);
338*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_add(): %s\n", idn_result_tostring(r)));
339*00b67f09SDavid van Moolenbroek 	return (r);
340*00b67f09SDavid van Moolenbroek }
341*00b67f09SDavid van Moolenbroek 
342*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_addall(idn_checker_t ctx,const char ** scheme_names,int nschemes)343*00b67f09SDavid van Moolenbroek idn_checker_addall(idn_checker_t ctx, const char **scheme_names,
344*00b67f09SDavid van Moolenbroek 		   int nschemes) {
345*00b67f09SDavid van Moolenbroek 	idn_result_t r;
346*00b67f09SDavid van Moolenbroek 	int i;
347*00b67f09SDavid van Moolenbroek 
348*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
349*00b67f09SDavid van Moolenbroek 	assert(ctx != NULL && scheme_names != NULL);
350*00b67f09SDavid van Moolenbroek 
351*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_addall(nschemes=%d)\n", nschemes));
352*00b67f09SDavid van Moolenbroek 
353*00b67f09SDavid van Moolenbroek 	for (i = 0; i < nschemes; i++) {
354*00b67f09SDavid van Moolenbroek 		r = idn_checker_add(ctx, (const char *)*scheme_names);
355*00b67f09SDavid van Moolenbroek 		if (r != idn_success)
356*00b67f09SDavid van Moolenbroek 			goto ret;
357*00b67f09SDavid van Moolenbroek 		scheme_names++;
358*00b67f09SDavid van Moolenbroek 	}
359*00b67f09SDavid van Moolenbroek 
360*00b67f09SDavid van Moolenbroek 	r = idn_success;
361*00b67f09SDavid van Moolenbroek ret:
362*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_addall(): %s\n", idn_result_tostring(r)));
363*00b67f09SDavid van Moolenbroek 	return (r);
364*00b67f09SDavid van Moolenbroek }
365*00b67f09SDavid van Moolenbroek 
366*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_lookup(idn_checker_t ctx,const unsigned long * ucs4,const unsigned long ** found)367*00b67f09SDavid van Moolenbroek idn_checker_lookup(idn_checker_t ctx, const unsigned long *ucs4,
368*00b67f09SDavid van Moolenbroek 		   const unsigned long **found) {
369*00b67f09SDavid van Moolenbroek 	idn_result_t r;
370*00b67f09SDavid van Moolenbroek 	int i;
371*00b67f09SDavid van Moolenbroek 
372*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
373*00b67f09SDavid van Moolenbroek 	assert(ctx != NULL && ucs4 != NULL && found != NULL);
374*00b67f09SDavid van Moolenbroek 
375*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_lookup(ucs4=\"%s\")\n",
376*00b67f09SDavid van Moolenbroek 		idn__debug_ucs4xstring(ucs4, 50)));
377*00b67f09SDavid van Moolenbroek 
378*00b67f09SDavid van Moolenbroek 	/*
379*00b67f09SDavid van Moolenbroek 	 * Lookup.
380*00b67f09SDavid van Moolenbroek 	 */
381*00b67f09SDavid van Moolenbroek 	*found = NULL;
382*00b67f09SDavid van Moolenbroek 
383*00b67f09SDavid van Moolenbroek 	for (i = 0; i < ctx->nschemes; i++) {
384*00b67f09SDavid van Moolenbroek 		TRACE(("idn_checker_lookup(): lookup %s\n",
385*00b67f09SDavid van Moolenbroek 		       ctx->schemes[i].prefix));
386*00b67f09SDavid van Moolenbroek 
387*00b67f09SDavid van Moolenbroek 		r = (ctx->schemes[i].lookup)(ctx->schemes[i].context, ucs4,
388*00b67f09SDavid van Moolenbroek 					     found);
389*00b67f09SDavid van Moolenbroek 		if (r != idn_success)
390*00b67f09SDavid van Moolenbroek 			goto ret;
391*00b67f09SDavid van Moolenbroek 		if (*found != NULL)
392*00b67f09SDavid van Moolenbroek 			break;
393*00b67f09SDavid van Moolenbroek 	}
394*00b67f09SDavid van Moolenbroek 
395*00b67f09SDavid van Moolenbroek 	r = idn_success;
396*00b67f09SDavid van Moolenbroek ret:
397*00b67f09SDavid van Moolenbroek 	if (*found == NULL) {
398*00b67f09SDavid van Moolenbroek 		TRACE(("idn_checker_lookup(): %s (not found)\n",
399*00b67f09SDavid van Moolenbroek 		       idn_result_tostring(r)));
400*00b67f09SDavid van Moolenbroek 	} else {
401*00b67f09SDavid van Moolenbroek 		TRACE(("idn_checker_lookup(): %s (found \\x%04lx)\n",
402*00b67f09SDavid van Moolenbroek 		       idn_result_tostring(r), **found));
403*00b67f09SDavid van Moolenbroek 	}
404*00b67f09SDavid van Moolenbroek 	return (r);
405*00b67f09SDavid van Moolenbroek }
406*00b67f09SDavid van Moolenbroek 
407*00b67f09SDavid van Moolenbroek idn_result_t
idn_checker_register(const char * prefix,idn_checker_createproc_t create,idn_checker_destroyproc_t destroy,idn_checker_lookupproc_t lookup)408*00b67f09SDavid van Moolenbroek idn_checker_register(const char *prefix,
409*00b67f09SDavid van Moolenbroek 		    idn_checker_createproc_t create,
410*00b67f09SDavid van Moolenbroek 		    idn_checker_destroyproc_t destroy,
411*00b67f09SDavid van Moolenbroek 		    idn_checker_lookupproc_t lookup) {
412*00b67f09SDavid van Moolenbroek 	idn_result_t r;
413*00b67f09SDavid van Moolenbroek 	check_scheme_t *scheme = NULL;
414*00b67f09SDavid van Moolenbroek 
415*00b67f09SDavid van Moolenbroek 	assert(scheme_hash != NULL);
416*00b67f09SDavid van Moolenbroek 	assert(prefix != NULL && create != NULL && destroy != NULL &&
417*00b67f09SDavid van Moolenbroek 		lookup != NULL);
418*00b67f09SDavid van Moolenbroek 
419*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_register(prefix=%s)\n", prefix));
420*00b67f09SDavid van Moolenbroek 
421*00b67f09SDavid van Moolenbroek 	scheme = (check_scheme_t *) malloc(sizeof(check_scheme_t));
422*00b67f09SDavid van Moolenbroek 	if (scheme == NULL) {
423*00b67f09SDavid van Moolenbroek 		r = idn_nomemory;
424*00b67f09SDavid van Moolenbroek 		goto ret;
425*00b67f09SDavid van Moolenbroek 	}
426*00b67f09SDavid van Moolenbroek 
427*00b67f09SDavid van Moolenbroek 	scheme->prefix = (char *) malloc(strlen(prefix) + 1);
428*00b67f09SDavid van Moolenbroek 	if (scheme->prefix == NULL) {
429*00b67f09SDavid van Moolenbroek 		r = idn_nomemory;
430*00b67f09SDavid van Moolenbroek 		goto ret;
431*00b67f09SDavid van Moolenbroek 	}
432*00b67f09SDavid van Moolenbroek 
433*00b67f09SDavid van Moolenbroek 	strcpy(scheme->prefix, prefix);
434*00b67f09SDavid van Moolenbroek 	scheme->parameter = NULL;
435*00b67f09SDavid van Moolenbroek 	scheme->create    = create;
436*00b67f09SDavid van Moolenbroek 	scheme->destroy   = destroy;
437*00b67f09SDavid van Moolenbroek 	scheme->lookup    = lookup;
438*00b67f09SDavid van Moolenbroek 
439*00b67f09SDavid van Moolenbroek 	r = idn__strhash_put(scheme_hash, prefix, scheme);
440*00b67f09SDavid van Moolenbroek ret:
441*00b67f09SDavid van Moolenbroek 	if (r != idn_success) {
442*00b67f09SDavid van Moolenbroek 		if (scheme != NULL)
443*00b67f09SDavid van Moolenbroek 			free(scheme->prefix);
444*00b67f09SDavid van Moolenbroek 		free(scheme);
445*00b67f09SDavid van Moolenbroek 	}
446*00b67f09SDavid van Moolenbroek 	TRACE(("idn_checker_register(): %s\n", idn_result_tostring(r)));
447*00b67f09SDavid van Moolenbroek 	return (r);
448*00b67f09SDavid van Moolenbroek }
449