]>
git.saurik.com Git - apple/libinfo.git/blob - nis.subproj/yp_bind.c
2 * Copyright (c) 1999 Apple Computer, Inc. All rights reserved.
4 * @APPLE_LICENSE_HEADER_START@
6 * Portions Copyright (c) 1999 Apple Computer, Inc. All Rights
7 * Reserved. This file contains Original Code and/or Modifications of
8 * Original Code as defined in and that are subject to the Apple Public
9 * Source License Version 1.1 (the "License"). You may not use this file
10 * except in compliance with the License. Please obtain a copy of the
11 * License at http://www.apple.com/publicsource and read it before using
14 * The Original Code and all software distributed under the License are
15 * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
16 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
17 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE OR NON- INFRINGEMENT. Please see the
19 * License for the specific language governing rights and limitations
22 * @APPLE_LICENSE_HEADER_END@
25 * Copyright (c) 1996 Theo de Raadt <deraadt@theos.com>
26 * Copyright (c) 1992, 1993 Theo de Raadt <deraadt@theos.com>
27 * All rights reserved.
29 * Redistribution and use in source and binary forms, with or without
30 * modification, are permitted provided that the following conditions
32 * 1. Redistributions of source code must retain the above copyright
33 * notice, this list of conditions and the following disclaimer.
34 * 2. Redistributions in binary form must reproduce the above copyright
35 * notice, this list of conditions and the following disclaimer in the
36 * documentation and/or other materials provided with the distribution.
37 * 3. All advertising materials mentioning features or use of this software
38 * must display the following acknowledgement:
39 * This product includes software developed by Theo de Raadt.
40 * 4. The name of the author may not be used to endorse or promote products
41 * derived from this software without specific prior written permission.
43 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
44 * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
45 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
46 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
47 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
48 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
49 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
50 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
51 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
52 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
56 #if defined(LIBC_SCCS) && !defined(lint)
57 static char *rcsid
= "$OpenBSD: yp_bind.c,v 1.9 1997/04/29 21:25:20 deraadt Exp $";
58 #endif /* LIBC_SCCS and not lint */
60 #include <sys/param.h>
61 #include <sys/types.h>
62 #include <sys/socket.h>
72 #include <rpcsvc/yp.h>
73 #include <rpcsvc/ypclnt.h>
74 #include "ypinternal.h"
76 struct dom_binding
*_ypbindlist
= NULL
;
77 char _yp_domain
[MAXHOSTNAMELEN
] = { '\0' };
79 int _yplib_timeout
= 10;
84 struct dom_binding
**ypdb
;
87 char path
[MAXPATHLEN
];
88 struct dom_binding
*ysd
, *ysd2
;
89 struct ypbind_resp ypbr
;
91 struct sockaddr_in clnt_sin
;
92 struct ypbind_binding
*bn
;
93 int clnt_sock
, fd
, gpid
;
100 * test if YP is running or not
102 if ((fd
= open(YPBINDLOCK
, O_RDONLY
)) == -1)
104 if (!(flock(fd
, LOCK_EX
| LOCK_NB
) == -1 && errno
== EWOULDBLOCK
)) {
111 if (!(pid
== -1 || pid
== gpid
)) {
115 clnt_destroy(ysd
->dom_client
);
116 ysd2
= ysd
->dom_pnext
;
127 if (dom
== NULL
|| strlen(dom
) == 0)
128 return YPERR_BADARGS
;
130 for (ysd
= _ypbindlist
; ysd
; ysd
= ysd
->dom_pnext
)
131 if (strcmp(dom
, ysd
->dom_domain
) == 0)
134 if ((ysd
= malloc(sizeof *ysd
)) == NULL
)
136 (void)memset(ysd
, 0, sizeof *ysd
);
137 ysd
->dom_socket
= -1;
142 if (ysd
->dom_vers
== 0) {
143 (void) snprintf(path
, sizeof(path
), "%s/%s.%d",
145 if ((fd
= open(path
, O_RDONLY
)) == -1) {
147 * no binding file, YP is dead, or not yet fully
152 if (flock(fd
, LOCK_EX
| LOCK_NB
) == -1 &&
153 errno
== EWOULDBLOCK
) {
158 * we fetch the ypbind port number, but do
161 iov
[0].iov_base
= (caddr_t
) &ypb_port
;
162 iov
[0].iov_len
= sizeof ypb_port
;
163 iov
[1].iov_base
= (caddr_t
) &ypbr
;
164 iov
[1].iov_len
= sizeof ypbr
;
166 r
= readv(fd
, iov
, 2);
167 if (r
!= iov
[0].iov_len
+ iov
[1].iov_len
) {
175 /* no lock on binding file, YP is dead. */
183 if (ysd
->dom_vers
== -1 || ysd
->dom_vers
== 0) {
184 (void)memset(&clnt_sin
, 0, sizeof clnt_sin
);
185 clnt_sin
.sin_len
= sizeof(struct sockaddr_in
);
186 clnt_sin
.sin_family
= AF_INET
;
187 clnt_sin
.sin_addr
.s_addr
= htonl(INADDR_LOOPBACK
);
189 clnt_sock
= RPC_ANYSOCK
;
190 client
= clnttcp_create(&clnt_sin
, YPBINDPROG
, YPBINDVERS
,
192 if (client
== NULL
) {
193 clnt_pcreateerror("clnttcp_create");
198 if (ntohs(clnt_sin
.sin_port
) >= IPPORT_RESERVED
||
199 ntohs(clnt_sin
.sin_port
) == 20) {
201 * YP was not running, but someone has registered
202 * ypbind with portmap -- this simply means YP is
205 clnt_destroy(client
);
210 tv
.tv_sec
= _yplib_timeout
;
212 r
= clnt_call(client
, YPBINDPROC_DOMAIN
, xdr_domainname
,
213 &dom
, xdr_ypbind_resp
, &ypbr
, tv
);
214 if (r
!= RPC_SUCCESS
) {
215 if (new == 0 || count
)
217 "YP server for domain %s not responding, still trying\n",
220 clnt_destroy(client
);
224 clnt_destroy(client
);
226 bn
= &ypbr
.ypbind_resp_u
.ypbind_bindinfo
;
227 memcpy(&port
, &bn
->ypbind_binding_port
, sizeof port
);
228 if (ntohs(port
) >= IPPORT_RESERVED
||
231 * This is bogus -- the ypbind wants me to
232 * communicate to an insecure ypserv. We are
233 * within rights to syslog this as an attack,
234 * but for now we'll simply ignore it; real YP
235 * is obviously not running.
241 (void)memset(&ysd
->dom_server_addr
, 0,
242 sizeof ysd
->dom_server_addr
);
243 ysd
->dom_server_addr
.sin_len
= sizeof(struct sockaddr_in
);
244 ysd
->dom_server_addr
.sin_family
= AF_INET
;
245 memcpy(&ysd
->dom_server_addr
.sin_port
,
246 &bn
->ypbind_binding_port
,
247 sizeof(ysd
->dom_server_addr
.sin_port
));
248 memcpy(&ysd
->dom_server_addr
.sin_addr
.s_addr
,
249 &bn
->ypbind_binding_addr
,
250 sizeof(ysd
->dom_server_addr
.sin_addr
.s_addr
));
251 ysd
->dom_server_port
= ysd
->dom_server_addr
.sin_port
;
252 ysd
->dom_vers
= YPVERS
;
253 (void)strncpy(ysd
->dom_domain
, dom
, sizeof ysd
->dom_domain
-1);
254 ysd
->dom_domain
[sizeof ysd
->dom_domain
-1] = '\0';
256 tv
.tv_sec
= _yplib_timeout
/ 2;
259 clnt_destroy(ysd
->dom_client
);
260 ysd
->dom_socket
= RPC_ANYSOCK
;
261 ysd
->dom_client
= clntudp_create(&ysd
->dom_server_addr
,
262 YPPROG
, YPVERS
, tv
, &ysd
->dom_socket
);
263 if (ysd
->dom_client
== NULL
) {
264 clnt_pcreateerror("clntudp_create");
268 if (fcntl(ysd
->dom_socket
, F_SETFD
, 1) == -1)
269 perror("fcntl: F_SETFD");
272 ysd
->dom_pnext
= _ypbindlist
;
282 struct dom_binding
*ypb
;
284 if (ypb
->dom_client
) clnt_destroy(ypb
->dom_client
);
285 ypb
->dom_client
= NULL
;
286 ypb
->dom_socket
= -1;
293 return _yp_dobind(dom
, NULL
);
300 struct dom_binding
*ypb
, *ypbp
;
303 for (ypb
= _ypbindlist
; ypb
; ypb
= ypb
->dom_pnext
) {
304 if (strcmp(dom
, ypb
->dom_domain
) == 0) {
305 if (ypb
->dom_client
) clnt_destroy(ypb
->dom_client
);
307 ypbp
->dom_pnext
= ypb
->dom_pnext
;
309 _ypbindlist
= ypb
->dom_pnext
;