res_mkquery.c
上传用户:nvosite88
上传日期:2007-01-17
资源大小:4983k
文件大小:7k
源码类别:

VxWorks

开发平台:

C/C++

  1. /* res_mkquery.c - creates DNS queries */
  2. /* Copyright 1984-2001 Wind River Systems, Inc. */
  3. /*
  4.  * Copyright (c) 1985 Regents of the University of California.
  5.  * All rights reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  * 1. Redistributions of source code must retain the above copyright
  11.  *    notice, this list of conditions and the following disclaimer.
  12.  * 2. Redistributions in binary form must reproduce the above copyright
  13.  *    notice, this list of conditions and the following disclaimer in the
  14.  *    documentation and/or other materials provided with the distribution.
  15.  * 3. All advertising materials mentioning features or use of this software
  16.  *    must display the following acknowledgement:
  17.  * This product includes software developed by the University of
  18.  * California, Berkeley and its contributors.
  19.  * 4. Neither the name of the University nor the names of its contributors
  20.  *    may be used to endorse or promote products derived from this software
  21.  *    without specific prior written permission.
  22.  *
  23.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  24.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  25.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  26.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  27.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  28.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  29.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  30.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  31.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  32.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  33.  * SUCH DAMAGE.
  34.  */
  35. /*
  36. modification history
  37. --------------------
  38. 01c,17sep01,vvv fixed compilation warning
  39. 01b,29apr97,jag Changed T_NULL to T_NULL_RR to fix conflict with loadCoffLib.h
  40. 01b,03apr97,jag Move the increment of _res.id outside of the htons macro.
  41. 01a,13dec96,jag Cleaned up and rename user I/F functions with the resolv
  42. prefix.  Man pages can be found in resolvLib.c
  43. */
  44. #include <resolvLib.h>
  45. #include <stdio.h>
  46. #include <string.h>
  47. /*
  48.  * Form all types of queries.
  49.  * Returns the size of the result or ERROR
  50.  */
  51. int 
  52. resolvMkQuery(op, dname, class, type, data, datalen, newrr_in, buf, buflen)
  53. int op; /* opcode of query */
  54. const char *dname; /* domain name */
  55. int class, type; /* class and type of query */
  56. const char *data; /* resource record data */
  57. int datalen; /* length of data */
  58. const char *newrr_in; /* new rr for modify or append */
  59. char *buf; /* buffer to put query */
  60. int buflen; /* size of buffer */
  61. {
  62. register HEADER *hp;
  63. register char *cp;
  64. register int n;
  65. char *dnptrs[10], **dpp, **lastdnptr;
  66. #ifdef DEBUG
  67. if (_res.options & RES_DEBUG)
  68. printf(";; resolvMkQuery(%d, %s, %d, %d)n", op, dname, class, type);
  69. #endif
  70. /*
  71.  * Initialize header fields.
  72.  */
  73. if ((buf == NULL) || (buflen < sizeof(HEADER)))
  74. return(ERROR);
  75. bzero(buf, sizeof(HEADER));
  76. hp = (HEADER *) buf;
  77. ++_res.id;
  78. hp->id = htons(_res.id);
  79. hp->opcode = op;
  80. hp->pr = (_res.options & RES_PRIMARY) != 0;
  81. hp->rd = (_res.options & RES_RECURSE) != 0;
  82. hp->rcode = NOERROR;
  83. cp = buf + sizeof(HEADER);
  84. buflen -= sizeof(HEADER);
  85. dpp = dnptrs;
  86. *dpp++ = buf;
  87. *dpp++ = NULL;
  88. lastdnptr = dnptrs + sizeof(dnptrs)/sizeof(dnptrs[0]);
  89. /*
  90.  * perform opcode specific processing
  91.  */
  92. switch (op) {
  93. case QUERY:
  94. if ((buflen -= QFIXEDSZ) < 0)
  95. return(ERROR);
  96. if ((n = resolvDNComp((u_char *)dname, (u_char *)cp, buflen,
  97.     (u_char **)dnptrs, (u_char **)lastdnptr)) < 0)
  98. return (ERROR);
  99. cp += n;
  100. buflen -= n;
  101. __putshort(type, (u_char *)cp);
  102. cp += sizeof(u_short);
  103. __putshort(class, (u_char *)cp);
  104. cp += sizeof(u_short);
  105. hp->qdcount = htons(1);
  106. if (op == QUERY || data == NULL)
  107. break;
  108. /*
  109.  * Make an additional record for completion domain.
  110.  */
  111. buflen -= RRFIXEDSZ;
  112. if ((n = resolvDNComp((u_char *)data, (u_char *)cp, buflen,
  113.     (u_char **)dnptrs, (u_char **)lastdnptr)) < 0)
  114. return (ERROR);
  115. cp += n;
  116. buflen -= n;
  117. __putshort(T_NULL_RR, (u_char *)cp);
  118. cp += sizeof(u_short);
  119. __putshort(class, (u_char *)cp);
  120. cp += sizeof(u_short);
  121. __putlong(0, (u_char *)cp);
  122. cp += sizeof(u_long);
  123. __putshort(0, (u_char *)cp);
  124. cp += sizeof(u_short);
  125. hp->arcount = htons(1);
  126. break;
  127. case IQUERY:
  128. /*
  129.  * Initialize answer section
  130.  */
  131. if (buflen < 1 + RRFIXEDSZ + datalen)
  132. return (ERROR);
  133. *cp++ = ''; /* no domain name */
  134. __putshort(type, (u_char *)cp);
  135. cp += sizeof(u_short);
  136. __putshort(class, (u_char *)cp);
  137. cp += sizeof(u_short);
  138. __putlong(0, (u_char *)cp);
  139. cp += sizeof(u_long);
  140. __putshort(datalen, (u_char *)cp);
  141. cp += sizeof(u_short);
  142. if (datalen) {
  143. bcopy(data, cp, datalen);
  144. cp += datalen;
  145. }
  146. hp->ancount = htons(1);
  147. break;
  148. #ifdef ALLOW_UPDATES
  149. /*
  150.  * For UPDATEM/UPDATEMA, do UPDATED/UPDATEDA followed by UPDATEA
  151.  * (Record to be modified is followed by its replacement in msg.)
  152.  */
  153. case UPDATEM:
  154. case UPDATEMA:
  155. case UPDATED:
  156. /*
  157.  * The res code for UPDATED and UPDATEDA is the same; user
  158.  * calls them differently: specifies data for UPDATED; server
  159.  * ignores data if specified for UPDATEDA.
  160.  */
  161. case UPDATEDA:
  162. buflen -= RRFIXEDSZ + datalen;
  163. if ((n = resolvDNComp(dname, cp, buflen, dnptrs, lastdnptr)) < 0)
  164. return (ERROR);
  165. cp += n;
  166. __putshort(type, cp);
  167.                 cp += sizeof(u_short);
  168.                 __putshort(class, cp);
  169.                 cp += sizeof(u_short);
  170. __putlong(0, cp);
  171. cp += sizeof(u_long);
  172. __putshort(datalen, cp);
  173.                 cp += sizeof(u_short);
  174. if (datalen) {
  175. bcopy(data, cp, datalen);
  176. cp += datalen;
  177. }
  178. if ( (op == UPDATED) || (op == UPDATEDA) ) {
  179. hp->ancount = htons(0);
  180. break;
  181. }
  182. /* Else UPDATEM/UPDATEMA, so drop into code for UPDATEA */
  183. case UPDATEA: /* Add new resource record */
  184. buflen -= RRFIXEDSZ + datalen;
  185. if ((n = resolvDNComp(dname, cp, buflen, dnptrs, lastdnptr)) < 0)
  186. return (ERROR);
  187. cp += n;
  188. __putshort(newrr->r_type, cp);
  189.                 cp += sizeof(u_short);
  190.                 __putshort(newrr->r_class, cp);
  191.                 cp += sizeof(u_short);
  192. __putlong(0, cp);
  193. cp += sizeof(u_long);
  194. __putshort(newrr->r_size, cp);
  195.                 cp += sizeof(u_short);
  196. if (newrr->r_size) {
  197. bcopy(newrr->r_data, cp, newrr->r_size);
  198. cp += newrr->r_size;
  199. }
  200. hp->ancount = htons(0);
  201. break;
  202. #endif /* ALLOW_UPDATES */
  203. }
  204. return (cp - buf);
  205. }