res_send.c
上传用户:baixin
上传日期:2008-03-13
资源大小:4795k
文件大小:12k
开发平台:

MultiPlatform

  1. /* res_send.c - */
  2. /* Copyright 1984-2001 Wind River Systems, Inc. */
  3. #include "copyright_wrs.h"
  4. /*
  5.  * Copyright (c) 1985, 1989 Regents of the University of California.
  6.  * All rights reserved.
  7.  *
  8.  * Redistribution and use in source and binary forms, with or without
  9.  * modification, are permitted provided that the following conditions
  10.  * are met:
  11.  * 1. Redistributions of source code must retain the above copyright
  12.  *    notice, this list of conditions and the following disclaimer.
  13.  * 2. Redistributions in binary form must reproduce the above copyright
  14.  *    notice, this list of conditions and the following disclaimer in the
  15.  *    documentation and/or other materials provided with the distribution.
  16.  * 3. All advertising materials mentioning features or use of this software
  17.  *    must display the following acknowledgement:
  18.  * This product includes software developed by the University of
  19.  * California, Berkeley and its contributors.
  20.  * 4. Neither the name of the University nor the names of its contributors
  21.  *    may be used to endorse or promote products derived from this software
  22.  *    without specific prior written permission.
  23.  *
  24.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  25.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  26.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  27.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  28.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  29.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  30.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  31.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  32.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  33.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  34.  * SUCH DAMAGE.
  35.  */
  36. /*
  37. modification history
  38. --------------------
  39. 01c,05nov01,vvv fixed compilation warning; cleaned up use of register
  40. 01b,04sep01,vvv fixed to correctly query multiple servers (SPR #67238);
  41.                 fixed compilation warnings
  42. 01a,13dec96,jag Cleaned up.  Added resolv prefix. Man pages can be found in
  43.                 files resolvLib.c.  Added code for pdnsDebugFunc.
  44. */
  45. /*
  46.  * Send query to name server and wait for reply.
  47.  */
  48. #include <sys/times.h>
  49. #include <resolvLib.h>
  50. #include <sockLib.h>
  51. #include <net/uio.h>
  52. #include <unistd.h>
  53. #include <string.h>
  54. #include "ioLib.h"
  55. extern FUNCPTR pdnsDebugFunc;
  56. #ifndef FD_SET
  57. #define NFDBITS 32
  58. #define FD_SETSIZE 32
  59. #define FD_SET(n, p) ((p)->fds_bits[(n)/NFDBITS] |= (1 << ((n) % NFDBITS)))
  60. #define FD_CLR(n, p) ((p)->fds_bits[(n)/NFDBITS] &= ~(1 << ((n) % NFDBITS)))
  61. #define FD_ISSET(n, p) ((p)->fds_bits[(n)/NFDBITS] & (1 << ((n) % NFDBITS)))
  62. #define FD_ZERO(p) bzero((char *)(p), sizeof(*(p)))
  63. #endif
  64. int 
  65. resolvSend(buf, buflen, answer, anslen)
  66. const char *buf;
  67. int buflen;
  68. char *answer;
  69. int anslen;
  70. {
  71. int n = 0;
  72. int try, v_circuit, resplen, ns;
  73. int gotsomewhere = 0, connected = 0;
  74. int connreset = 0;
  75. u_short id, len;
  76. char *cp;
  77. fd_set dsmask;
  78. struct timeval timeout;
  79. HEADER *hp = (HEADER *) buf;
  80. HEADER *anhp = (HEADER *) answer;
  81. struct iovec iov[2];
  82. int terrno = ETIMEDOUT;
  83. char junk[512];
  84. int start;      /* pointer to server we start querying with */
  85.         int s = -1; /* socket used for communications */
  86.         struct sockaddr no_addr;
  87.         bzero ((char*) &no_addr, sizeof no_addr);
  88. #ifdef ORG_RESOLVER
  89. #ifdef DEBUG
  90. if (_res.options & RES_DEBUG) {
  91. printf("resSend()n");
  92. __p_query(buf);
  93. }
  94. #endif /* DEBUG */
  95. #else
  96. if (pdnsDebugFunc != (FUNCPTR) NULL)
  97.     {
  98.     printf("resolvSend()n");
  99.     (*pdnsDebugFunc)(buf);
  100.     }
  101. #endif /* ORG_RESOLVER */
  102. #ifdef ORG_RESOLVER
  103. /* Unecessary test under VxWorks */
  104. if (!(_res.options & RES_INIT))
  105. if (res_init() == -1) {
  106. return(ERROR);
  107. }
  108. #endif /* ORG_RESOLVER */
  109. v_circuit = (_res.options & RES_USEVC) || buflen > PACKETSZ;
  110. id = hp->id;
  111. /*
  112.  * Send request, RETRY times, or until successful
  113.  */
  114. for (try = 0; try < _res.retry; try++) {
  115.    start = MAXDNSLUS;
  116.    for (ns = 0; ns < _res.nscount; ns++) {
  117. /*
  118.  * Only query servers marked active or appearing after
  119.  * a server marked as active. This prevents this loop
  120.  * from rescanning the entire array every time resolvSend
  121.            * is called
  122.  */
  123. if ((_res.lookups [ns] != 'a') && (ns < start))
  124.     continue;
  125. if (ns > start)
  126.     _res.lookups [ns] = 'd';  /* mark as 'done' */
  127. start = ns;
  128. #ifdef DEBUG
  129. if (_res.options & RES_DEBUG)
  130. printf("Querying server (# %d) address = %sn", ns+1,
  131.       inet_ntoa(_res.nsaddr_list[ns].sin_addr));
  132. #endif /* DEBUG */
  133. usevc:
  134. if (v_circuit) {
  135. int truncated = 0;
  136. /*
  137.  * Use virtual circuit;
  138.  * at most one attempt per server.
  139.  */
  140. try = _res.retry;
  141. if (s < 0) {
  142. s = socket(AF_INET, SOCK_STREAM, 0);
  143. if (s < 0) {
  144. terrno = errno;
  145. #ifdef DEBUG
  146. if (_res.options & RES_DEBUG)
  147.     perror("socket (vc) failed");
  148. #endif /* DEBUG */
  149. continue;
  150. }
  151. if (connect(s,
  152.     (struct sockaddr *)&(_res.nsaddr_list[ns]),
  153.     sizeof(struct sockaddr)) < 0) {
  154. terrno = errno;
  155. #ifdef DEBUG
  156. if (_res.options & RES_DEBUG)
  157.     perror("connect failed");
  158. #endif /* DEBUG */
  159. (void) close(s);
  160. s = -1;
  161. continue;
  162. }
  163. }
  164. /*
  165.  * Send length & message
  166.  */
  167. len = htons((u_short)buflen);
  168. iov[0].iov_base = (caddr_t)&len;
  169. iov[0].iov_len = sizeof(len);
  170. iov[1].iov_base = (char *)buf;
  171. iov[1].iov_len = buflen;
  172. if (writev(s, iov, 2) != sizeof(len) + buflen) {
  173. terrno = errno;
  174. #ifdef DEBUG
  175. if (_res.options & RES_DEBUG)
  176. perror("write failed");
  177. #endif /* DEBUG */
  178. (void) close(s);
  179. s = -1;
  180. continue;
  181. }
  182. /*
  183.  * Receive length & response
  184.  */
  185. cp = answer;
  186. len = sizeof(short);
  187. while (len != 0 &&
  188.     (n = read(s, (char *)cp, (int)len)) > 0) {
  189. cp += n;
  190. len -= n;
  191. }
  192. if (n <= 0) {
  193. terrno = errno;
  194. #ifdef DEBUG
  195. if (_res.options & RES_DEBUG)
  196. perror("read failed");
  197. #endif /* DEBUG */
  198. (void) close(s);
  199. s = -1;
  200. /*
  201.  * A long running process might get its TCP
  202.  * connection reset if the remote server was
  203.  * restarted.  Requery the server instead of
  204.  * trying a new one.  When there is only one
  205.  * server, this means that a query might work
  206.  * instead of failing.  We only allow one reset
  207.  * per query to prevent looping.
  208.  */
  209. if (terrno == ECONNRESET && !connreset) {
  210. connreset = 1;
  211. ns--;
  212. }
  213. continue;
  214. }
  215. cp = answer;
  216. if ((resplen = ntohs(*(u_short *)cp)) > anslen) {
  217. #ifdef DEBUG
  218. if (_res.options & RES_DEBUG)
  219. fprintf(stderr, "response truncatedn");
  220. #endif /* DEBUG */
  221. len = anslen;
  222. truncated = 1;
  223. } else
  224. len = resplen;
  225. while (len != 0 &&
  226.    (n = read(s, (char *)cp, (int)len)) > 0) {
  227. cp += n;
  228. len -= n;
  229. }
  230. if (n <= 0) {
  231. terrno = errno;
  232. #ifdef DEBUG
  233. if (_res.options & RES_DEBUG)
  234. perror("read failed");
  235. #endif /* DEBUG */
  236. (void) close(s);
  237. s = -1;
  238. continue;
  239. }
  240. if (truncated) {
  241. /*
  242.  * Flush rest of answer
  243.  * so connection stays in synch.
  244.  */
  245. anhp->tc = 1;
  246. len = resplen - anslen;
  247. while (len != 0) {
  248. n = (len > sizeof(junk) ?
  249.     sizeof(junk) : len);
  250. if ((n = read(s, junk, n)) > 0)
  251. len -= n;
  252. else
  253. break;
  254. }
  255. }
  256. } else {
  257. /*
  258.  * Use datagrams.
  259.  */
  260. if (s < 0) {
  261. s = socket(AF_INET, SOCK_DGRAM, 0);
  262. if (s < 0) {
  263. terrno = errno;
  264. #ifdef DEBUG
  265. if (_res.options & RES_DEBUG)
  266.     perror("socket (dg) failed");
  267. #endif /* DEBUG */
  268. continue;
  269. }
  270. }
  271. #if BSD >= 43
  272. /*
  273.  * I'm tired of answering this question, so:
  274.  * On a 4.3BSD+ machine (client and server,
  275.  * actually), sending to a nameserver datagram
  276.  * port with no nameserver will cause an
  277.  * ICMP port unreachable message to be returned.
  278.  * If our datagram socket is "connected" to the
  279.  * server, we get an ECONNREFUSED error on the next
  280.  * socket operation, and select returns if the
  281.  * error message is received.  We can thus detect
  282.  * the absence of a nameserver without timing out.
  283.  * If we have sent queries to at least two servers,
  284.  * however, we don't want to remain connected,
  285.  * as we wish to receive answers from the first
  286.  * server to respond.
  287.  */
  288. if (_res.nscount == 1 || (try == 0 && ns == 0)) {
  289. /*
  290.  * Don't use connect if we might
  291.  * still receive a response
  292.  * from another server.
  293.  */
  294. if (connected == 0) {
  295. if (connect(s, (struct sockaddr *)&_res.nsaddr_list[ns],
  296.     sizeof(struct sockaddr)) < 0) {
  297. #ifdef DEBUG
  298. if (_res.options & RES_DEBUG)
  299. perror("connect");
  300. #endif /* DEBUG */
  301. continue;
  302. }
  303. connected = 1;
  304. }
  305. if (send(s, buf, buflen, 0) != buflen) {
  306. #ifdef DEBUG
  307. if (_res.options & RES_DEBUG)
  308. perror("send");
  309. #endif /* DEBUG */
  310. continue;
  311. }
  312. } else {
  313. /*
  314.  * Disconnect if we want to listen
  315.  * for responses from more than one server.
  316.  */
  317. if (connected) {
  318. (void) connect(s, &no_addr,
  319.     sizeof(no_addr));
  320. connected = 0;
  321. }
  322. #endif /* BSD */
  323. if (sendto(s, (char *) buf, buflen, 0,
  324.     (struct sockaddr *)&_res.nsaddr_list[ns],
  325.     sizeof(struct sockaddr)) != buflen) {
  326. #ifdef DEBUG
  327. if (_res.options & RES_DEBUG)
  328. perror("sendto");
  329. #endif /* DEBUG */
  330. continue;
  331. }
  332. #if BSD >= 43
  333. }
  334. #endif
  335. /*
  336.  * Wait for reply
  337.  */
  338. timeout.tv_sec = (_res.retrans << try);
  339. if (try > 0)
  340. timeout.tv_sec /= _res.nscount;
  341. if (timeout.tv_sec <= 0)
  342. timeout.tv_sec = 1;
  343. timeout.tv_usec = 0;
  344. wait:
  345. FD_ZERO(&dsmask);
  346. FD_SET(s, &dsmask);
  347. n = select(s+1, &dsmask, (fd_set *)NULL,
  348. (fd_set *)NULL, &timeout);
  349. if (n < 0) {
  350. #ifdef DEBUG
  351. if (_res.options & RES_DEBUG)
  352. perror("select");
  353. #endif /* DEBUG */
  354. continue;
  355. }
  356. if (n == 0) {
  357. /*
  358.  * timeout
  359.  */
  360. #ifdef DEBUG
  361. if (_res.options & RES_DEBUG)
  362. printf("timeoutn");
  363. #endif /* DEBUG */
  364. #if BSD >= 43
  365. gotsomewhere = 1;
  366. #endif
  367. continue;
  368. }
  369. if ((resplen = recv(s, answer, anslen, 0)) <= 0) {
  370. #ifdef DEBUG
  371. if (_res.options & RES_DEBUG)
  372. perror("recvfrom");
  373. #endif /* DEBUG */
  374. continue;
  375. }
  376. gotsomewhere = 1;
  377. if (id != anhp->id) {
  378. /*
  379.  * response from old query, ignore it
  380.  */
  381. #ifdef ORG_RESOLVER
  382. #ifdef DEBUG
  383. if (_res.options & RES_DEBUG) {
  384. printf("old answer:n");
  385. __p_query(answer);
  386. }
  387. #endif /* DEBUG */
  388. #else
  389. if (pdnsDebugFunc != (FUNCPTR) NULL)
  390.     {
  391.     printf("old answer:n");
  392.     (*pdnsDebugFunc)(answer);
  393.     }
  394. #endif /* ORG_RESOLVER */
  395. goto wait;
  396. }
  397. if (!(_res.options & RES_IGNTC) && anhp->tc) {
  398. /*
  399.  * get rest of answer;
  400.  * use TCP with same server.
  401.  */
  402. #ifdef DEBUG
  403. if (_res.options & RES_DEBUG)
  404. printf("truncated answern");
  405. #endif /* DEBUG */
  406. (void) close(s);
  407. s = -1;
  408. v_circuit = 1;
  409. goto usevc;
  410. }
  411. }
  412. #ifdef ORG_RESOLVER
  413. #ifdef DEBUG
  414. if (_res.options & RES_DEBUG) {
  415. printf("got answer:n");
  416. __p_query(answer);
  417. }
  418. #endif /* DEBUG */
  419. #else
  420. if (pdnsDebugFunc != (FUNCPTR) NULL)
  421.     {
  422.     printf("got answer:n");
  423.     (*pdnsDebugFunc)(answer);
  424.     }
  425. #endif /* ORG_RESOLVER */
  426. /*
  427.  * If using virtual circuits, we assume that the first server
  428.  * is preferred * over the rest (i.e. it is on the local
  429.  * machine) and only keep that one open.
  430.  * If we have temporarily opened a virtual circuit,
  431.  * or if we haven't been asked to keep a socket open,
  432.  * close the socket.
  433.  */
  434. if ((v_circuit &&
  435.     ((_res.options & RES_USEVC) == 0 || ns != 0)) ||
  436.     (_res.options & RES_STAYOPEN) == 0) {
  437. (void) close(s);
  438. s = -1;
  439. }
  440. return (resplen);
  441.    }
  442. }
  443. if (s >= 0) {
  444. (void) close(s);
  445. s = -1;
  446. }
  447. if (v_circuit == 0)
  448. if (gotsomewhere == 0)
  449. errno = ECONNREFUSED; /* no nameservers found */
  450. else
  451. errno = ETIMEDOUT; /* no answer obtained */
  452. else
  453. errno = terrno;
  454. return (ERROR);
  455. }