1*9525b14bSRao Shoaib /* 2*9525b14bSRao Shoaib * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 3*9525b14bSRao Shoaib * Use is subject to license terms. 4*9525b14bSRao Shoaib */ 5*9525b14bSRao Shoaib 6*9525b14bSRao Shoaib 7*9525b14bSRao Shoaib /* 8*9525b14bSRao Shoaib * Copyright (C) 2009 Internet Systems Consortium, Inc. ("ISC") 9*9525b14bSRao Shoaib * 10*9525b14bSRao Shoaib * Permission to use, copy, modify, and/or distribute this software for any 11*9525b14bSRao Shoaib * purpose with or without fee is hereby granted, provided that the above 12*9525b14bSRao Shoaib * copyright notice and this permission notice appear in all copies. 13*9525b14bSRao Shoaib * 14*9525b14bSRao Shoaib * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH 15*9525b14bSRao Shoaib * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY 16*9525b14bSRao Shoaib * AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT, 17*9525b14bSRao Shoaib * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM 18*9525b14bSRao Shoaib * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE 19*9525b14bSRao Shoaib * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR 20*9525b14bSRao Shoaib * PERFORMANCE OF THIS SOFTWARE. 21*9525b14bSRao Shoaib */ 22*9525b14bSRao Shoaib 23*9525b14bSRao Shoaib #ifndef lint 24*9525b14bSRao Shoaib static const char rcsid[] = "$Id: ns_rdata.c,v 1.2 2009/01/23 23:49:15 tbox Exp $"; 25*9525b14bSRao Shoaib #endif 26*9525b14bSRao Shoaib 27*9525b14bSRao Shoaib #include "port_before.h" 28*9525b14bSRao Shoaib 29*9525b14bSRao Shoaib #if __OpenBSD__ 30*9525b14bSRao Shoaib #include <sys/types.h> 31*9525b14bSRao Shoaib #endif 32*9525b14bSRao Shoaib #include <netinet/in.h> 33*9525b14bSRao Shoaib #include <arpa/nameser.h> 34*9525b14bSRao Shoaib 35*9525b14bSRao Shoaib #include <errno.h> 36*9525b14bSRao Shoaib #include <resolv.h> 37*9525b14bSRao Shoaib #include <string.h> 38*9525b14bSRao Shoaib 39*9525b14bSRao Shoaib #include "port_after.h" 40*9525b14bSRao Shoaib 41*9525b14bSRao Shoaib #define CONSUME_SRC \ 42*9525b14bSRao Shoaib do { \ 43*9525b14bSRao Shoaib rdata += n, rdlen -= n; \ 44*9525b14bSRao Shoaib } while (0) 45*9525b14bSRao Shoaib 46*9525b14bSRao Shoaib #define CONSUME_DST \ 47*9525b14bSRao Shoaib do { \ 48*9525b14bSRao Shoaib nrdata += n, nrdsiz -= n, nrdlen += n; \ 49*9525b14bSRao Shoaib } while (0) 50*9525b14bSRao Shoaib 51*9525b14bSRao Shoaib #define UNPACK_DNAME \ 52*9525b14bSRao Shoaib do { \ 53*9525b14bSRao Shoaib size_t t; \ 54*9525b14bSRao Shoaib \ 55*9525b14bSRao Shoaib if ((n = ns_name_unpack2(msg,eom,rdata,nrdata,nrdsiz,&t))<0) {\ 56*9525b14bSRao Shoaib errno = EMSGSIZE; \ 57*9525b14bSRao Shoaib return (-1); \ 58*9525b14bSRao Shoaib } \ 59*9525b14bSRao Shoaib CONSUME_SRC; \ 60*9525b14bSRao Shoaib n = t; \ 61*9525b14bSRao Shoaib CONSUME_DST; \ 62*9525b14bSRao Shoaib } while (0) 63*9525b14bSRao Shoaib 64*9525b14bSRao Shoaib #define UNPACK_SOME(x) \ 65*9525b14bSRao Shoaib do { \ 66*9525b14bSRao Shoaib n = (x); \ 67*9525b14bSRao Shoaib if ((size_t)n > rdlen || (size_t)n > nrdsiz) { \ 68*9525b14bSRao Shoaib errno = EMSGSIZE; \ 69*9525b14bSRao Shoaib return (-1); \ 70*9525b14bSRao Shoaib } \ 71*9525b14bSRao Shoaib memcpy(nrdata, rdata, n); \ 72*9525b14bSRao Shoaib CONSUME_SRC; CONSUME_DST; \ 73*9525b14bSRao Shoaib } while (0) 74*9525b14bSRao Shoaib 75*9525b14bSRao Shoaib #define UNPACK_REST(x) \ 76*9525b14bSRao Shoaib do { \ 77*9525b14bSRao Shoaib n = (x); \ 78*9525b14bSRao Shoaib if ((size_t)n != rdlen) { \ 79*9525b14bSRao Shoaib errno = EMSGSIZE; \ 80*9525b14bSRao Shoaib return (-1); \ 81*9525b14bSRao Shoaib } \ 82*9525b14bSRao Shoaib memcpy(nrdata, rdata, n); \ 83*9525b14bSRao Shoaib CONSUME_SRC; CONSUME_DST; \ 84*9525b14bSRao Shoaib } while (0) 85*9525b14bSRao Shoaib 86*9525b14bSRao Shoaib ssize_t 87*9525b14bSRao Shoaib ns_rdata_unpack(const u_char *msg, const u_char *eom, 88*9525b14bSRao Shoaib ns_type type, const u_char *rdata, size_t rdlen, 89*9525b14bSRao Shoaib u_char *nrdata, size_t nrdsiz) 90*9525b14bSRao Shoaib { 91*9525b14bSRao Shoaib size_t nrdlen = 0; 92*9525b14bSRao Shoaib int n; 93*9525b14bSRao Shoaib 94*9525b14bSRao Shoaib switch (type) { 95*9525b14bSRao Shoaib case ns_t_a: 96*9525b14bSRao Shoaib UNPACK_REST(NS_INADDRSZ); 97*9525b14bSRao Shoaib break; 98*9525b14bSRao Shoaib case ns_t_aaaa: 99*9525b14bSRao Shoaib UNPACK_REST(NS_IN6ADDRSZ); 100*9525b14bSRao Shoaib break; 101*9525b14bSRao Shoaib case ns_t_cname: 102*9525b14bSRao Shoaib case ns_t_mb: 103*9525b14bSRao Shoaib case ns_t_mg: 104*9525b14bSRao Shoaib case ns_t_mr: 105*9525b14bSRao Shoaib case ns_t_ns: 106*9525b14bSRao Shoaib case ns_t_ptr: 107*9525b14bSRao Shoaib case ns_t_dname: 108*9525b14bSRao Shoaib UNPACK_DNAME; 109*9525b14bSRao Shoaib break; 110*9525b14bSRao Shoaib case ns_t_soa: 111*9525b14bSRao Shoaib UNPACK_DNAME; 112*9525b14bSRao Shoaib UNPACK_DNAME; 113*9525b14bSRao Shoaib UNPACK_SOME(NS_INT32SZ * 5); 114*9525b14bSRao Shoaib break; 115*9525b14bSRao Shoaib case ns_t_mx: 116*9525b14bSRao Shoaib case ns_t_afsdb: 117*9525b14bSRao Shoaib case ns_t_rt: 118*9525b14bSRao Shoaib UNPACK_SOME(NS_INT16SZ); 119*9525b14bSRao Shoaib UNPACK_DNAME; 120*9525b14bSRao Shoaib break; 121*9525b14bSRao Shoaib case ns_t_px: 122*9525b14bSRao Shoaib UNPACK_SOME(NS_INT16SZ); 123*9525b14bSRao Shoaib UNPACK_DNAME; 124*9525b14bSRao Shoaib UNPACK_DNAME; 125*9525b14bSRao Shoaib break; 126*9525b14bSRao Shoaib case ns_t_srv: 127*9525b14bSRao Shoaib UNPACK_SOME(NS_INT16SZ * 3); 128*9525b14bSRao Shoaib UNPACK_DNAME; 129*9525b14bSRao Shoaib break; 130*9525b14bSRao Shoaib case ns_t_minfo: 131*9525b14bSRao Shoaib case ns_t_rp: 132*9525b14bSRao Shoaib UNPACK_DNAME; 133*9525b14bSRao Shoaib UNPACK_DNAME; 134*9525b14bSRao Shoaib break; 135*9525b14bSRao Shoaib default: 136*9525b14bSRao Shoaib UNPACK_SOME(rdlen); 137*9525b14bSRao Shoaib break; 138*9525b14bSRao Shoaib } 139*9525b14bSRao Shoaib if (rdlen > 0) { 140*9525b14bSRao Shoaib errno = EMSGSIZE; 141*9525b14bSRao Shoaib return (-1); 142*9525b14bSRao Shoaib } 143*9525b14bSRao Shoaib return (nrdlen); 144*9525b14bSRao Shoaib } 145*9525b14bSRao Shoaib 146*9525b14bSRao Shoaib #define EQUAL_CONSUME \ 147*9525b14bSRao Shoaib do { \ 148*9525b14bSRao Shoaib rdata1 += n, rdlen1 -= n; \ 149*9525b14bSRao Shoaib rdata2 += n, rdlen2 -= n; \ 150*9525b14bSRao Shoaib } while (0) 151*9525b14bSRao Shoaib 152*9525b14bSRao Shoaib #define EQUAL_DNAME \ 153*9525b14bSRao Shoaib do { \ 154*9525b14bSRao Shoaib ssize_t n; \ 155*9525b14bSRao Shoaib \ 156*9525b14bSRao Shoaib if (rdlen1 != rdlen2) \ 157*9525b14bSRao Shoaib return (0); \ 158*9525b14bSRao Shoaib n = ns_name_eq(rdata1, rdlen1, rdata2, rdlen2); \ 159*9525b14bSRao Shoaib if (n <= 0) \ 160*9525b14bSRao Shoaib return (n); \ 161*9525b14bSRao Shoaib n = rdlen1; \ 162*9525b14bSRao Shoaib EQUAL_CONSUME; \ 163*9525b14bSRao Shoaib } while (0) 164*9525b14bSRao Shoaib 165*9525b14bSRao Shoaib #define EQUAL_SOME(x) \ 166*9525b14bSRao Shoaib do { \ 167*9525b14bSRao Shoaib size_t n = (x); \ 168*9525b14bSRao Shoaib \ 169*9525b14bSRao Shoaib if (n > rdlen1 || n > rdlen2) { \ 170*9525b14bSRao Shoaib errno = EMSGSIZE; \ 171*9525b14bSRao Shoaib return (-1); \ 172*9525b14bSRao Shoaib } \ 173*9525b14bSRao Shoaib if (memcmp(rdata1, rdata2, n) != 0) \ 174*9525b14bSRao Shoaib return (0); \ 175*9525b14bSRao Shoaib EQUAL_CONSUME; \ 176*9525b14bSRao Shoaib } while (0) 177*9525b14bSRao Shoaib 178*9525b14bSRao Shoaib int 179*9525b14bSRao Shoaib ns_rdata_equal(ns_type type, 180*9525b14bSRao Shoaib const u_char *rdata1, size_t rdlen1, 181*9525b14bSRao Shoaib const u_char *rdata2, size_t rdlen2) 182*9525b14bSRao Shoaib { 183*9525b14bSRao Shoaib switch (type) { 184*9525b14bSRao Shoaib case ns_t_cname: 185*9525b14bSRao Shoaib case ns_t_mb: 186*9525b14bSRao Shoaib case ns_t_mg: 187*9525b14bSRao Shoaib case ns_t_mr: 188*9525b14bSRao Shoaib case ns_t_ns: 189*9525b14bSRao Shoaib case ns_t_ptr: 190*9525b14bSRao Shoaib case ns_t_dname: 191*9525b14bSRao Shoaib EQUAL_DNAME; 192*9525b14bSRao Shoaib break; 193*9525b14bSRao Shoaib case ns_t_soa: 194*9525b14bSRao Shoaib /* "There can be only one." --Highlander */ 195*9525b14bSRao Shoaib break; 196*9525b14bSRao Shoaib case ns_t_mx: 197*9525b14bSRao Shoaib case ns_t_afsdb: 198*9525b14bSRao Shoaib case ns_t_rt: 199*9525b14bSRao Shoaib EQUAL_SOME(NS_INT16SZ); 200*9525b14bSRao Shoaib EQUAL_DNAME; 201*9525b14bSRao Shoaib break; 202*9525b14bSRao Shoaib case ns_t_px: 203*9525b14bSRao Shoaib EQUAL_SOME(NS_INT16SZ); 204*9525b14bSRao Shoaib EQUAL_DNAME; 205*9525b14bSRao Shoaib EQUAL_DNAME; 206*9525b14bSRao Shoaib break; 207*9525b14bSRao Shoaib case ns_t_srv: 208*9525b14bSRao Shoaib EQUAL_SOME(NS_INT16SZ * 3); 209*9525b14bSRao Shoaib EQUAL_DNAME; 210*9525b14bSRao Shoaib break; 211*9525b14bSRao Shoaib case ns_t_minfo: 212*9525b14bSRao Shoaib case ns_t_rp: 213*9525b14bSRao Shoaib EQUAL_DNAME; 214*9525b14bSRao Shoaib EQUAL_DNAME; 215*9525b14bSRao Shoaib break; 216*9525b14bSRao Shoaib default: 217*9525b14bSRao Shoaib EQUAL_SOME(rdlen1); 218*9525b14bSRao Shoaib break; 219*9525b14bSRao Shoaib } 220*9525b14bSRao Shoaib if (rdlen1 != 0 || rdlen2 != 0) 221*9525b14bSRao Shoaib return (0); 222*9525b14bSRao Shoaib return (1); 223*9525b14bSRao Shoaib } 224*9525b14bSRao Shoaib 225*9525b14bSRao Shoaib #define REFERS_DNAME \ 226*9525b14bSRao Shoaib do { \ 227*9525b14bSRao Shoaib int n; \ 228*9525b14bSRao Shoaib \ 229*9525b14bSRao Shoaib n = ns_name_eq(rdata, rdlen, nname, NS_MAXNNAME); \ 230*9525b14bSRao Shoaib if (n < 0) \ 231*9525b14bSRao Shoaib return (-1); \ 232*9525b14bSRao Shoaib if (n > 0) \ 233*9525b14bSRao Shoaib return (1); \ 234*9525b14bSRao Shoaib n = dn_skipname(rdata, rdata + rdlen); \ 235*9525b14bSRao Shoaib if (n < 0) \ 236*9525b14bSRao Shoaib return (-1); \ 237*9525b14bSRao Shoaib CONSUME_SRC; \ 238*9525b14bSRao Shoaib } while (0) 239*9525b14bSRao Shoaib 240*9525b14bSRao Shoaib #define REFERS_SOME(x) \ 241*9525b14bSRao Shoaib do { \ 242*9525b14bSRao Shoaib size_t n = (x); \ 243*9525b14bSRao Shoaib \ 244*9525b14bSRao Shoaib if (n > rdlen) { \ 245*9525b14bSRao Shoaib errno = EMSGSIZE; \ 246*9525b14bSRao Shoaib return (-1); \ 247*9525b14bSRao Shoaib } \ 248*9525b14bSRao Shoaib CONSUME_SRC; \ 249*9525b14bSRao Shoaib } while (0) 250*9525b14bSRao Shoaib 251*9525b14bSRao Shoaib int 252*9525b14bSRao Shoaib ns_rdata_refers(ns_type type, 253*9525b14bSRao Shoaib const u_char *rdata, size_t rdlen, 254*9525b14bSRao Shoaib const u_char *nname) 255*9525b14bSRao Shoaib { 256*9525b14bSRao Shoaib switch (type) { 257*9525b14bSRao Shoaib case ns_t_cname: 258*9525b14bSRao Shoaib case ns_t_mb: 259*9525b14bSRao Shoaib case ns_t_mg: 260*9525b14bSRao Shoaib case ns_t_mr: 261*9525b14bSRao Shoaib case ns_t_ns: 262*9525b14bSRao Shoaib case ns_t_ptr: 263*9525b14bSRao Shoaib case ns_t_dname: 264*9525b14bSRao Shoaib REFERS_DNAME; 265*9525b14bSRao Shoaib break; 266*9525b14bSRao Shoaib case ns_t_soa: 267*9525b14bSRao Shoaib REFERS_DNAME; 268*9525b14bSRao Shoaib REFERS_DNAME; 269*9525b14bSRao Shoaib REFERS_SOME(NS_INT32SZ * 5); 270*9525b14bSRao Shoaib break; 271*9525b14bSRao Shoaib case ns_t_mx: 272*9525b14bSRao Shoaib case ns_t_afsdb: 273*9525b14bSRao Shoaib case ns_t_rt: 274*9525b14bSRao Shoaib REFERS_SOME(NS_INT16SZ); 275*9525b14bSRao Shoaib REFERS_DNAME; 276*9525b14bSRao Shoaib break; 277*9525b14bSRao Shoaib case ns_t_px: 278*9525b14bSRao Shoaib REFERS_SOME(NS_INT16SZ); 279*9525b14bSRao Shoaib REFERS_DNAME; 280*9525b14bSRao Shoaib REFERS_DNAME; 281*9525b14bSRao Shoaib break; 282*9525b14bSRao Shoaib case ns_t_srv: 283*9525b14bSRao Shoaib REFERS_SOME(NS_INT16SZ * 3); 284*9525b14bSRao Shoaib REFERS_DNAME; 285*9525b14bSRao Shoaib break; 286*9525b14bSRao Shoaib case ns_t_minfo: 287*9525b14bSRao Shoaib case ns_t_rp: 288*9525b14bSRao Shoaib REFERS_DNAME; 289*9525b14bSRao Shoaib REFERS_DNAME; 290*9525b14bSRao Shoaib break; 291*9525b14bSRao Shoaib default: 292*9525b14bSRao Shoaib REFERS_SOME(rdlen); 293*9525b14bSRao Shoaib break; 294*9525b14bSRao Shoaib } 295*9525b14bSRao Shoaib if (rdlen != 0) { 296*9525b14bSRao Shoaib errno = EMSGSIZE; 297*9525b14bSRao Shoaib return (-1); 298*9525b14bSRao Shoaib } 299*9525b14bSRao Shoaib return (0); 300*9525b14bSRao Shoaib } 301