xref: /minix3/external/bsd/bind/dist/lib/dns/rdata/generic/loc_29.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: loc_29.c,v 1.4 2014/12/10 04:37:59 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004, 2005, 2007, 2009  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 1999-2003  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /* Id: loc_29.c,v 1.50 2009/12/04 21:09:33 marka Exp  */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek /* Reviewed: Wed Mar 15 18:13:09 PST 2000 by explorer */
23*00b67f09SDavid van Moolenbroek 
24*00b67f09SDavid van Moolenbroek /* RFC1876 */
25*00b67f09SDavid van Moolenbroek 
26*00b67f09SDavid van Moolenbroek #ifndef RDATA_GENERIC_LOC_29_C
27*00b67f09SDavid van Moolenbroek #define RDATA_GENERIC_LOC_29_C
28*00b67f09SDavid van Moolenbroek 
29*00b67f09SDavid van Moolenbroek #define RRTYPE_LOC_ATTRIBUTES (0)
30*00b67f09SDavid van Moolenbroek 
31*00b67f09SDavid van Moolenbroek static inline isc_result_t
fromtext_loc(ARGS_FROMTEXT)32*00b67f09SDavid van Moolenbroek fromtext_loc(ARGS_FROMTEXT) {
33*00b67f09SDavid van Moolenbroek 	isc_token_t token;
34*00b67f09SDavid van Moolenbroek 	int d1, m1, s1;
35*00b67f09SDavid van Moolenbroek 	int d2, m2, s2;
36*00b67f09SDavid van Moolenbroek 	unsigned char size;
37*00b67f09SDavid van Moolenbroek 	unsigned char hp;
38*00b67f09SDavid van Moolenbroek 	unsigned char vp;
39*00b67f09SDavid van Moolenbroek 	unsigned char version;
40*00b67f09SDavid van Moolenbroek 	isc_boolean_t east = ISC_FALSE;
41*00b67f09SDavid van Moolenbroek 	isc_boolean_t north = ISC_FALSE;
42*00b67f09SDavid van Moolenbroek 	long tmp;
43*00b67f09SDavid van Moolenbroek 	long m;
44*00b67f09SDavid van Moolenbroek 	long cm;
45*00b67f09SDavid van Moolenbroek 	long poweroften[8] = { 1, 10, 100, 1000,
46*00b67f09SDavid van Moolenbroek 			       10000, 100000, 1000000, 10000000 };
47*00b67f09SDavid van Moolenbroek 	int man;
48*00b67f09SDavid van Moolenbroek 	int exp;
49*00b67f09SDavid van Moolenbroek 	char *e;
50*00b67f09SDavid van Moolenbroek 	int i;
51*00b67f09SDavid van Moolenbroek 	unsigned long latitude;
52*00b67f09SDavid van Moolenbroek 	unsigned long longitude;
53*00b67f09SDavid van Moolenbroek 	unsigned long altitude;
54*00b67f09SDavid van Moolenbroek 
55*00b67f09SDavid van Moolenbroek 	REQUIRE(type == 29);
56*00b67f09SDavid van Moolenbroek 
57*00b67f09SDavid van Moolenbroek 	UNUSED(type);
58*00b67f09SDavid van Moolenbroek 	UNUSED(rdclass);
59*00b67f09SDavid van Moolenbroek 	UNUSED(origin);
60*00b67f09SDavid van Moolenbroek 	UNUSED(options);
61*00b67f09SDavid van Moolenbroek 
62*00b67f09SDavid van Moolenbroek 	/*
63*00b67f09SDavid van Moolenbroek 	 * Defaults.
64*00b67f09SDavid van Moolenbroek 	 */
65*00b67f09SDavid van Moolenbroek 	m1 = s1 = 0;
66*00b67f09SDavid van Moolenbroek 	m2 = s2 = 0;
67*00b67f09SDavid van Moolenbroek 	size = 0x12;	/* 1.00m */
68*00b67f09SDavid van Moolenbroek 	hp = 0x16;	/* 10000.00 m */
69*00b67f09SDavid van Moolenbroek 	vp = 0x13;	/* 10.00 m */
70*00b67f09SDavid van Moolenbroek 	version = 0;
71*00b67f09SDavid van Moolenbroek 
72*00b67f09SDavid van Moolenbroek 	/*
73*00b67f09SDavid van Moolenbroek 	 * Degrees.
74*00b67f09SDavid van Moolenbroek 	 */
75*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_number,
76*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
77*00b67f09SDavid van Moolenbroek 	if (token.value.as_ulong > 90U)
78*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
79*00b67f09SDavid van Moolenbroek 	d1 = (int)token.value.as_ulong;
80*00b67f09SDavid van Moolenbroek 	/*
81*00b67f09SDavid van Moolenbroek 	 * Minutes.
82*00b67f09SDavid van Moolenbroek 	 */
83*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
84*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
85*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "N") == 0)
86*00b67f09SDavid van Moolenbroek 		north = ISC_TRUE;
87*00b67f09SDavid van Moolenbroek 	if (north || strcasecmp(DNS_AS_STR(token), "S") == 0)
88*00b67f09SDavid van Moolenbroek 		goto getlong;
89*00b67f09SDavid van Moolenbroek 	m1 = strtol(DNS_AS_STR(token), &e, 10);
90*00b67f09SDavid van Moolenbroek 	if (*e != 0)
91*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
92*00b67f09SDavid van Moolenbroek 	if (m1 < 0 || m1 > 59)
93*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
94*00b67f09SDavid van Moolenbroek 	if (d1 == 90 && m1 != 0)
95*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
96*00b67f09SDavid van Moolenbroek 
97*00b67f09SDavid van Moolenbroek 	/*
98*00b67f09SDavid van Moolenbroek 	 * Seconds.
99*00b67f09SDavid van Moolenbroek 	 */
100*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
101*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
102*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "N") == 0)
103*00b67f09SDavid van Moolenbroek 		north = ISC_TRUE;
104*00b67f09SDavid van Moolenbroek 	if (north || strcasecmp(DNS_AS_STR(token), "S") == 0)
105*00b67f09SDavid van Moolenbroek 		goto getlong;
106*00b67f09SDavid van Moolenbroek 	s1 = strtol(DNS_AS_STR(token), &e, 10);
107*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.')
108*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
109*00b67f09SDavid van Moolenbroek 	if (s1 < 0 || s1 > 59)
110*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
111*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
112*00b67f09SDavid van Moolenbroek 		const char *l;
113*00b67f09SDavid van Moolenbroek 		e++;
114*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 3; i++) {
115*00b67f09SDavid van Moolenbroek 			if (*e == 0)
116*00b67f09SDavid van Moolenbroek 				break;
117*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
118*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
119*00b67f09SDavid van Moolenbroek 			s1 *= 10;
120*00b67f09SDavid van Moolenbroek 			s1 += tmp;
121*00b67f09SDavid van Moolenbroek 		}
122*00b67f09SDavid van Moolenbroek 		for (; i < 3; i++)
123*00b67f09SDavid van Moolenbroek 			s1 *= 10;
124*00b67f09SDavid van Moolenbroek 		l = e;
125*00b67f09SDavid van Moolenbroek 		while (*e != 0) {
126*00b67f09SDavid van Moolenbroek 			if (decvalue(*e++) < 0)
127*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
128*00b67f09SDavid van Moolenbroek 		}
129*00b67f09SDavid van Moolenbroek 		if (*l != '\0' && callbacks != NULL) {
130*00b67f09SDavid van Moolenbroek 			const char *file = isc_lex_getsourcename(lexer);
131*00b67f09SDavid van Moolenbroek 			unsigned long line = isc_lex_getsourceline(lexer);
132*00b67f09SDavid van Moolenbroek 
133*00b67f09SDavid van Moolenbroek 			if (file == NULL)
134*00b67f09SDavid van Moolenbroek 				file = "UNKNOWN";
135*00b67f09SDavid van Moolenbroek 			(*callbacks->warn)(callbacks, "%s: %s:%u: '%s' extra "
136*00b67f09SDavid van Moolenbroek 					   "precision digits ignored",
137*00b67f09SDavid van Moolenbroek 					   "dns_rdata_fromtext", file, line,
138*00b67f09SDavid van Moolenbroek 					   DNS_AS_STR(token));
139*00b67f09SDavid van Moolenbroek 		}
140*00b67f09SDavid van Moolenbroek 	} else
141*00b67f09SDavid van Moolenbroek 		s1 *= 1000;
142*00b67f09SDavid van Moolenbroek 	if (d1 == 90 && s1 != 0)
143*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
144*00b67f09SDavid van Moolenbroek 
145*00b67f09SDavid van Moolenbroek 	/*
146*00b67f09SDavid van Moolenbroek 	 * Direction.
147*00b67f09SDavid van Moolenbroek 	 */
148*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
149*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
150*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "N") == 0)
151*00b67f09SDavid van Moolenbroek 		north = ISC_TRUE;
152*00b67f09SDavid van Moolenbroek 	if (!north && strcasecmp(DNS_AS_STR(token), "S") != 0)
153*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
154*00b67f09SDavid van Moolenbroek 
155*00b67f09SDavid van Moolenbroek  getlong:
156*00b67f09SDavid van Moolenbroek 	/*
157*00b67f09SDavid van Moolenbroek 	 * Degrees.
158*00b67f09SDavid van Moolenbroek 	 */
159*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_number,
160*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
161*00b67f09SDavid van Moolenbroek 	if (token.value.as_ulong > 180U)
162*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
163*00b67f09SDavid van Moolenbroek 	d2 = (int)token.value.as_ulong;
164*00b67f09SDavid van Moolenbroek 
165*00b67f09SDavid van Moolenbroek 	/*
166*00b67f09SDavid van Moolenbroek 	 * Minutes.
167*00b67f09SDavid van Moolenbroek 	 */
168*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
169*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
170*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "E") == 0)
171*00b67f09SDavid van Moolenbroek 		east = ISC_TRUE;
172*00b67f09SDavid van Moolenbroek 	if (east || strcasecmp(DNS_AS_STR(token), "W") == 0)
173*00b67f09SDavid van Moolenbroek 		goto getalt;
174*00b67f09SDavid van Moolenbroek 	m2 = strtol(DNS_AS_STR(token), &e, 10);
175*00b67f09SDavid van Moolenbroek 	if (*e != 0)
176*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
177*00b67f09SDavid van Moolenbroek 	if (m2 < 0 || m2 > 59)
178*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
179*00b67f09SDavid van Moolenbroek 	if (d2 == 180 && m2 != 0)
180*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
181*00b67f09SDavid van Moolenbroek 
182*00b67f09SDavid van Moolenbroek 	/*
183*00b67f09SDavid van Moolenbroek 	 * Seconds.
184*00b67f09SDavid van Moolenbroek 	 */
185*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
186*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
187*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "E") == 0)
188*00b67f09SDavid van Moolenbroek 		east = ISC_TRUE;
189*00b67f09SDavid van Moolenbroek 	if (east || strcasecmp(DNS_AS_STR(token), "W") == 0)
190*00b67f09SDavid van Moolenbroek 		goto getalt;
191*00b67f09SDavid van Moolenbroek 	s2 = strtol(DNS_AS_STR(token), &e, 10);
192*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.')
193*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
194*00b67f09SDavid van Moolenbroek 	if (s2 < 0 || s2 > 59)
195*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
196*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
197*00b67f09SDavid van Moolenbroek 		const char *l;
198*00b67f09SDavid van Moolenbroek 		e++;
199*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 3; i++) {
200*00b67f09SDavid van Moolenbroek 			if (*e == 0)
201*00b67f09SDavid van Moolenbroek 				break;
202*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
203*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
204*00b67f09SDavid van Moolenbroek 			s2 *= 10;
205*00b67f09SDavid van Moolenbroek 			s2 += tmp;
206*00b67f09SDavid van Moolenbroek 		}
207*00b67f09SDavid van Moolenbroek 		for (; i < 3; i++)
208*00b67f09SDavid van Moolenbroek 			s2 *= 10;
209*00b67f09SDavid van Moolenbroek 		l = e;
210*00b67f09SDavid van Moolenbroek 		while (*e != 0) {
211*00b67f09SDavid van Moolenbroek 			if (decvalue(*e++) < 0)
212*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
213*00b67f09SDavid van Moolenbroek 		}
214*00b67f09SDavid van Moolenbroek 		if (*l != '\0' && callbacks != NULL) {
215*00b67f09SDavid van Moolenbroek 			const char *file = isc_lex_getsourcename(lexer);
216*00b67f09SDavid van Moolenbroek 			unsigned long line = isc_lex_getsourceline(lexer);
217*00b67f09SDavid van Moolenbroek 
218*00b67f09SDavid van Moolenbroek 			if (file == NULL)
219*00b67f09SDavid van Moolenbroek 				file = "UNKNOWN";
220*00b67f09SDavid van Moolenbroek 			(*callbacks->warn)(callbacks, "%s: %s:%u: '%s' extra "
221*00b67f09SDavid van Moolenbroek 					   "precision digits ignored",
222*00b67f09SDavid van Moolenbroek 					   "dns_rdata_fromtext",
223*00b67f09SDavid van Moolenbroek 					   file, line, DNS_AS_STR(token));
224*00b67f09SDavid van Moolenbroek 		}
225*00b67f09SDavid van Moolenbroek 	} else
226*00b67f09SDavid van Moolenbroek 		s2 *= 1000;
227*00b67f09SDavid van Moolenbroek 	if (d2 == 180 && s2 != 0)
228*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
229*00b67f09SDavid van Moolenbroek 
230*00b67f09SDavid van Moolenbroek 	/*
231*00b67f09SDavid van Moolenbroek 	 * Direction.
232*00b67f09SDavid van Moolenbroek 	 */
233*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
234*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
235*00b67f09SDavid van Moolenbroek 	if (strcasecmp(DNS_AS_STR(token), "E") == 0)
236*00b67f09SDavid van Moolenbroek 		east = ISC_TRUE;
237*00b67f09SDavid van Moolenbroek 	if (!east && strcasecmp(DNS_AS_STR(token), "W") != 0)
238*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
239*00b67f09SDavid van Moolenbroek 
240*00b67f09SDavid van Moolenbroek  getalt:
241*00b67f09SDavid van Moolenbroek 	/*
242*00b67f09SDavid van Moolenbroek 	 * Altitude.
243*00b67f09SDavid van Moolenbroek 	 */
244*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
245*00b67f09SDavid van Moolenbroek 				      ISC_FALSE));
246*00b67f09SDavid van Moolenbroek 	m = strtol(DNS_AS_STR(token), &e, 10);
247*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.' && *e != 'm')
248*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
249*00b67f09SDavid van Moolenbroek 	if (m < -100000 || m > 42849672)
250*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
251*00b67f09SDavid van Moolenbroek 	cm = 0;
252*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
253*00b67f09SDavid van Moolenbroek 		e++;
254*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 2; i++) {
255*00b67f09SDavid van Moolenbroek 			if (*e == 0 || *e == 'm')
256*00b67f09SDavid van Moolenbroek 				break;
257*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
258*00b67f09SDavid van Moolenbroek 				return (DNS_R_SYNTAX);
259*00b67f09SDavid van Moolenbroek 			cm *= 10;
260*00b67f09SDavid van Moolenbroek 			if (m < 0)
261*00b67f09SDavid van Moolenbroek 				cm -= tmp;
262*00b67f09SDavid van Moolenbroek 			else
263*00b67f09SDavid van Moolenbroek 				cm += tmp;
264*00b67f09SDavid van Moolenbroek 		}
265*00b67f09SDavid van Moolenbroek 		for (; i < 2; i++)
266*00b67f09SDavid van Moolenbroek 			cm *= 10;
267*00b67f09SDavid van Moolenbroek 	}
268*00b67f09SDavid van Moolenbroek 	if (*e == 'm')
269*00b67f09SDavid van Moolenbroek 		e++;
270*00b67f09SDavid van Moolenbroek 	if (*e != 0)
271*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
272*00b67f09SDavid van Moolenbroek 	if (m == -100000 && cm != 0)
273*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
274*00b67f09SDavid van Moolenbroek 	if (m == 42849672 && cm > 95)
275*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
276*00b67f09SDavid van Moolenbroek 	/*
277*00b67f09SDavid van Moolenbroek 	 * Adjust base.
278*00b67f09SDavid van Moolenbroek 	 */
279*00b67f09SDavid van Moolenbroek 	altitude = m + 100000;
280*00b67f09SDavid van Moolenbroek 	altitude *= 100;
281*00b67f09SDavid van Moolenbroek 	altitude += cm;
282*00b67f09SDavid van Moolenbroek 
283*00b67f09SDavid van Moolenbroek 	/*
284*00b67f09SDavid van Moolenbroek 	 * Size: optional.
285*00b67f09SDavid van Moolenbroek 	 */
286*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
287*00b67f09SDavid van Moolenbroek 				      ISC_TRUE));
288*00b67f09SDavid van Moolenbroek 	if (token.type == isc_tokentype_eol ||
289*00b67f09SDavid van Moolenbroek 	    token.type == isc_tokentype_eof) {
290*00b67f09SDavid van Moolenbroek 		isc_lex_ungettoken(lexer, &token);
291*00b67f09SDavid van Moolenbroek 		goto encode;
292*00b67f09SDavid van Moolenbroek 	}
293*00b67f09SDavid van Moolenbroek 	m = strtol(DNS_AS_STR(token), &e, 10);
294*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.' && *e != 'm')
295*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
296*00b67f09SDavid van Moolenbroek 	if (m < 0 || m > 90000000)
297*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
298*00b67f09SDavid van Moolenbroek 	cm = 0;
299*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
300*00b67f09SDavid van Moolenbroek 		e++;
301*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 2; i++) {
302*00b67f09SDavid van Moolenbroek 			if (*e == 0 || *e == 'm')
303*00b67f09SDavid van Moolenbroek 				break;
304*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
305*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
306*00b67f09SDavid van Moolenbroek 			cm *= 10;
307*00b67f09SDavid van Moolenbroek 			cm += tmp;
308*00b67f09SDavid van Moolenbroek 		}
309*00b67f09SDavid van Moolenbroek 		for (; i < 2; i++)
310*00b67f09SDavid van Moolenbroek 			cm *= 10;
311*00b67f09SDavid van Moolenbroek 	}
312*00b67f09SDavid van Moolenbroek 	if (*e == 'm')
313*00b67f09SDavid van Moolenbroek 		e++;
314*00b67f09SDavid van Moolenbroek 	if (*e != 0)
315*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
316*00b67f09SDavid van Moolenbroek 	/*
317*00b67f09SDavid van Moolenbroek 	 * We don't just multiply out as we will overflow.
318*00b67f09SDavid van Moolenbroek 	 */
319*00b67f09SDavid van Moolenbroek 	if (m > 0) {
320*00b67f09SDavid van Moolenbroek 		for (exp = 0; exp < 7; exp++)
321*00b67f09SDavid van Moolenbroek 			if (m < poweroften[exp+1])
322*00b67f09SDavid van Moolenbroek 				break;
323*00b67f09SDavid van Moolenbroek 		man = m / poweroften[exp];
324*00b67f09SDavid van Moolenbroek 		exp += 2;
325*00b67f09SDavid van Moolenbroek 	} else {
326*00b67f09SDavid van Moolenbroek 		if (cm >= 10) {
327*00b67f09SDavid van Moolenbroek 			man = cm / 10;
328*00b67f09SDavid van Moolenbroek 			exp = 1;
329*00b67f09SDavid van Moolenbroek 		} else {
330*00b67f09SDavid van Moolenbroek 			man = cm;
331*00b67f09SDavid van Moolenbroek 			exp = 0;
332*00b67f09SDavid van Moolenbroek 		}
333*00b67f09SDavid van Moolenbroek 	}
334*00b67f09SDavid van Moolenbroek 	size = (man << 4) + exp;
335*00b67f09SDavid van Moolenbroek 
336*00b67f09SDavid van Moolenbroek 	/*
337*00b67f09SDavid van Moolenbroek 	 * Horizontal precision: optional.
338*00b67f09SDavid van Moolenbroek 	 */
339*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
340*00b67f09SDavid van Moolenbroek 				      ISC_TRUE));
341*00b67f09SDavid van Moolenbroek 	if (token.type == isc_tokentype_eol ||
342*00b67f09SDavid van Moolenbroek 	    token.type == isc_tokentype_eof) {
343*00b67f09SDavid van Moolenbroek 		isc_lex_ungettoken(lexer, &token);
344*00b67f09SDavid van Moolenbroek 		goto encode;
345*00b67f09SDavid van Moolenbroek 	}
346*00b67f09SDavid van Moolenbroek 	m = strtol(DNS_AS_STR(token), &e, 10);
347*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.' && *e != 'm')
348*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
349*00b67f09SDavid van Moolenbroek 	if (m < 0 || m > 90000000)
350*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
351*00b67f09SDavid van Moolenbroek 	cm = 0;
352*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
353*00b67f09SDavid van Moolenbroek 		e++;
354*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 2; i++) {
355*00b67f09SDavid van Moolenbroek 			if (*e == 0 || *e == 'm')
356*00b67f09SDavid van Moolenbroek 				break;
357*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
358*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
359*00b67f09SDavid van Moolenbroek 			cm *= 10;
360*00b67f09SDavid van Moolenbroek 			cm += tmp;
361*00b67f09SDavid van Moolenbroek 		}
362*00b67f09SDavid van Moolenbroek 		for (; i < 2; i++)
363*00b67f09SDavid van Moolenbroek 			cm *= 10;
364*00b67f09SDavid van Moolenbroek 	}
365*00b67f09SDavid van Moolenbroek 	if (*e == 'm')
366*00b67f09SDavid van Moolenbroek 		e++;
367*00b67f09SDavid van Moolenbroek 	if (*e != 0)
368*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
369*00b67f09SDavid van Moolenbroek 	/*
370*00b67f09SDavid van Moolenbroek 	 * We don't just multiply out as we will overflow.
371*00b67f09SDavid van Moolenbroek 	 */
372*00b67f09SDavid van Moolenbroek 	if (m > 0) {
373*00b67f09SDavid van Moolenbroek 		for (exp = 0; exp < 7; exp++)
374*00b67f09SDavid van Moolenbroek 			if (m < poweroften[exp+1])
375*00b67f09SDavid van Moolenbroek 				break;
376*00b67f09SDavid van Moolenbroek 		man = m / poweroften[exp];
377*00b67f09SDavid van Moolenbroek 		exp += 2;
378*00b67f09SDavid van Moolenbroek 	} else if (cm >= 10) {
379*00b67f09SDavid van Moolenbroek 		man = cm / 10;
380*00b67f09SDavid van Moolenbroek 		exp = 1;
381*00b67f09SDavid van Moolenbroek 	} else  {
382*00b67f09SDavid van Moolenbroek 		man = cm;
383*00b67f09SDavid van Moolenbroek 		exp = 0;
384*00b67f09SDavid van Moolenbroek 	}
385*00b67f09SDavid van Moolenbroek 	hp = (man << 4) + exp;
386*00b67f09SDavid van Moolenbroek 
387*00b67f09SDavid van Moolenbroek 	/*
388*00b67f09SDavid van Moolenbroek 	 * Vertical precision: optional.
389*00b67f09SDavid van Moolenbroek 	 */
390*00b67f09SDavid van Moolenbroek 	RETERR(isc_lex_getmastertoken(lexer, &token, isc_tokentype_string,
391*00b67f09SDavid van Moolenbroek 				      ISC_TRUE));
392*00b67f09SDavid van Moolenbroek 	if (token.type == isc_tokentype_eol ||
393*00b67f09SDavid van Moolenbroek 	    token.type == isc_tokentype_eof) {
394*00b67f09SDavid van Moolenbroek 		isc_lex_ungettoken(lexer, &token);
395*00b67f09SDavid van Moolenbroek 		goto encode;
396*00b67f09SDavid van Moolenbroek 	}
397*00b67f09SDavid van Moolenbroek 	m = strtol(DNS_AS_STR(token), &e, 10);
398*00b67f09SDavid van Moolenbroek 	if (*e != 0 && *e != '.' && *e != 'm')
399*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
400*00b67f09SDavid van Moolenbroek 	if (m < 0 || m > 90000000)
401*00b67f09SDavid van Moolenbroek 		RETTOK(ISC_R_RANGE);
402*00b67f09SDavid van Moolenbroek 	cm = 0;
403*00b67f09SDavid van Moolenbroek 	if (*e == '.') {
404*00b67f09SDavid van Moolenbroek 		e++;
405*00b67f09SDavid van Moolenbroek 		for (i = 0; i < 2; i++) {
406*00b67f09SDavid van Moolenbroek 			if (*e == 0 || *e == 'm')
407*00b67f09SDavid van Moolenbroek 				break;
408*00b67f09SDavid van Moolenbroek 			if ((tmp = decvalue(*e++)) < 0)
409*00b67f09SDavid van Moolenbroek 				RETTOK(DNS_R_SYNTAX);
410*00b67f09SDavid van Moolenbroek 			cm *= 10;
411*00b67f09SDavid van Moolenbroek 			cm += tmp;
412*00b67f09SDavid van Moolenbroek 		}
413*00b67f09SDavid van Moolenbroek 		for (; i < 2; i++)
414*00b67f09SDavid van Moolenbroek 			cm *= 10;
415*00b67f09SDavid van Moolenbroek 	}
416*00b67f09SDavid van Moolenbroek 	if (*e == 'm')
417*00b67f09SDavid van Moolenbroek 		e++;
418*00b67f09SDavid van Moolenbroek 	if (*e != 0)
419*00b67f09SDavid van Moolenbroek 		RETTOK(DNS_R_SYNTAX);
420*00b67f09SDavid van Moolenbroek 	/*
421*00b67f09SDavid van Moolenbroek 	 * We don't just multiply out as we will overflow.
422*00b67f09SDavid van Moolenbroek 	 */
423*00b67f09SDavid van Moolenbroek 	if (m > 0) {
424*00b67f09SDavid van Moolenbroek 		for (exp = 0; exp < 7; exp++)
425*00b67f09SDavid van Moolenbroek 			if (m < poweroften[exp+1])
426*00b67f09SDavid van Moolenbroek 				break;
427*00b67f09SDavid van Moolenbroek 		man = m / poweroften[exp];
428*00b67f09SDavid van Moolenbroek 		exp += 2;
429*00b67f09SDavid van Moolenbroek 	} else if (cm >= 10) {
430*00b67f09SDavid van Moolenbroek 		man = cm / 10;
431*00b67f09SDavid van Moolenbroek 		exp = 1;
432*00b67f09SDavid van Moolenbroek 	} else {
433*00b67f09SDavid van Moolenbroek 		man = cm;
434*00b67f09SDavid van Moolenbroek 		exp = 0;
435*00b67f09SDavid van Moolenbroek 	}
436*00b67f09SDavid van Moolenbroek 	vp = (man << 4) + exp;
437*00b67f09SDavid van Moolenbroek 
438*00b67f09SDavid van Moolenbroek  encode:
439*00b67f09SDavid van Moolenbroek 	RETERR(mem_tobuffer(target, &version, 1));
440*00b67f09SDavid van Moolenbroek 	RETERR(mem_tobuffer(target, &size, 1));
441*00b67f09SDavid van Moolenbroek 	RETERR(mem_tobuffer(target, &hp, 1));
442*00b67f09SDavid van Moolenbroek 	RETERR(mem_tobuffer(target, &vp, 1));
443*00b67f09SDavid van Moolenbroek 	if (north)
444*00b67f09SDavid van Moolenbroek 		latitude = 0x80000000 + ( d1 * 3600 + m1 * 60 ) * 1000 + s1;
445*00b67f09SDavid van Moolenbroek 	else
446*00b67f09SDavid van Moolenbroek 		latitude = 0x80000000 - ( d1 * 3600 + m1 * 60 ) * 1000 - s1;
447*00b67f09SDavid van Moolenbroek 	RETERR(uint32_tobuffer(latitude, target));
448*00b67f09SDavid van Moolenbroek 
449*00b67f09SDavid van Moolenbroek 	if (east)
450*00b67f09SDavid van Moolenbroek 		longitude = 0x80000000 + ( d2 * 3600 + m2 * 60 ) * 1000 + s2;
451*00b67f09SDavid van Moolenbroek 	else
452*00b67f09SDavid van Moolenbroek 		longitude = 0x80000000 - ( d2 * 3600 + m2 * 60 ) * 1000 - s2;
453*00b67f09SDavid van Moolenbroek 	RETERR(uint32_tobuffer(longitude, target));
454*00b67f09SDavid van Moolenbroek 
455*00b67f09SDavid van Moolenbroek 	return (uint32_tobuffer(altitude, target));
456*00b67f09SDavid van Moolenbroek }
457*00b67f09SDavid van Moolenbroek 
458*00b67f09SDavid van Moolenbroek static inline isc_result_t
totext_loc(ARGS_TOTEXT)459*00b67f09SDavid van Moolenbroek totext_loc(ARGS_TOTEXT) {
460*00b67f09SDavid van Moolenbroek 	int d1, m1, s1, fs1;
461*00b67f09SDavid van Moolenbroek 	int d2, m2, s2, fs2;
462*00b67f09SDavid van Moolenbroek 	unsigned long latitude;
463*00b67f09SDavid van Moolenbroek 	unsigned long longitude;
464*00b67f09SDavid van Moolenbroek 	unsigned long altitude;
465*00b67f09SDavid van Moolenbroek 	isc_boolean_t north;
466*00b67f09SDavid van Moolenbroek 	isc_boolean_t east;
467*00b67f09SDavid van Moolenbroek 	isc_boolean_t below;
468*00b67f09SDavid van Moolenbroek 	isc_region_t sr;
469*00b67f09SDavid van Moolenbroek 	char buf[sizeof("89 59 59.999 N 179 59 59.999 E "
470*00b67f09SDavid van Moolenbroek 			"42849672.95m 90000000m 90000000m 90000000m")];
471*00b67f09SDavid van Moolenbroek 	char sbuf[sizeof("90000000m")];
472*00b67f09SDavid van Moolenbroek 	char hbuf[sizeof("90000000m")];
473*00b67f09SDavid van Moolenbroek 	char vbuf[sizeof("90000000m")];
474*00b67f09SDavid van Moolenbroek 	unsigned char size, hp, vp;
475*00b67f09SDavid van Moolenbroek 	unsigned long poweroften[8] = { 1, 10, 100, 1000,
476*00b67f09SDavid van Moolenbroek 					10000, 100000, 1000000, 10000000 };
477*00b67f09SDavid van Moolenbroek 
478*00b67f09SDavid van Moolenbroek 	UNUSED(tctx);
479*00b67f09SDavid van Moolenbroek 
480*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
481*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->length != 0);
482*00b67f09SDavid van Moolenbroek 
483*00b67f09SDavid van Moolenbroek 	dns_rdata_toregion(rdata, &sr);
484*00b67f09SDavid van Moolenbroek 
485*00b67f09SDavid van Moolenbroek 	/* version = sr.base[0]; */
486*00b67f09SDavid van Moolenbroek 	size = sr.base[1];
487*00b67f09SDavid van Moolenbroek 	INSIST((size&0x0f) < 10 && (size>>4) < 10);
488*00b67f09SDavid van Moolenbroek 	if ((size&0x0f)> 1)
489*00b67f09SDavid van Moolenbroek 		sprintf(sbuf, "%lum", (size>>4) * poweroften[(size&0x0f)-2]);
490*00b67f09SDavid van Moolenbroek 	else
491*00b67f09SDavid van Moolenbroek 		sprintf(sbuf, "0.%02lum", (size>>4) * poweroften[(size&0x0f)]);
492*00b67f09SDavid van Moolenbroek 	hp = sr.base[2];
493*00b67f09SDavid van Moolenbroek 	INSIST((hp&0x0f) < 10 && (hp>>4) < 10);
494*00b67f09SDavid van Moolenbroek 	if ((hp&0x0f)> 1)
495*00b67f09SDavid van Moolenbroek 		sprintf(hbuf, "%lum", (hp>>4) * poweroften[(hp&0x0f)-2]);
496*00b67f09SDavid van Moolenbroek 	else
497*00b67f09SDavid van Moolenbroek 		sprintf(hbuf, "0.%02lum", (hp>>4) * poweroften[(hp&0x0f)]);
498*00b67f09SDavid van Moolenbroek 	vp = sr.base[3];
499*00b67f09SDavid van Moolenbroek 	INSIST((vp&0x0f) < 10 && (vp>>4) < 10);
500*00b67f09SDavid van Moolenbroek 	if ((vp&0x0f)> 1)
501*00b67f09SDavid van Moolenbroek 		sprintf(vbuf, "%lum", (vp>>4) * poweroften[(vp&0x0f)-2]);
502*00b67f09SDavid van Moolenbroek 	else
503*00b67f09SDavid van Moolenbroek 		sprintf(vbuf, "0.%02lum", (vp>>4) * poweroften[(vp&0x0f)]);
504*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
505*00b67f09SDavid van Moolenbroek 
506*00b67f09SDavid van Moolenbroek 	latitude = uint32_fromregion(&sr);
507*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
508*00b67f09SDavid van Moolenbroek 	if (latitude >= 0x80000000) {
509*00b67f09SDavid van Moolenbroek 		north = ISC_TRUE;
510*00b67f09SDavid van Moolenbroek 		latitude -= 0x80000000;
511*00b67f09SDavid van Moolenbroek 	} else {
512*00b67f09SDavid van Moolenbroek 		north = ISC_FALSE;
513*00b67f09SDavid van Moolenbroek 		latitude = 0x80000000 - latitude;
514*00b67f09SDavid van Moolenbroek 	}
515*00b67f09SDavid van Moolenbroek 	fs1 = (int)(latitude % 1000);
516*00b67f09SDavid van Moolenbroek 	latitude /= 1000;
517*00b67f09SDavid van Moolenbroek 	s1 = (int)(latitude % 60);
518*00b67f09SDavid van Moolenbroek 	latitude /= 60;
519*00b67f09SDavid van Moolenbroek 	m1 = (int)(latitude % 60);
520*00b67f09SDavid van Moolenbroek 	latitude /= 60;
521*00b67f09SDavid van Moolenbroek 	d1 = (int)latitude;
522*00b67f09SDavid van Moolenbroek 	INSIST(latitude <= 90U);
523*00b67f09SDavid van Moolenbroek 
524*00b67f09SDavid van Moolenbroek 	longitude = uint32_fromregion(&sr);
525*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
526*00b67f09SDavid van Moolenbroek 	if (longitude >= 0x80000000) {
527*00b67f09SDavid van Moolenbroek 		east = ISC_TRUE;
528*00b67f09SDavid van Moolenbroek 		longitude -= 0x80000000;
529*00b67f09SDavid van Moolenbroek 	} else {
530*00b67f09SDavid van Moolenbroek 		east = ISC_FALSE;
531*00b67f09SDavid van Moolenbroek 		longitude = 0x80000000 - longitude;
532*00b67f09SDavid van Moolenbroek 	}
533*00b67f09SDavid van Moolenbroek 	fs2 = (int)(longitude % 1000);
534*00b67f09SDavid van Moolenbroek 	longitude /= 1000;
535*00b67f09SDavid van Moolenbroek 	s2 = (int)(longitude % 60);
536*00b67f09SDavid van Moolenbroek 	longitude /= 60;
537*00b67f09SDavid van Moolenbroek 	m2 = (int)(longitude % 60);
538*00b67f09SDavid van Moolenbroek 	longitude /= 60;
539*00b67f09SDavid van Moolenbroek 	d2 = (int)longitude;
540*00b67f09SDavid van Moolenbroek 	INSIST(longitude <= 180U);
541*00b67f09SDavid van Moolenbroek 
542*00b67f09SDavid van Moolenbroek 	altitude = uint32_fromregion(&sr);
543*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
544*00b67f09SDavid van Moolenbroek 	if (altitude < 10000000U) {
545*00b67f09SDavid van Moolenbroek 		below = ISC_TRUE;
546*00b67f09SDavid van Moolenbroek 		altitude = 10000000 - altitude;
547*00b67f09SDavid van Moolenbroek 	} else {
548*00b67f09SDavid van Moolenbroek 		below =ISC_FALSE;
549*00b67f09SDavid van Moolenbroek 		altitude -= 10000000;
550*00b67f09SDavid van Moolenbroek 	}
551*00b67f09SDavid van Moolenbroek 
552*00b67f09SDavid van Moolenbroek 	sprintf(buf, "%d %d %d.%03d %s %d %d %d.%03d %s %s%ld.%02ldm %s %s %s",
553*00b67f09SDavid van Moolenbroek 		d1, m1, s1, fs1, north ? "N" : "S",
554*00b67f09SDavid van Moolenbroek 		d2, m2, s2, fs2, east ? "E" : "W",
555*00b67f09SDavid van Moolenbroek 		below ? "-" : "", altitude/100, altitude % 100,
556*00b67f09SDavid van Moolenbroek 		sbuf, hbuf, vbuf);
557*00b67f09SDavid van Moolenbroek 
558*00b67f09SDavid van Moolenbroek 	return (str_totext(buf, target));
559*00b67f09SDavid van Moolenbroek }
560*00b67f09SDavid van Moolenbroek 
561*00b67f09SDavid van Moolenbroek static inline isc_result_t
fromwire_loc(ARGS_FROMWIRE)562*00b67f09SDavid van Moolenbroek fromwire_loc(ARGS_FROMWIRE) {
563*00b67f09SDavid van Moolenbroek 	isc_region_t sr;
564*00b67f09SDavid van Moolenbroek 	unsigned char c;
565*00b67f09SDavid van Moolenbroek 	unsigned long latitude;
566*00b67f09SDavid van Moolenbroek 	unsigned long longitude;
567*00b67f09SDavid van Moolenbroek 
568*00b67f09SDavid van Moolenbroek 	REQUIRE(type == 29);
569*00b67f09SDavid van Moolenbroek 
570*00b67f09SDavid van Moolenbroek 	UNUSED(type);
571*00b67f09SDavid van Moolenbroek 	UNUSED(rdclass);
572*00b67f09SDavid van Moolenbroek 	UNUSED(dctx);
573*00b67f09SDavid van Moolenbroek 	UNUSED(options);
574*00b67f09SDavid van Moolenbroek 
575*00b67f09SDavid van Moolenbroek 	isc_buffer_activeregion(source, &sr);
576*00b67f09SDavid van Moolenbroek 	if (sr.length < 1)
577*00b67f09SDavid van Moolenbroek 		return (ISC_R_UNEXPECTEDEND);
578*00b67f09SDavid van Moolenbroek 	if (sr.base[0] != 0)
579*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTIMPLEMENTED);
580*00b67f09SDavid van Moolenbroek 	if (sr.length < 16)
581*00b67f09SDavid van Moolenbroek 		return (ISC_R_UNEXPECTEDEND);
582*00b67f09SDavid van Moolenbroek 
583*00b67f09SDavid van Moolenbroek 	/*
584*00b67f09SDavid van Moolenbroek 	 * Size.
585*00b67f09SDavid van Moolenbroek 	 */
586*00b67f09SDavid van Moolenbroek 	c = sr.base[1];
587*00b67f09SDavid van Moolenbroek 	if (c != 0)
588*00b67f09SDavid van Moolenbroek 		if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
589*00b67f09SDavid van Moolenbroek 			return (ISC_R_RANGE);
590*00b67f09SDavid van Moolenbroek 
591*00b67f09SDavid van Moolenbroek 	/*
592*00b67f09SDavid van Moolenbroek 	 * Horizontal precision.
593*00b67f09SDavid van Moolenbroek 	 */
594*00b67f09SDavid van Moolenbroek 	c = sr.base[2];
595*00b67f09SDavid van Moolenbroek 	if (c != 0)
596*00b67f09SDavid van Moolenbroek 		if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
597*00b67f09SDavid van Moolenbroek 			return (ISC_R_RANGE);
598*00b67f09SDavid van Moolenbroek 
599*00b67f09SDavid van Moolenbroek 	/*
600*00b67f09SDavid van Moolenbroek 	 * Vertical precision.
601*00b67f09SDavid van Moolenbroek 	 */
602*00b67f09SDavid van Moolenbroek 	c = sr.base[3];
603*00b67f09SDavid van Moolenbroek 	if (c != 0)
604*00b67f09SDavid van Moolenbroek 		if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
605*00b67f09SDavid van Moolenbroek 			return (ISC_R_RANGE);
606*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
607*00b67f09SDavid van Moolenbroek 
608*00b67f09SDavid van Moolenbroek 	/*
609*00b67f09SDavid van Moolenbroek 	 * Latitude.
610*00b67f09SDavid van Moolenbroek 	 */
611*00b67f09SDavid van Moolenbroek 	latitude = uint32_fromregion(&sr);
612*00b67f09SDavid van Moolenbroek 	if (latitude < (0x80000000UL - 90 * 3600000) ||
613*00b67f09SDavid van Moolenbroek 	    latitude > (0x80000000UL + 90 * 3600000))
614*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
615*00b67f09SDavid van Moolenbroek 	isc_region_consume(&sr, 4);
616*00b67f09SDavid van Moolenbroek 
617*00b67f09SDavid van Moolenbroek 	/*
618*00b67f09SDavid van Moolenbroek 	 * Longitude.
619*00b67f09SDavid van Moolenbroek 	 */
620*00b67f09SDavid van Moolenbroek 	longitude = uint32_fromregion(&sr);
621*00b67f09SDavid van Moolenbroek 	if (longitude < (0x80000000UL - 180 * 3600000) ||
622*00b67f09SDavid van Moolenbroek 	    longitude > (0x80000000UL + 180 * 3600000))
623*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
624*00b67f09SDavid van Moolenbroek 
625*00b67f09SDavid van Moolenbroek 	/*
626*00b67f09SDavid van Moolenbroek 	 * Altitude.
627*00b67f09SDavid van Moolenbroek 	 * All values possible.
628*00b67f09SDavid van Moolenbroek 	 */
629*00b67f09SDavid van Moolenbroek 
630*00b67f09SDavid van Moolenbroek 	isc_buffer_activeregion(source, &sr);
631*00b67f09SDavid van Moolenbroek 	isc_buffer_forward(source, 16);
632*00b67f09SDavid van Moolenbroek 	return (mem_tobuffer(target, sr.base, 16));
633*00b67f09SDavid van Moolenbroek }
634*00b67f09SDavid van Moolenbroek 
635*00b67f09SDavid van Moolenbroek static inline isc_result_t
towire_loc(ARGS_TOWIRE)636*00b67f09SDavid van Moolenbroek towire_loc(ARGS_TOWIRE) {
637*00b67f09SDavid van Moolenbroek 	UNUSED(cctx);
638*00b67f09SDavid van Moolenbroek 
639*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
640*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->length != 0);
641*00b67f09SDavid van Moolenbroek 
642*00b67f09SDavid van Moolenbroek 	return (mem_tobuffer(target, rdata->data, rdata->length));
643*00b67f09SDavid van Moolenbroek }
644*00b67f09SDavid van Moolenbroek 
645*00b67f09SDavid van Moolenbroek static inline int
compare_loc(ARGS_COMPARE)646*00b67f09SDavid van Moolenbroek compare_loc(ARGS_COMPARE) {
647*00b67f09SDavid van Moolenbroek 	isc_region_t r1;
648*00b67f09SDavid van Moolenbroek 	isc_region_t r2;
649*00b67f09SDavid van Moolenbroek 
650*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata1->type == rdata2->type);
651*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata1->rdclass == rdata2->rdclass);
652*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata1->type == 29);
653*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata1->length != 0);
654*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata2->length != 0);
655*00b67f09SDavid van Moolenbroek 
656*00b67f09SDavid van Moolenbroek 	dns_rdata_toregion(rdata1, &r1);
657*00b67f09SDavid van Moolenbroek 	dns_rdata_toregion(rdata2, &r2);
658*00b67f09SDavid van Moolenbroek 	return (isc_region_compare(&r1, &r2));
659*00b67f09SDavid van Moolenbroek }
660*00b67f09SDavid van Moolenbroek 
661*00b67f09SDavid van Moolenbroek static inline isc_result_t
fromstruct_loc(ARGS_FROMSTRUCT)662*00b67f09SDavid van Moolenbroek fromstruct_loc(ARGS_FROMSTRUCT) {
663*00b67f09SDavid van Moolenbroek 	dns_rdata_loc_t *loc = source;
664*00b67f09SDavid van Moolenbroek 	isc_uint8_t c;
665*00b67f09SDavid van Moolenbroek 
666*00b67f09SDavid van Moolenbroek 	REQUIRE(type == 29);
667*00b67f09SDavid van Moolenbroek 	REQUIRE(source != NULL);
668*00b67f09SDavid van Moolenbroek 	REQUIRE(loc->common.rdtype == type);
669*00b67f09SDavid van Moolenbroek 	REQUIRE(loc->common.rdclass == rdclass);
670*00b67f09SDavid van Moolenbroek 
671*00b67f09SDavid van Moolenbroek 	UNUSED(type);
672*00b67f09SDavid van Moolenbroek 	UNUSED(rdclass);
673*00b67f09SDavid van Moolenbroek 
674*00b67f09SDavid van Moolenbroek 	if (loc->v.v0.version != 0)
675*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTIMPLEMENTED);
676*00b67f09SDavid van Moolenbroek 	RETERR(uint8_tobuffer(loc->v.v0.version, target));
677*00b67f09SDavid van Moolenbroek 
678*00b67f09SDavid van Moolenbroek 	c = loc->v.v0.size;
679*00b67f09SDavid van Moolenbroek 	if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
680*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
681*00b67f09SDavid van Moolenbroek 	RETERR(uint8_tobuffer(loc->v.v0.size, target));
682*00b67f09SDavid van Moolenbroek 
683*00b67f09SDavid van Moolenbroek 	c = loc->v.v0.horizontal;
684*00b67f09SDavid van Moolenbroek 	if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
685*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
686*00b67f09SDavid van Moolenbroek 	RETERR(uint8_tobuffer(loc->v.v0.horizontal, target));
687*00b67f09SDavid van Moolenbroek 
688*00b67f09SDavid van Moolenbroek 	c = loc->v.v0.vertical;
689*00b67f09SDavid van Moolenbroek 	if ((c&0xf) > 9 || ((c>>4)&0xf) > 9 || ((c>>4)&0xf) == 0)
690*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
691*00b67f09SDavid van Moolenbroek 	RETERR(uint8_tobuffer(loc->v.v0.vertical, target));
692*00b67f09SDavid van Moolenbroek 
693*00b67f09SDavid van Moolenbroek 	if (loc->v.v0.latitude < (0x80000000UL - 90 * 3600000) ||
694*00b67f09SDavid van Moolenbroek 	    loc->v.v0.latitude > (0x80000000UL + 90 * 3600000))
695*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
696*00b67f09SDavid van Moolenbroek 	RETERR(uint32_tobuffer(loc->v.v0.latitude, target));
697*00b67f09SDavid van Moolenbroek 
698*00b67f09SDavid van Moolenbroek 	if (loc->v.v0.longitude < (0x80000000UL - 180 * 3600000) ||
699*00b67f09SDavid van Moolenbroek 	    loc->v.v0.longitude > (0x80000000UL + 180 * 3600000))
700*00b67f09SDavid van Moolenbroek 		return (ISC_R_RANGE);
701*00b67f09SDavid van Moolenbroek 	RETERR(uint32_tobuffer(loc->v.v0.longitude, target));
702*00b67f09SDavid van Moolenbroek 	return (uint32_tobuffer(loc->v.v0.altitude, target));
703*00b67f09SDavid van Moolenbroek }
704*00b67f09SDavid van Moolenbroek 
705*00b67f09SDavid van Moolenbroek static inline isc_result_t
tostruct_loc(ARGS_TOSTRUCT)706*00b67f09SDavid van Moolenbroek tostruct_loc(ARGS_TOSTRUCT) {
707*00b67f09SDavid van Moolenbroek 	dns_rdata_loc_t *loc = target;
708*00b67f09SDavid van Moolenbroek 	isc_region_t r;
709*00b67f09SDavid van Moolenbroek 	isc_uint8_t version;
710*00b67f09SDavid van Moolenbroek 
711*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
712*00b67f09SDavid van Moolenbroek 	REQUIRE(target != NULL);
713*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->length != 0);
714*00b67f09SDavid van Moolenbroek 
715*00b67f09SDavid van Moolenbroek 	UNUSED(mctx);
716*00b67f09SDavid van Moolenbroek 
717*00b67f09SDavid van Moolenbroek 	dns_rdata_toregion(rdata, &r);
718*00b67f09SDavid van Moolenbroek 	version = uint8_fromregion(&r);
719*00b67f09SDavid van Moolenbroek 	if (version != 0)
720*00b67f09SDavid van Moolenbroek 		return (ISC_R_NOTIMPLEMENTED);
721*00b67f09SDavid van Moolenbroek 
722*00b67f09SDavid van Moolenbroek 	loc->common.rdclass = rdata->rdclass;
723*00b67f09SDavid van Moolenbroek 	loc->common.rdtype = rdata->type;
724*00b67f09SDavid van Moolenbroek 	ISC_LINK_INIT(&loc->common, link);
725*00b67f09SDavid van Moolenbroek 
726*00b67f09SDavid van Moolenbroek 	loc->v.v0.version = version;
727*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 1);
728*00b67f09SDavid van Moolenbroek 	loc->v.v0.size = uint8_fromregion(&r);
729*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 1);
730*00b67f09SDavid van Moolenbroek 	loc->v.v0.horizontal = uint8_fromregion(&r);
731*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 1);
732*00b67f09SDavid van Moolenbroek 	loc->v.v0.vertical = uint8_fromregion(&r);
733*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 1);
734*00b67f09SDavid van Moolenbroek 	loc->v.v0.latitude = uint32_fromregion(&r);
735*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 4);
736*00b67f09SDavid van Moolenbroek 	loc->v.v0.longitude = uint32_fromregion(&r);
737*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 4);
738*00b67f09SDavid van Moolenbroek 	loc->v.v0.altitude = uint32_fromregion(&r);
739*00b67f09SDavid van Moolenbroek 	isc_region_consume(&r, 4);
740*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
741*00b67f09SDavid van Moolenbroek }
742*00b67f09SDavid van Moolenbroek 
743*00b67f09SDavid van Moolenbroek static inline void
freestruct_loc(ARGS_FREESTRUCT)744*00b67f09SDavid van Moolenbroek freestruct_loc(ARGS_FREESTRUCT) {
745*00b67f09SDavid van Moolenbroek 	dns_rdata_loc_t *loc = source;
746*00b67f09SDavid van Moolenbroek 
747*00b67f09SDavid van Moolenbroek 	REQUIRE(source != NULL);
748*00b67f09SDavid van Moolenbroek 	REQUIRE(loc->common.rdtype == 29);
749*00b67f09SDavid van Moolenbroek 
750*00b67f09SDavid van Moolenbroek 	UNUSED(source);
751*00b67f09SDavid van Moolenbroek 	UNUSED(loc);
752*00b67f09SDavid van Moolenbroek }
753*00b67f09SDavid van Moolenbroek 
754*00b67f09SDavid van Moolenbroek static inline isc_result_t
additionaldata_loc(ARGS_ADDLDATA)755*00b67f09SDavid van Moolenbroek additionaldata_loc(ARGS_ADDLDATA) {
756*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
757*00b67f09SDavid van Moolenbroek 
758*00b67f09SDavid van Moolenbroek 	UNUSED(rdata);
759*00b67f09SDavid van Moolenbroek 	UNUSED(add);
760*00b67f09SDavid van Moolenbroek 	UNUSED(arg);
761*00b67f09SDavid van Moolenbroek 
762*00b67f09SDavid van Moolenbroek 	return (ISC_R_SUCCESS);
763*00b67f09SDavid van Moolenbroek }
764*00b67f09SDavid van Moolenbroek 
765*00b67f09SDavid van Moolenbroek static inline isc_result_t
digest_loc(ARGS_DIGEST)766*00b67f09SDavid van Moolenbroek digest_loc(ARGS_DIGEST) {
767*00b67f09SDavid van Moolenbroek 	isc_region_t r;
768*00b67f09SDavid van Moolenbroek 
769*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
770*00b67f09SDavid van Moolenbroek 
771*00b67f09SDavid van Moolenbroek 	dns_rdata_toregion(rdata, &r);
772*00b67f09SDavid van Moolenbroek 
773*00b67f09SDavid van Moolenbroek 	return ((digest)(arg, &r));
774*00b67f09SDavid van Moolenbroek }
775*00b67f09SDavid van Moolenbroek 
776*00b67f09SDavid van Moolenbroek static inline isc_boolean_t
checkowner_loc(ARGS_CHECKOWNER)777*00b67f09SDavid van Moolenbroek checkowner_loc(ARGS_CHECKOWNER) {
778*00b67f09SDavid van Moolenbroek 
779*00b67f09SDavid van Moolenbroek 	REQUIRE(type == 29);
780*00b67f09SDavid van Moolenbroek 
781*00b67f09SDavid van Moolenbroek 	UNUSED(name);
782*00b67f09SDavid van Moolenbroek 	UNUSED(type);
783*00b67f09SDavid van Moolenbroek 	UNUSED(rdclass);
784*00b67f09SDavid van Moolenbroek 	UNUSED(wildcard);
785*00b67f09SDavid van Moolenbroek 
786*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
787*00b67f09SDavid van Moolenbroek }
788*00b67f09SDavid van Moolenbroek 
789*00b67f09SDavid van Moolenbroek static inline isc_boolean_t
checknames_loc(ARGS_CHECKNAMES)790*00b67f09SDavid van Moolenbroek checknames_loc(ARGS_CHECKNAMES) {
791*00b67f09SDavid van Moolenbroek 
792*00b67f09SDavid van Moolenbroek 	REQUIRE(rdata->type == 29);
793*00b67f09SDavid van Moolenbroek 
794*00b67f09SDavid van Moolenbroek 	UNUSED(rdata);
795*00b67f09SDavid van Moolenbroek 	UNUSED(owner);
796*00b67f09SDavid van Moolenbroek 	UNUSED(bad);
797*00b67f09SDavid van Moolenbroek 
798*00b67f09SDavid van Moolenbroek 	return (ISC_TRUE);
799*00b67f09SDavid van Moolenbroek }
800*00b67f09SDavid van Moolenbroek 
801*00b67f09SDavid van Moolenbroek static inline int
casecompare_loc(ARGS_COMPARE)802*00b67f09SDavid van Moolenbroek casecompare_loc(ARGS_COMPARE) {
803*00b67f09SDavid van Moolenbroek 	return (compare_loc(rdata1, rdata2));
804*00b67f09SDavid van Moolenbroek }
805*00b67f09SDavid van Moolenbroek 
806*00b67f09SDavid van Moolenbroek #endif	/* RDATA_GENERIC_LOC_29_C */
807