Annotation of 43BSDTahoe/lib/libc/net/res_send.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 1985 Regents of the University of California.
                      3:  * All rights reserved.
                      4:  *
                      5:  * Redistribution and use in source and binary forms are permitted
                      6:  * provided that the above copyright notice and this paragraph are
                      7:  * duplicated in all such forms and that any documentation,
                      8:  * advertising materials, and other materials related to such
                      9:  * distribution and use acknowledge that the software was developed
                     10:  * by the University of California, Berkeley.  The name of the
                     11:  * University may not be used to endorse or promote products derived
                     12:  * from this software without specific prior written permission.
                     13:  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
                     14:  * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
                     15:  * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
                     16:  */
                     17: 
                     18: #if defined(LIBC_SCCS) && !defined(lint)
                     19: static char sccsid[] = "@(#)res_send.c 6.21 (Berkeley) 6/27/88";
                     20: #endif /* LIBC_SCCS and not lint */
                     21: 
                     22: /*
                     23:  * Send query to name server and wait for reply.
                     24:  */
                     25: 
                     26: #include <sys/param.h>
                     27: #include <sys/time.h>
                     28: #include <sys/socket.h>
                     29: #include <sys/uio.h>
                     30: #include <netinet/in.h>
                     31: #include <stdio.h>
                     32: #include <errno.h>
                     33: #include <arpa/nameser.h>
                     34: #include <resolv.h>
                     35: 
                     36: extern int errno;
                     37: 
                     38: static int s = -1;     /* socket used for communications */
                     39: static struct sockaddr no_addr;
                     40:   
                     41: 
                     42: #ifndef FD_SET
                     43: #define        NFDBITS         32
                     44: #define        FD_SETSIZE      32
                     45: #define        FD_SET(n, p)    ((p)->fds_bits[(n)/NFDBITS] |= (1 << ((n) % NFDBITS)))
                     46: #define        FD_CLR(n, p)    ((p)->fds_bits[(n)/NFDBITS] &= ~(1 << ((n) % NFDBITS)))
                     47: #define        FD_ISSET(n, p)  ((p)->fds_bits[(n)/NFDBITS] & (1 << ((n) % NFDBITS)))
                     48: #define FD_ZERO(p)     bzero((char *)(p), sizeof(*(p)))
                     49: #endif
                     50: 
                     51: #define KEEPOPEN (RES_USEVC|RES_STAYOPEN)
                     52: 
                     53: res_send(buf, buflen, answer, anslen)
                     54:        char *buf;
                     55:        int buflen;
                     56:        char *answer;
                     57:        int anslen;
                     58: {
                     59:        register int n;
                     60:        int retry, v_circuit, resplen, ns;
                     61:        int gotsomewhere = 0, connected = 0;
                     62:        u_short id, len;
                     63:        char *cp;
                     64:        fd_set dsmask;
                     65:        struct timeval timeout;
                     66:        HEADER *hp = (HEADER *) buf;
                     67:        HEADER *anhp = (HEADER *) answer;
                     68:        struct iovec iov[2];
                     69:        int terrno = ETIMEDOUT;
                     70:        char junk[512];
                     71: 
                     72: #ifdef DEBUG
                     73:        if (_res.options & RES_DEBUG) {
                     74:                printf("res_send()\n");
                     75:                p_query(buf);
                     76:        }
                     77: #endif DEBUG
                     78:        if (!(_res.options & RES_INIT))
                     79:                if (res_init() == -1) {
                     80:                        return(-1);
                     81:                }
                     82:        v_circuit = (_res.options & RES_USEVC) || buflen > PACKETSZ;
                     83:        id = hp->id;
                     84:        /*
                     85:         * Send request, RETRY times, or until successful
                     86:         */
                     87:        for (retry = _res.retry; retry > 0; retry--) {
                     88:           for (ns = 0; ns < _res.nscount; ns++) {
                     89: #ifdef DEBUG
                     90:                if (_res.options & RES_DEBUG)
                     91:                        printf("Querying server (# %d) address = %s\n", ns+1,
                     92:                              inet_ntoa(_res.nsaddr_list[ns].sin_addr));
                     93: #endif DEBUG
                     94:                if (v_circuit) {
                     95:                        int truncated = 0;
                     96: 
                     97:                        /*
                     98:                         * Use virtual circuit.
                     99:                         */
                    100:                        if (s < 0) {
                    101:                                s = socket(AF_INET, SOCK_STREAM, 0);
                    102:                                if (s < 0) {
                    103:                                        terrno = errno;
                    104: #ifdef DEBUG
                    105:                                        if (_res.options & RES_DEBUG)
                    106:                                            perror("socket failed");
                    107: #endif DEBUG
                    108:                                        continue;
                    109:                                }
                    110:                                if (connect(s, &(_res.nsaddr_list[ns]),
                    111:                                   sizeof(struct sockaddr)) < 0) {
                    112:                                        terrno = errno;
                    113: #ifdef DEBUG
                    114:                                        if (_res.options & RES_DEBUG)
                    115:                                            perror("connect failed");
                    116: #endif DEBUG
                    117:                                        (void) close(s);
                    118:                                        s = -1;
                    119:                                        continue;
                    120:                                }
                    121:                        }
                    122:                        /*
                    123:                         * Send length & message
                    124:                         */
                    125:                        len = htons((u_short)buflen);
                    126:                        iov[0].iov_base = (caddr_t)&len;
                    127:                        iov[0].iov_len = sizeof(len);
                    128:                        iov[1].iov_base = buf;
                    129:                        iov[1].iov_len = buflen;
                    130:                        if (writev(s, iov, 2) != sizeof(len) + buflen) {
                    131:                                terrno = errno;
                    132: #ifdef DEBUG
                    133:                                if (_res.options & RES_DEBUG)
                    134:                                        perror("write failed");
                    135: #endif DEBUG
                    136:                                (void) close(s);
                    137:                                s = -1;
                    138:                                continue;
                    139:                        }
                    140:                        /*
                    141:                         * Receive length & response
                    142:                         */
                    143:                        cp = answer;
                    144:                        len = sizeof(short);
                    145:                        while (len != 0 &&
                    146:                            (n = read(s, (char *)cp, (int)len)) > 0) {
                    147:                                cp += n;
                    148:                                len -= n;
                    149:                        }
                    150:                        if (n <= 0) {
                    151:                                terrno = errno;
                    152: #ifdef DEBUG
                    153:                                if (_res.options & RES_DEBUG)
                    154:                                        perror("read failed");
                    155: #endif DEBUG
                    156:                                (void) close(s);
                    157:                                s = -1;
                    158:                                continue;
                    159:                        }
                    160:                        cp = answer;
                    161:                        if ((resplen = ntohs(*(u_short *)cp)) > anslen) {
                    162: #ifdef DEBUG
                    163:                                if (_res.options & RES_DEBUG)
                    164:                                        fprintf(stderr, "response truncated\n");
                    165: #endif DEBUG
                    166:                                len = anslen;
                    167:                                truncated = 1;
                    168:                        } else
                    169:                                len = resplen;
                    170:                        while (len != 0 &&
                    171:                           (n = read(s, (char *)cp, (int)len)) > 0) {
                    172:                                cp += n;
                    173:                                len -= n;
                    174:                        }
                    175:                        if (n <= 0) {
                    176:                                terrno = errno;
                    177: #ifdef DEBUG
                    178:                                if (_res.options & RES_DEBUG)
                    179:                                        perror("read failed");
                    180: #endif DEBUG
                    181:                                (void) close(s);
                    182:                                s = -1;
                    183:                                continue;
                    184:                        }
                    185:                        if (truncated) {
                    186:                                /*
                    187:                                 * Flush rest of answer
                    188:                                 * so connection stays in synch.
                    189:                                 */
                    190:                                anhp->tc = 1;
                    191:                                len = resplen - anslen;
                    192:                                while (len != 0) {
                    193:                                        n = (len > sizeof(junk) ?
                    194:                                            sizeof(junk) : len);
                    195:                                        if ((n = read(s, junk, n)) > 0)
                    196:                                                len -= n;
                    197:                                        else
                    198:                                                break;
                    199:                                }
                    200:                        }
                    201:                } else {
                    202:                        /*
                    203:                         * Use datagrams.
                    204:                         */
                    205:                        if (s < 0)
                    206:                                s = socket(AF_INET, SOCK_DGRAM, 0);
                    207: #if    BSD >= 43
                    208:                        if (_res.nscount == 1 || retry == _res.retry) {
                    209:                                /*
                    210:                                 * Don't use connect if we might
                    211:                                 * still receive a response
                    212:                                 * from another server.
                    213:                                 */
                    214:                                if (connected == 0) {
                    215:                                        if (connect(s, &_res.nsaddr_list[ns],
                    216:                                            sizeof(struct sockaddr)) < 0) {
                    217: #ifdef DEBUG
                    218:                                                if (_res.options & RES_DEBUG)
                    219:                                                        perror("connect");
                    220: #endif DEBUG
                    221:                                                continue;
                    222:                                        }
                    223:                                        connected = 1;
                    224:                                }
                    225:                                if (send(s, buf, buflen, 0) != buflen) {
                    226: #ifdef DEBUG
                    227:                                        if (_res.options & RES_DEBUG)
                    228:                                                perror("send");
                    229: #endif DEBUG
                    230:                                        continue;
                    231:                                }
                    232:                        } else {
                    233:                                /*
                    234:                                 * Disconnect if we want to listen
                    235:                                 * for responses from more than one server.
                    236:                                 */
                    237:                                if (connected) {
                    238:                                        (void) connect(s, &no_addr,
                    239:                                            sizeof(no_addr));
                    240:                                        connected = 0;
                    241:                                }
                    242: #endif BSD
                    243:                                if (sendto(s, buf, buflen, 0,
                    244:                                    &_res.nsaddr_list[ns],
                    245:                                    sizeof(struct sockaddr)) != buflen) {
                    246: #ifdef DEBUG
                    247:                                        if (_res.options & RES_DEBUG)
                    248:                                                perror("sendto");
                    249: #endif DEBUG
                    250:                                        continue;
                    251:                                }
                    252: #if    BSD >= 43
                    253:                        }
                    254: #endif
                    255: 
                    256:                        /*
                    257:                         * Wait for reply
                    258:                         */
                    259:                        timeout.tv_sec = (_res.retrans << (_res.retry - retry))
                    260:                                / _res.nscount;
                    261:                        if (timeout.tv_sec <= 0)
                    262:                                timeout.tv_sec = 1;
                    263:                        timeout.tv_usec = 0;
                    264: wait:
                    265:                        FD_ZERO(&dsmask);
                    266:                        FD_SET(s, &dsmask);
                    267:                        n = select(s+1, &dsmask, (fd_set *)NULL,
                    268:                                (fd_set *)NULL, &timeout);
                    269:                        if (n < 0) {
                    270: #ifdef DEBUG
                    271:                                if (_res.options & RES_DEBUG)
                    272:                                        perror("select");
                    273: #endif DEBUG
                    274:                                continue;
                    275:                        }
                    276:                        if (n == 0) {
                    277:                                /*
                    278:                                 * timeout
                    279:                                 */
                    280: #ifdef DEBUG
                    281:                                if (_res.options & RES_DEBUG)
                    282:                                        printf("timeout\n");
                    283: #endif DEBUG
                    284:                                gotsomewhere = 1;
                    285:                                continue;
                    286:                        }
                    287:                        if ((resplen = recv(s, answer, anslen, 0)) <= 0) {
                    288: #ifdef DEBUG
                    289:                                if (_res.options & RES_DEBUG)
                    290:                                        perror("recvfrom");
                    291: #endif DEBUG
                    292:                                continue;
                    293:                        }
                    294:                        gotsomewhere = 1;
                    295:                        if (id != anhp->id) {
                    296:                                /*
                    297:                                 * response from old query, ignore it
                    298:                                 */
                    299: #ifdef DEBUG
                    300:                                if (_res.options & RES_DEBUG) {
                    301:                                        printf("old answer:\n");
                    302:                                        p_query(answer);
                    303:                                }
                    304: #endif DEBUG
                    305:                                goto wait;
                    306:                        }
                    307:                        if (!(_res.options & RES_IGNTC) && anhp->tc) {
                    308:                                /*
                    309:                                 * get rest of answer
                    310:                                 */
                    311: #ifdef DEBUG
                    312:                                if (_res.options & RES_DEBUG)
                    313:                                        printf("truncated answer\n");
                    314: #endif DEBUG
                    315:                                (void) close(s);
                    316:                                s = -1;
                    317:                                /*
                    318:                                 * retry decremented on continue
                    319:                                 * to desired starting value
                    320:                                 */
                    321:                                retry = _res.retry + 1;
                    322:                                v_circuit = 1;
                    323:                                continue;
                    324:                        }
                    325:                }
                    326: #ifdef DEBUG
                    327:                if (_res.options & RES_DEBUG) {
                    328:                        printf("got answer:\n");
                    329:                        p_query(answer);
                    330:                }
                    331: #endif DEBUG
                    332:                /*
                    333:                 * We are going to assume that the first server is preferred
                    334:                 * over the rest (i.e. it is on the local machine) and only
                    335:                 * keep that one open.
                    336:                 */
                    337:                if ((_res.options & KEEPOPEN) == 0 || ns != 0) {
                    338:                        (void) close(s);
                    339:                        s = -1;
                    340:                }
                    341:                return (resplen);
                    342:           }
                    343:        }
                    344:        if (s >= 0) {
                    345:                (void) close(s);
                    346:                s = -1;
                    347:        }
                    348:        if (v_circuit == 0)
                    349:                if (gotsomewhere == 0)
                    350:                        errno = ECONNREFUSED;
                    351:                else
                    352:                        errno = ETIMEDOUT;
                    353:        else
                    354:                errno = terrno;
                    355:        return (-1);
                    356: }
                    357: 
                    358: /*
                    359:  * This routine is for closing the socket if a virtual circuit is used and
                    360:  * the program wants to close it.  This provides support for endhostent()
                    361:  * which expects to close the socket.
                    362:  *
                    363:  * This routine is not expected to be user visible.
                    364:  */
                    365: _res_close()
                    366: {
                    367:        if (s != -1) {
                    368:                (void) close(s);
                    369:                s = -1;
                    370:        }
                    371: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.