Annotation of 43BSD/etc/timed/measure.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 1983 Regents of the University of California.
                      3:  * All rights reserved.  The Berkeley software License Agreement
                      4:  * specifies the terms and conditions for redistribution.
                      5:  */
                      6: 
                      7: #ifndef lint
                      8: static char sccsid[] = "@(#)measure.c  2.4 (Berkeley) 5/28/86";
                      9: #endif not lint
                     10: 
                     11: #include "globals.h"
                     12: #include <protocols/timed.h>
                     13: #include <netinet/in_systm.h>
                     14: #include <netinet/ip.h>
                     15: #include <netinet/ip_icmp.h>
                     16: 
                     17: #define BIASP          43199999
                     18: #define BIASN          -43200000
                     19: #define MODULO         86400000
                     20: #define PROCESSING_TIME        5       /* ms. to reduce error in measurement */
                     21: 
                     22: #define PACKET_IN      1024
                     23: 
                     24: extern int id;
                     25: int measure_delta;
                     26: extern int sock_raw;
                     27: static n_short seqno = 0;
                     28: 
                     29: /*
                     30:  * Measures the differences between machines' clocks using
                     31:  * ICMP timestamp messages.
                     32:  */
                     33: 
                     34: measure(wait, addr)
                     35: struct timeval *wait;
                     36: struct sockaddr_in *addr;
                     37: {
                     38:        int length;
                     39:        int status;
                     40:        int msgcount, trials;
                     41:        int cc, count;
                     42:        fd_set ready;
                     43:        long sendtime, recvtime, histime;
                     44:        long min1, min2, diff;
                     45:        register long delta1, delta2;
                     46:        struct timeval tv1, tout;
                     47:        u_char packet[PACKET_IN], opacket[64];
                     48:        register struct icmp *icp = (struct icmp *) packet;
                     49:        register struct icmp *oicp = (struct icmp *) opacket;
                     50:        struct ip *ip = (struct ip *) packet;
                     51: 
                     52:        min1 = min2 = 0x7fffffff;
                     53:        status = HOSTDOWN;
                     54:        measure_delta = HOSTDOWN;
                     55: 
                     56: /* empties the icmp input queue */
                     57:        FD_ZERO(&ready);
                     58: empty:
                     59:        tout.tv_sec = tout.tv_usec = 0;
                     60:        FD_SET(sock_raw, &ready);
                     61:        if (select(FD_SETSIZE, &ready, (fd_set *)0, (fd_set *)0, &tout)) {
                     62:                length = sizeof(struct sockaddr_in);
                     63:                cc = recvfrom(sock_raw, (char *)packet, PACKET_IN, 0, 
                     64:                    (struct sockaddr_in *)NULL, &length);
                     65:                if (cc < 0)
                     66:                        return(-1);
                     67:                goto empty;
                     68:        }
                     69: 
                     70:        /*
                     71:         * To measure the difference, select MSGS messages whose round-trip
                     72:         * time is smaller than RANGE if ckrange is 1, otherwise simply
                     73:         * select MSGS messages regardless of round-trip transmission time.
                     74:         * Choose the smallest transmission time in each of the two directions.
                     75:         * Use these two latter quantities to compute the delta between
                     76:         * the two clocks.
                     77:         */
                     78: 
                     79:        length = sizeof(struct sockaddr_in);
                     80:        oicp->icmp_type = ICMP_TSTAMP;
                     81:        oicp->icmp_code = 0;
                     82:        oicp->icmp_cksum = 0;
                     83:        oicp->icmp_id = id;
                     84:        oicp->icmp_rtime = 0;
                     85:        oicp->icmp_ttime = 0;
                     86:        FD_ZERO(&ready);
                     87:        msgcount = 0;
                     88:        for (trials = 0; msgcount < MSGS && trials < TRIALS; ++trials) {
                     89:                oicp->icmp_seq = ++seqno;
                     90:                oicp->icmp_cksum = 0;
                     91: 
                     92:                tout.tv_sec = wait->tv_sec;
                     93:                tout.tv_usec = wait->tv_usec;
                     94: 
                     95:                (void)gettimeofday (&tv1, (struct timezone *)0);
                     96:                sendtime = oicp->icmp_otime = (tv1.tv_sec % (24*60*60)) * 1000 
                     97:                                                        + tv1.tv_usec / 1000;
                     98:                oicp->icmp_cksum = in_cksum((u_short *)oicp, sizeof(*oicp));
                     99:        
                    100:                count = sendto(sock_raw, (char *)opacket, sizeof(*oicp), 0, 
                    101:                                addr, sizeof(struct sockaddr_in));
                    102:                if (count < 0) {
                    103:                        status = UNREACHABLE;
                    104:                        return(-1);
                    105:                }
                    106:                for (;;) {
                    107:                        FD_SET(sock_raw, &ready);
                    108:                        if ((count = select(FD_SETSIZE, &ready, (fd_set *)0,
                    109:                            (fd_set *)0, &tout)) <= 0)
                    110:                                break;
                    111:                        cc = recvfrom(sock_raw, (char *)packet, PACKET_IN, 0, 
                    112:                            (struct sockaddr_in *)NULL, &length);
                    113:                        (void)gettimeofday(&tv1, (struct timezone *)0);
                    114:                        if (cc < 0)
                    115:                                return(-1);
                    116:                        icp = (struct icmp *)(packet + (ip->ip_hl << 2));
                    117:                        if((icp->icmp_type == ICMP_TSTAMPREPLY) &&
                    118:                            icp->icmp_id == id && icp->icmp_seq == seqno)
                    119:                                break;
                    120:                }
                    121:                if (count <= 0)
                    122:                        continue;               /* resend */
                    123:                recvtime = (tv1.tv_sec % (24*60*60)) * 1000 +
                    124:                    tv1.tv_usec / 1000;
                    125:                diff = recvtime - sendtime;
                    126:                /*
                    127:                 * diff can be less than 0 aroud midnight
                    128:                 */
                    129:                if (diff < 0)
                    130:                        continue;
                    131:                msgcount++;
                    132:                histime = ntohl((u_long)icp->icmp_rtime);
                    133:                /*
                    134:                 * a hosts using a time format different from 
                    135:                 * ms. since midnight UT (as per RFC792) should
                    136:                 * set the high order bit of the 32-bit time
                    137:                 * value it transmits.
                    138:                 */
                    139:                if ((histime & 0x80000000) != 0) {
                    140:                        status = NONSTDTIME;
                    141:                        break;
                    142:                }
                    143:                status = GOOD;
                    144:                delta1 = histime - sendtime;
                    145:                /*
                    146:                 * Handles wrap-around to avoid that around 
                    147:                 * midnight small time differences appear 
                    148:                 * enormous. However, the two machine's clocks
                    149:                 * must be within 12 hours from each other.
                    150:                 */
                    151:                if (delta1 < BIASN)
                    152:                        delta1 += MODULO;
                    153:                else if (delta1 > BIASP)
                    154:                        delta1 -= MODULO;
                    155:                delta2 = recvtime - histime;
                    156:                if (delta2 < BIASN)
                    157:                        delta2 += MODULO;
                    158:                else if (delta2 > BIASP)
                    159:                        delta2 -= MODULO;
                    160:                if (delta1 < min1)  
                    161:                        min1 = delta1;
                    162:                if (delta2 < min2)
                    163:                        min2 = delta2;
                    164:                if (diff < RANGE) {
                    165:                        min1 = delta1;
                    166:                        min2 = delta2;
                    167:                        break;
                    168:                }
                    169:        }
                    170: 
                    171:        /*
                    172:         * If no answer is received for TRIALS consecutive times, 
                    173:         * the machine is assumed to be down
                    174:         */
                    175:         if (status == GOOD) {
                    176:                measure_delta = (min1 - min2)/2 + PROCESSING_TIME;
                    177:        }
                    178:        return(status);
                    179: }

unix.superglobalmegacorp.com

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