165e96449SHajimu UMEMOTO /* 265e96449SHajimu UMEMOTO * Copyright (c) 2004 by Internet Systems Consortium, Inc. ("ISC") 365e96449SHajimu UMEMOTO * Copyright (c) 1996,1999 by Internet Software Consortium. 465e96449SHajimu UMEMOTO * 565e96449SHajimu UMEMOTO * Permission to use, copy, modify, and distribute this software for any 665e96449SHajimu UMEMOTO * purpose with or without fee is hereby granted, provided that the above 765e96449SHajimu UMEMOTO * copyright notice and this permission notice appear in all copies. 865e96449SHajimu UMEMOTO * 965e96449SHajimu UMEMOTO * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES 1065e96449SHajimu UMEMOTO * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 1165e96449SHajimu UMEMOTO * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR 1265e96449SHajimu UMEMOTO * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 1365e96449SHajimu UMEMOTO * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 1465e96449SHajimu UMEMOTO * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT 1565e96449SHajimu UMEMOTO * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 1665e96449SHajimu UMEMOTO */ 1765e96449SHajimu UMEMOTO 1865e96449SHajimu UMEMOTO #ifndef lint 19*046c3635SPedro F. Giffuni static const char rcsid[] = "$Id: ns_parse.c,v 1.10 2009/01/23 19:59:16 each Exp $"; 2065e96449SHajimu UMEMOTO #endif 2165e96449SHajimu UMEMOTO 2265e96449SHajimu UMEMOTO /* Import. */ 2365e96449SHajimu UMEMOTO 2465e96449SHajimu UMEMOTO #include "port_before.h" 2565e96449SHajimu UMEMOTO 2665e96449SHajimu UMEMOTO #include <sys/types.h> 2765e96449SHajimu UMEMOTO 2865e96449SHajimu UMEMOTO #include <netinet/in.h> 2965e96449SHajimu UMEMOTO #include <arpa/nameser.h> 3065e96449SHajimu UMEMOTO 3165e96449SHajimu UMEMOTO #include <errno.h> 3265e96449SHajimu UMEMOTO #include <resolv.h> 3365e96449SHajimu UMEMOTO #include <string.h> 3465e96449SHajimu UMEMOTO 3565e96449SHajimu UMEMOTO #include "port_after.h" 3665e96449SHajimu UMEMOTO 3765e96449SHajimu UMEMOTO /* Forward. */ 3865e96449SHajimu UMEMOTO 3965e96449SHajimu UMEMOTO static void setsection(ns_msg *msg, ns_sect sect); 4065e96449SHajimu UMEMOTO 4165e96449SHajimu UMEMOTO /* Macros. */ 4265e96449SHajimu UMEMOTO 4341177159SHajimu UMEMOTO #if !defined(SOLARIS2) || defined(__COVERITY__) 4465e96449SHajimu UMEMOTO #define RETERR(err) do { errno = (err); return (-1); } while (0) 4565e96449SHajimu UMEMOTO #else 4665e96449SHajimu UMEMOTO #define RETERR(err) \ 4765e96449SHajimu UMEMOTO do { errno = (err); if (errno == errno) return (-1); } while (0) 4865e96449SHajimu UMEMOTO #endif 4965e96449SHajimu UMEMOTO 50*046c3635SPedro F. Giffuni #define PARSE_FMT_PRESO 0 /* Parse using presentation-format names */ 51*046c3635SPedro F. Giffuni #define PARSE_FMT_WIRE 1 /* Parse using network-format names */ 52*046c3635SPedro F. Giffuni 5365e96449SHajimu UMEMOTO /* Public. */ 5465e96449SHajimu UMEMOTO 5565e96449SHajimu UMEMOTO /* These need to be in the same order as the nres.h:ns_flag enum. */ 5665e96449SHajimu UMEMOTO struct _ns_flagdata _ns_flagdata[16] = { 57861249f5SHajimu UMEMOTO { 0x8000, 15 }, /*%< qr. */ 58861249f5SHajimu UMEMOTO { 0x7800, 11 }, /*%< opcode. */ 59861249f5SHajimu UMEMOTO { 0x0400, 10 }, /*%< aa. */ 60861249f5SHajimu UMEMOTO { 0x0200, 9 }, /*%< tc. */ 61861249f5SHajimu UMEMOTO { 0x0100, 8 }, /*%< rd. */ 62861249f5SHajimu UMEMOTO { 0x0080, 7 }, /*%< ra. */ 63861249f5SHajimu UMEMOTO { 0x0040, 6 }, /*%< z. */ 64861249f5SHajimu UMEMOTO { 0x0020, 5 }, /*%< ad. */ 65861249f5SHajimu UMEMOTO { 0x0010, 4 }, /*%< cd. */ 66861249f5SHajimu UMEMOTO { 0x000f, 0 }, /*%< rcode. */ 67861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (1/6). */ 68861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (2/6). */ 69861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (3/6). */ 70861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (4/6). */ 71861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (5/6). */ 72861249f5SHajimu UMEMOTO { 0x0000, 0 }, /*%< expansion (6/6). */ 7365e96449SHajimu UMEMOTO }; 7465e96449SHajimu UMEMOTO 7565e96449SHajimu UMEMOTO int ns_msg_getflag(ns_msg handle, int flag) { 7665e96449SHajimu UMEMOTO return(((handle)._flags & _ns_flagdata[flag].mask) >> _ns_flagdata[flag].shift); 7765e96449SHajimu UMEMOTO } 7865e96449SHajimu UMEMOTO 7965e96449SHajimu UMEMOTO int 8065e96449SHajimu UMEMOTO ns_skiprr(const u_char *ptr, const u_char *eom, ns_sect section, int count) { 8165e96449SHajimu UMEMOTO const u_char *optr = ptr; 8265e96449SHajimu UMEMOTO 8365e96449SHajimu UMEMOTO for ((void)NULL; count > 0; count--) { 8465e96449SHajimu UMEMOTO int b, rdlength; 8565e96449SHajimu UMEMOTO 8665e96449SHajimu UMEMOTO b = dn_skipname(ptr, eom); 8765e96449SHajimu UMEMOTO if (b < 0) 8865e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 8965e96449SHajimu UMEMOTO ptr += b/*Name*/ + NS_INT16SZ/*Type*/ + NS_INT16SZ/*Class*/; 9065e96449SHajimu UMEMOTO if (section != ns_s_qd) { 9165e96449SHajimu UMEMOTO if (ptr + NS_INT32SZ + NS_INT16SZ > eom) 9265e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 9365e96449SHajimu UMEMOTO ptr += NS_INT32SZ/*TTL*/; 9465e96449SHajimu UMEMOTO NS_GET16(rdlength, ptr); 9565e96449SHajimu UMEMOTO ptr += rdlength/*RData*/; 9665e96449SHajimu UMEMOTO } 9765e96449SHajimu UMEMOTO } 9865e96449SHajimu UMEMOTO if (ptr > eom) 9965e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 10065e96449SHajimu UMEMOTO return (ptr - optr); 10165e96449SHajimu UMEMOTO } 10265e96449SHajimu UMEMOTO 10365e96449SHajimu UMEMOTO int 10465e96449SHajimu UMEMOTO ns_initparse(const u_char *msg, int msglen, ns_msg *handle) { 10565e96449SHajimu UMEMOTO const u_char *eom = msg + msglen; 10665e96449SHajimu UMEMOTO int i; 10765e96449SHajimu UMEMOTO 10865e96449SHajimu UMEMOTO handle->_msg = msg; 10965e96449SHajimu UMEMOTO handle->_eom = eom; 11065e96449SHajimu UMEMOTO if (msg + NS_INT16SZ > eom) 11165e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 11265e96449SHajimu UMEMOTO NS_GET16(handle->_id, msg); 11365e96449SHajimu UMEMOTO if (msg + NS_INT16SZ > eom) 11465e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 11565e96449SHajimu UMEMOTO NS_GET16(handle->_flags, msg); 11665e96449SHajimu UMEMOTO for (i = 0; i < ns_s_max; i++) { 11765e96449SHajimu UMEMOTO if (msg + NS_INT16SZ > eom) 11865e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 11965e96449SHajimu UMEMOTO NS_GET16(handle->_counts[i], msg); 12065e96449SHajimu UMEMOTO } 12165e96449SHajimu UMEMOTO for (i = 0; i < ns_s_max; i++) 12265e96449SHajimu UMEMOTO if (handle->_counts[i] == 0) 12365e96449SHajimu UMEMOTO handle->_sections[i] = NULL; 12465e96449SHajimu UMEMOTO else { 12565e96449SHajimu UMEMOTO int b = ns_skiprr(msg, eom, (ns_sect)i, 12665e96449SHajimu UMEMOTO handle->_counts[i]); 12765e96449SHajimu UMEMOTO 12865e96449SHajimu UMEMOTO if (b < 0) 12965e96449SHajimu UMEMOTO return (-1); 13065e96449SHajimu UMEMOTO handle->_sections[i] = msg; 13165e96449SHajimu UMEMOTO msg += b; 13265e96449SHajimu UMEMOTO } 13365e96449SHajimu UMEMOTO if (msg != eom) 13465e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 13565e96449SHajimu UMEMOTO setsection(handle, ns_s_max); 13665e96449SHajimu UMEMOTO return (0); 13765e96449SHajimu UMEMOTO } 13865e96449SHajimu UMEMOTO 13965e96449SHajimu UMEMOTO int 14065e96449SHajimu UMEMOTO ns_parserr(ns_msg *handle, ns_sect section, int rrnum, ns_rr *rr) { 14165e96449SHajimu UMEMOTO int b; 14265e96449SHajimu UMEMOTO int tmp; 14365e96449SHajimu UMEMOTO 14465e96449SHajimu UMEMOTO /* Make section right. */ 14565e96449SHajimu UMEMOTO tmp = section; 14665e96449SHajimu UMEMOTO if (tmp < 0 || section >= ns_s_max) 14765e96449SHajimu UMEMOTO RETERR(ENODEV); 14865e96449SHajimu UMEMOTO if (section != handle->_sect) 14965e96449SHajimu UMEMOTO setsection(handle, section); 15065e96449SHajimu UMEMOTO 15165e96449SHajimu UMEMOTO /* Make rrnum right. */ 15265e96449SHajimu UMEMOTO if (rrnum == -1) 15365e96449SHajimu UMEMOTO rrnum = handle->_rrnum; 15465e96449SHajimu UMEMOTO if (rrnum < 0 || rrnum >= handle->_counts[(int)section]) 15565e96449SHajimu UMEMOTO RETERR(ENODEV); 15665e96449SHajimu UMEMOTO if (rrnum < handle->_rrnum) 15765e96449SHajimu UMEMOTO setsection(handle, section); 15865e96449SHajimu UMEMOTO if (rrnum > handle->_rrnum) { 15965e96449SHajimu UMEMOTO b = ns_skiprr(handle->_msg_ptr, handle->_eom, section, 16065e96449SHajimu UMEMOTO rrnum - handle->_rrnum); 16165e96449SHajimu UMEMOTO 16265e96449SHajimu UMEMOTO if (b < 0) 16365e96449SHajimu UMEMOTO return (-1); 16465e96449SHajimu UMEMOTO handle->_msg_ptr += b; 16565e96449SHajimu UMEMOTO handle->_rrnum = rrnum; 16665e96449SHajimu UMEMOTO } 16765e96449SHajimu UMEMOTO 16865e96449SHajimu UMEMOTO /* Do the parse. */ 16965e96449SHajimu UMEMOTO b = dn_expand(handle->_msg, handle->_eom, 17065e96449SHajimu UMEMOTO handle->_msg_ptr, rr->name, NS_MAXDNAME); 17165e96449SHajimu UMEMOTO if (b < 0) 17265e96449SHajimu UMEMOTO return (-1); 17365e96449SHajimu UMEMOTO handle->_msg_ptr += b; 17465e96449SHajimu UMEMOTO if (handle->_msg_ptr + NS_INT16SZ + NS_INT16SZ > handle->_eom) 17565e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 17665e96449SHajimu UMEMOTO NS_GET16(rr->type, handle->_msg_ptr); 17765e96449SHajimu UMEMOTO NS_GET16(rr->rr_class, handle->_msg_ptr); 17865e96449SHajimu UMEMOTO if (section == ns_s_qd) { 17965e96449SHajimu UMEMOTO rr->ttl = 0; 18065e96449SHajimu UMEMOTO rr->rdlength = 0; 18165e96449SHajimu UMEMOTO rr->rdata = NULL; 18265e96449SHajimu UMEMOTO } else { 18365e96449SHajimu UMEMOTO if (handle->_msg_ptr + NS_INT32SZ + NS_INT16SZ > handle->_eom) 18465e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 18565e96449SHajimu UMEMOTO NS_GET32(rr->ttl, handle->_msg_ptr); 18665e96449SHajimu UMEMOTO NS_GET16(rr->rdlength, handle->_msg_ptr); 18765e96449SHajimu UMEMOTO if (handle->_msg_ptr + rr->rdlength > handle->_eom) 18865e96449SHajimu UMEMOTO RETERR(EMSGSIZE); 18965e96449SHajimu UMEMOTO rr->rdata = handle->_msg_ptr; 19065e96449SHajimu UMEMOTO handle->_msg_ptr += rr->rdlength; 19165e96449SHajimu UMEMOTO } 19265e96449SHajimu UMEMOTO if (++handle->_rrnum > handle->_counts[(int)section]) 19365e96449SHajimu UMEMOTO setsection(handle, (ns_sect)((int)section + 1)); 19465e96449SHajimu UMEMOTO 19565e96449SHajimu UMEMOTO /* All done. */ 19665e96449SHajimu UMEMOTO return (0); 19765e96449SHajimu UMEMOTO } 19865e96449SHajimu UMEMOTO 199*046c3635SPedro F. Giffuni /* 200*046c3635SPedro F. Giffuni * This is identical to the above but uses network-format (uncompressed) names. 201*046c3635SPedro F. Giffuni */ 202*046c3635SPedro F. Giffuni int 203*046c3635SPedro F. Giffuni ns_parserr2(ns_msg *handle, ns_sect section, int rrnum, ns_rr2 *rr) { 204*046c3635SPedro F. Giffuni int b; 205*046c3635SPedro F. Giffuni int tmp; 206*046c3635SPedro F. Giffuni 207*046c3635SPedro F. Giffuni /* Make section right. */ 208*046c3635SPedro F. Giffuni if ((tmp = section) < 0 || section >= ns_s_max) 209*046c3635SPedro F. Giffuni RETERR(ENODEV); 210*046c3635SPedro F. Giffuni if (section != handle->_sect) 211*046c3635SPedro F. Giffuni setsection(handle, section); 212*046c3635SPedro F. Giffuni 213*046c3635SPedro F. Giffuni /* Make rrnum right. */ 214*046c3635SPedro F. Giffuni if (rrnum == -1) 215*046c3635SPedro F. Giffuni rrnum = handle->_rrnum; 216*046c3635SPedro F. Giffuni if (rrnum < 0 || rrnum >= handle->_counts[(int)section]) 217*046c3635SPedro F. Giffuni RETERR(ENODEV); 218*046c3635SPedro F. Giffuni if (rrnum < handle->_rrnum) 219*046c3635SPedro F. Giffuni setsection(handle, section); 220*046c3635SPedro F. Giffuni if (rrnum > handle->_rrnum) { 221*046c3635SPedro F. Giffuni b = ns_skiprr(handle->_msg_ptr, handle->_eom, section, 222*046c3635SPedro F. Giffuni rrnum - handle->_rrnum); 223*046c3635SPedro F. Giffuni 224*046c3635SPedro F. Giffuni if (b < 0) 225*046c3635SPedro F. Giffuni return (-1); 226*046c3635SPedro F. Giffuni handle->_msg_ptr += b; 227*046c3635SPedro F. Giffuni handle->_rrnum = rrnum; 228*046c3635SPedro F. Giffuni } 229*046c3635SPedro F. Giffuni 230*046c3635SPedro F. Giffuni /* Do the parse. */ 231*046c3635SPedro F. Giffuni b = ns_name_unpack2(handle->_msg, handle->_eom, handle->_msg_ptr, 232*046c3635SPedro F. Giffuni rr->nname, NS_MAXNNAME, &rr->nnamel); 233*046c3635SPedro F. Giffuni if (b < 0) 234*046c3635SPedro F. Giffuni return (-1); 235*046c3635SPedro F. Giffuni handle->_msg_ptr += b; 236*046c3635SPedro F. Giffuni if (handle->_msg_ptr + NS_INT16SZ + NS_INT16SZ > handle->_eom) 237*046c3635SPedro F. Giffuni RETERR(EMSGSIZE); 238*046c3635SPedro F. Giffuni NS_GET16(rr->type, handle->_msg_ptr); 239*046c3635SPedro F. Giffuni NS_GET16(rr->rr_class, handle->_msg_ptr); 240*046c3635SPedro F. Giffuni if (section == ns_s_qd) { 241*046c3635SPedro F. Giffuni rr->ttl = 0; 242*046c3635SPedro F. Giffuni rr->rdlength = 0; 243*046c3635SPedro F. Giffuni rr->rdata = NULL; 244*046c3635SPedro F. Giffuni } else { 245*046c3635SPedro F. Giffuni if (handle->_msg_ptr + NS_INT32SZ + NS_INT16SZ > handle->_eom) 246*046c3635SPedro F. Giffuni RETERR(EMSGSIZE); 247*046c3635SPedro F. Giffuni NS_GET32(rr->ttl, handle->_msg_ptr); 248*046c3635SPedro F. Giffuni NS_GET16(rr->rdlength, handle->_msg_ptr); 249*046c3635SPedro F. Giffuni if (handle->_msg_ptr + rr->rdlength > handle->_eom) 250*046c3635SPedro F. Giffuni RETERR(EMSGSIZE); 251*046c3635SPedro F. Giffuni rr->rdata = handle->_msg_ptr; 252*046c3635SPedro F. Giffuni handle->_msg_ptr += rr->rdlength; 253*046c3635SPedro F. Giffuni } 254*046c3635SPedro F. Giffuni if (++handle->_rrnum > handle->_counts[(int)section]) 255*046c3635SPedro F. Giffuni setsection(handle, (ns_sect)((int)section + 1)); 256*046c3635SPedro F. Giffuni 257*046c3635SPedro F. Giffuni /* All done. */ 258*046c3635SPedro F. Giffuni return (0); 259*046c3635SPedro F. Giffuni } 260*046c3635SPedro F. Giffuni 26165e96449SHajimu UMEMOTO /* Private. */ 26265e96449SHajimu UMEMOTO 26365e96449SHajimu UMEMOTO static void 26465e96449SHajimu UMEMOTO setsection(ns_msg *msg, ns_sect sect) { 26565e96449SHajimu UMEMOTO msg->_sect = sect; 26665e96449SHajimu UMEMOTO if (sect == ns_s_max) { 26765e96449SHajimu UMEMOTO msg->_rrnum = -1; 26865e96449SHajimu UMEMOTO msg->_msg_ptr = NULL; 26965e96449SHajimu UMEMOTO } else { 27065e96449SHajimu UMEMOTO msg->_rrnum = 0; 27165e96449SHajimu UMEMOTO msg->_msg_ptr = msg->_sections[(int)sect]; 27265e96449SHajimu UMEMOTO } 27365e96449SHajimu UMEMOTO } 274861249f5SHajimu UMEMOTO 275861249f5SHajimu UMEMOTO /*! \file */ 276