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