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