15185a700Sflorian /*
25185a700Sflorian * Copyright (C) Internet Systems Consortium, Inc. ("ISC")
35185a700Sflorian *
45185a700Sflorian * Permission to use, copy, modify, and/or distribute this software for any
55185a700Sflorian * purpose with or without fee is hereby granted, provided that the above
65185a700Sflorian * copyright notice and this permission notice appear in all copies.
75185a700Sflorian *
85185a700Sflorian * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
95185a700Sflorian * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
105185a700Sflorian * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
115185a700Sflorian * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
125185a700Sflorian * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
135185a700Sflorian * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
145185a700Sflorian * PERFORMANCE OF THIS SOFTWARE.
155185a700Sflorian */
165185a700Sflorian
17*1fb015a8Sflorian /* $Id: apl_42.c,v 1.14 2020/09/14 08:40:44 florian Exp $ */
185185a700Sflorian
195185a700Sflorian /* RFC3123 */
205185a700Sflorian
215185a700Sflorian #ifndef RDATA_IN_1_APL_42_C
225185a700Sflorian #define RDATA_IN_1_APL_42_C
235185a700Sflorian
245185a700Sflorian static inline isc_result_t
totext_in_apl(ARGS_TOTEXT)255185a700Sflorian totext_in_apl(ARGS_TOTEXT) {
265185a700Sflorian isc_region_t sr;
275185a700Sflorian isc_region_t ir;
285185a700Sflorian uint16_t afi;
295185a700Sflorian uint8_t prefix;
305185a700Sflorian uint8_t len;
31*1fb015a8Sflorian int neg;
325185a700Sflorian unsigned char buf[16];
335185a700Sflorian char txt[sizeof(" !64000:")];
345185a700Sflorian const char *sep = "";
355185a700Sflorian int n;
365185a700Sflorian
375185a700Sflorian REQUIRE(rdata->type == dns_rdatatype_apl);
385185a700Sflorian REQUIRE(rdata->rdclass == dns_rdataclass_in);
395185a700Sflorian
405185a700Sflorian UNUSED(tctx);
415185a700Sflorian
425185a700Sflorian dns_rdata_toregion(rdata, &sr);
435185a700Sflorian ir.base = buf;
445185a700Sflorian ir.length = sizeof(buf);
455185a700Sflorian
465185a700Sflorian while (sr.length > 0) {
475185a700Sflorian INSIST(sr.length >= 4);
485185a700Sflorian afi = uint16_fromregion(&sr);
495185a700Sflorian isc_region_consume(&sr, 2);
505185a700Sflorian prefix = *sr.base;
515185a700Sflorian isc_region_consume(&sr, 1);
525185a700Sflorian len = (*sr.base & 0x7f);
53*1fb015a8Sflorian neg = (*sr.base & 0x80) != 0;
545185a700Sflorian isc_region_consume(&sr, 1);
555185a700Sflorian INSIST(len <= sr.length);
565185a700Sflorian n = snprintf(txt, sizeof(txt), "%s%s%u:", sep,
575185a700Sflorian neg ? "!" : "", afi);
585185a700Sflorian INSIST(n < (int)sizeof(txt));
59873f12b9Sflorian RETERR(isc_str_tobuffer(txt, target));
605185a700Sflorian switch (afi) {
615185a700Sflorian case 1:
625185a700Sflorian INSIST(len <= 4);
635185a700Sflorian INSIST(prefix <= 32);
645185a700Sflorian memset(buf, 0, sizeof(buf));
655185a700Sflorian memmove(buf, sr.base, len);
665185a700Sflorian RETERR(inet_totext(AF_INET, &ir, target));
675185a700Sflorian break;
685185a700Sflorian
695185a700Sflorian case 2:
705185a700Sflorian INSIST(len <= 16);
715185a700Sflorian INSIST(prefix <= 128);
725185a700Sflorian memset(buf, 0, sizeof(buf));
735185a700Sflorian memmove(buf, sr.base, len);
745185a700Sflorian RETERR(inet_totext(AF_INET6, &ir, target));
755185a700Sflorian break;
765185a700Sflorian
775185a700Sflorian default:
785185a700Sflorian return (ISC_R_NOTIMPLEMENTED);
795185a700Sflorian }
805185a700Sflorian n = snprintf(txt, sizeof(txt), "/%u", prefix);
815185a700Sflorian INSIST(n < (int)sizeof(txt));
82873f12b9Sflorian RETERR(isc_str_tobuffer(txt, target));
835185a700Sflorian isc_region_consume(&sr, len);
845185a700Sflorian sep = " ";
855185a700Sflorian }
865185a700Sflorian return (ISC_R_SUCCESS);
875185a700Sflorian }
885185a700Sflorian
895185a700Sflorian static inline isc_result_t
fromwire_in_apl(ARGS_FROMWIRE)905185a700Sflorian fromwire_in_apl(ARGS_FROMWIRE) {
915185a700Sflorian isc_region_t sr, sr2;
925185a700Sflorian isc_region_t tr;
935185a700Sflorian uint16_t afi;
945185a700Sflorian uint8_t prefix;
955185a700Sflorian uint8_t len;
965185a700Sflorian
975185a700Sflorian REQUIRE(type == dns_rdatatype_apl);
985185a700Sflorian REQUIRE(rdclass == dns_rdataclass_in);
995185a700Sflorian
1005185a700Sflorian UNUSED(type);
1015185a700Sflorian UNUSED(dctx);
1025185a700Sflorian UNUSED(rdclass);
1035185a700Sflorian UNUSED(options);
1045185a700Sflorian
1055185a700Sflorian isc_buffer_activeregion(source, &sr);
1065185a700Sflorian isc_buffer_availableregion(target, &tr);
1075185a700Sflorian if (sr.length > tr.length)
1085185a700Sflorian return (ISC_R_NOSPACE);
1095185a700Sflorian sr2 = sr;
1105185a700Sflorian
1115185a700Sflorian /* Zero or more items */
1125185a700Sflorian while (sr.length > 0) {
1135185a700Sflorian if (sr.length < 4)
1145185a700Sflorian return (ISC_R_UNEXPECTEDEND);
1155185a700Sflorian afi = uint16_fromregion(&sr);
1165185a700Sflorian isc_region_consume(&sr, 2);
1175185a700Sflorian prefix = *sr.base;
1185185a700Sflorian isc_region_consume(&sr, 1);
1195185a700Sflorian len = (*sr.base & 0x7f);
1205185a700Sflorian isc_region_consume(&sr, 1);
1215185a700Sflorian if (len > sr.length)
1225185a700Sflorian return (ISC_R_UNEXPECTEDEND);
1235185a700Sflorian switch (afi) {
1245185a700Sflorian case 1:
1255185a700Sflorian if (prefix > 32 || len > 4)
1265185a700Sflorian return (ISC_R_RANGE);
1275185a700Sflorian break;
1285185a700Sflorian case 2:
1295185a700Sflorian if (prefix > 128 || len > 16)
1305185a700Sflorian return (ISC_R_RANGE);
1315185a700Sflorian }
1325185a700Sflorian if (len > 0 && sr.base[len - 1] == 0)
1335185a700Sflorian return (DNS_R_FORMERR);
1345185a700Sflorian isc_region_consume(&sr, len);
1355185a700Sflorian }
1365185a700Sflorian isc_buffer_forward(source, sr2.length);
137637d8eb6Sflorian return (isc_mem_tobuffer(target, sr2.base, sr2.length));
1385185a700Sflorian }
1395185a700Sflorian
1405185a700Sflorian static inline isc_result_t
towire_in_apl(ARGS_TOWIRE)1415185a700Sflorian towire_in_apl(ARGS_TOWIRE) {
1425185a700Sflorian UNUSED(cctx);
1435185a700Sflorian
1445185a700Sflorian REQUIRE(rdata->type == dns_rdatatype_apl);
1455185a700Sflorian REQUIRE(rdata->rdclass == dns_rdataclass_in);
1465185a700Sflorian
147637d8eb6Sflorian return (isc_mem_tobuffer(target, rdata->data, rdata->length));
1485185a700Sflorian }
1495185a700Sflorian
1505185a700Sflorian #endif /* RDATA_IN_1_APL_42_C */
151