xref: /freebsd/lib/libc/rpc/rtime.c (revision dc36d6f9bb1753f3808552f3afd30eda9a7b206a)
12e322d37SHiroki Sato /*-
2*8a16b7a1SPedro F. Giffuni  * SPDX-License-Identifier: BSD-3-Clause
3*8a16b7a1SPedro F. Giffuni  *
42e322d37SHiroki Sato  * Copyright (c) 2009, Sun Microsystems, Inc.
52e322d37SHiroki Sato  * All rights reserved.
6e8636dfdSBill Paul  *
72e322d37SHiroki Sato  * Redistribution and use in source and binary forms, with or without
82e322d37SHiroki Sato  * modification, are permitted provided that the following conditions are met:
92e322d37SHiroki Sato  * - Redistributions of source code must retain the above copyright notice,
102e322d37SHiroki Sato  *   this list of conditions and the following disclaimer.
112e322d37SHiroki Sato  * - Redistributions in binary form must reproduce the above copyright notice,
122e322d37SHiroki Sato  *   this list of conditions and the following disclaimer in the documentation
132e322d37SHiroki Sato  *   and/or other materials provided with the distribution.
142e322d37SHiroki Sato  * - Neither the name of Sun Microsystems, Inc. nor the names of its
152e322d37SHiroki Sato  *   contributors may be used to endorse or promote products derived
162e322d37SHiroki Sato  *   from this software without specific prior written permission.
17e8636dfdSBill Paul  *
182e322d37SHiroki Sato  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
192e322d37SHiroki Sato  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
202e322d37SHiroki Sato  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
212e322d37SHiroki Sato  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
222e322d37SHiroki Sato  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
232e322d37SHiroki Sato  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
242e322d37SHiroki Sato  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
252e322d37SHiroki Sato  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
262e322d37SHiroki Sato  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
272e322d37SHiroki Sato  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
282e322d37SHiroki Sato  * POSSIBILITY OF SUCH DAMAGE.
29e8636dfdSBill Paul  */
30e8636dfdSBill Paul 
31e8636dfdSBill Paul /*
32e8636dfdSBill Paul  * Copyright (c) 1988 by Sun Microsystems, Inc.
33e8636dfdSBill Paul 
34e8636dfdSBill Paul  */
35e8636dfdSBill Paul 
36e8636dfdSBill Paul /*
37e8636dfdSBill Paul  * rtime - get time from remote machine
38e8636dfdSBill Paul  *
39e8636dfdSBill Paul  * gets time, obtaining value from host
40e8636dfdSBill Paul  * on the udp/time socket.  Since timeserver returns
41e8636dfdSBill Paul  * with time of day in seconds since Jan 1, 1900,  must
42e8636dfdSBill Paul  * subtract seconds before Jan 1, 1970 to get
43e8636dfdSBill Paul  * what unix uses.
44e8636dfdSBill Paul  */
45d201fe46SDaniel Eischen #include "namespace.h"
46e8636dfdSBill Paul #include <stdlib.h>
47e8636dfdSBill Paul #include <string.h>
48e8636dfdSBill Paul #include <unistd.h>
49e8636dfdSBill Paul #include <errno.h>
50e8636dfdSBill Paul #include <sys/types.h>
51e8636dfdSBill Paul #include <sys/socket.h>
52e8636dfdSBill Paul #include <sys/time.h>
53e8636dfdSBill Paul #include <netinet/in.h>
54e8636dfdSBill Paul #include <stdio.h>
55e8636dfdSBill Paul #include <netdb.h>
56d201fe46SDaniel Eischen #include "un-namespace.h"
57e8636dfdSBill Paul 
58c05ac53bSDavid E. O'Brien extern int _rpc_dtablesize( void );
59e8636dfdSBill Paul 
60e8636dfdSBill Paul #define	NYEARS	(unsigned long)(1970 - 1900)
61e8636dfdSBill Paul #define	TOFFSET (unsigned long)(60*60*24*(365*NYEARS + (NYEARS/4)))
62e8636dfdSBill Paul 
63c05ac53bSDavid E. O'Brien static void do_close( int );
64e8636dfdSBill Paul 
65e8636dfdSBill Paul int
rtime(struct sockaddr_in * addrp,struct timeval * timep,struct timeval * timeout)6668895e38SCraig Rodrigues rtime(struct sockaddr_in *addrp, struct timeval *timep,
6768895e38SCraig Rodrigues     struct timeval *timeout)
68e8636dfdSBill Paul {
69e8636dfdSBill Paul 	int s;
70e8636dfdSBill Paul 	fd_set readfds;
71e8636dfdSBill Paul 	int res;
72e8636dfdSBill Paul 	unsigned long thetime;
73e8636dfdSBill Paul 	struct sockaddr_in from;
74720138bbSStefan Farfeleder 	socklen_t fromlen;
75e8636dfdSBill Paul 	int type;
76e8636dfdSBill Paul 	struct servent *serv;
77e8636dfdSBill Paul 
78e8636dfdSBill Paul 	if (timeout == NULL) {
79e8636dfdSBill Paul 		type = SOCK_STREAM;
80e8636dfdSBill Paul 	} else {
81e8636dfdSBill Paul 		type = SOCK_DGRAM;
82e8636dfdSBill Paul 	}
83d201fe46SDaniel Eischen 	s = _socket(AF_INET, type, 0);
84e8636dfdSBill Paul 	if (s < 0) {
85e8636dfdSBill Paul 		return(-1);
86e8636dfdSBill Paul 	}
87e8636dfdSBill Paul 	addrp->sin_family = AF_INET;
88e8636dfdSBill Paul 
89e8636dfdSBill Paul 	/* TCP and UDP port are the same in this case */
90e8636dfdSBill Paul 	if ((serv = getservbyname("time", "tcp")) == NULL) {
91e8636dfdSBill Paul 		return(-1);
92e8636dfdSBill Paul 	}
93e8636dfdSBill Paul 
94e8636dfdSBill Paul 	addrp->sin_port = serv->s_port;
95e8636dfdSBill Paul 
96e8636dfdSBill Paul 	if (type == SOCK_DGRAM) {
97d201fe46SDaniel Eischen 		res = _sendto(s, (char *)&thetime, sizeof(thetime), 0,
98e8636dfdSBill Paul 			     (struct sockaddr *)addrp, sizeof(*addrp));
99e8636dfdSBill Paul 		if (res < 0) {
100e8636dfdSBill Paul 			do_close(s);
101e8636dfdSBill Paul 			return(-1);
102e8636dfdSBill Paul 		}
103e8636dfdSBill Paul 		do {
104e8636dfdSBill Paul 			FD_ZERO(&readfds);
105e8636dfdSBill Paul 			FD_SET(s, &readfds);
106d201fe46SDaniel Eischen 			res = _select(_rpc_dtablesize(), &readfds,
107e8636dfdSBill Paul 				     (fd_set *)NULL, (fd_set *)NULL, timeout);
108e8636dfdSBill Paul 		} while (res < 0 && errno == EINTR);
109e8636dfdSBill Paul 		if (res <= 0) {
110e8636dfdSBill Paul 			if (res == 0) {
111e8636dfdSBill Paul 				errno = ETIMEDOUT;
112e8636dfdSBill Paul 			}
113e8636dfdSBill Paul 			do_close(s);
114e8636dfdSBill Paul 			return(-1);
115e8636dfdSBill Paul 		}
116e8636dfdSBill Paul 		fromlen = sizeof(from);
117d201fe46SDaniel Eischen 		res = _recvfrom(s, (char *)&thetime, sizeof(thetime), 0,
118e8636dfdSBill Paul 			       (struct sockaddr *)&from, &fromlen);
119e8636dfdSBill Paul 		do_close(s);
120e8636dfdSBill Paul 		if (res < 0) {
121e8636dfdSBill Paul 			return(-1);
122e8636dfdSBill Paul 		}
123e8636dfdSBill Paul 	} else {
124d201fe46SDaniel Eischen 		if (_connect(s, (struct sockaddr *)addrp, sizeof(*addrp)) < 0) {
125e8636dfdSBill Paul 			do_close(s);
126e8636dfdSBill Paul 			return(-1);
127e8636dfdSBill Paul 		}
1289233c4d9SJason Evans 		res = _read(s, (char *)&thetime, sizeof(thetime));
129e8636dfdSBill Paul 		do_close(s);
130e8636dfdSBill Paul 		if (res < 0) {
131e8636dfdSBill Paul 			return(-1);
132e8636dfdSBill Paul 		}
133e8636dfdSBill Paul 	}
134e8636dfdSBill Paul 	if (res != sizeof(thetime)) {
135e8636dfdSBill Paul 		errno = EIO;
136e8636dfdSBill Paul 		return(-1);
137e8636dfdSBill Paul 	}
138e8636dfdSBill Paul 	thetime = ntohl(thetime);
139e8636dfdSBill Paul 	timep->tv_sec = thetime - TOFFSET;
140e8636dfdSBill Paul 	timep->tv_usec = 0;
141e8636dfdSBill Paul 	return(0);
142e8636dfdSBill Paul }
143e8636dfdSBill Paul 
144e8636dfdSBill Paul static void
do_close(int s)14568895e38SCraig Rodrigues do_close(int s)
146e8636dfdSBill Paul {
147e8636dfdSBill Paul 	int save;
148e8636dfdSBill Paul 
149e8636dfdSBill Paul 	save = errno;
1509233c4d9SJason Evans 	(void)_close(s);
151e8636dfdSBill Paul 	errno = save;
152e8636dfdSBill Paul }
153