#include <stdio.h>
#include <ctype.h>
#include <rpc/rpc.h>
#include <rpcsvc/yp_prot.h>
#include <rpcsvc/ypclnt.h>
#include "yp_b.h"
#include "ypv2_bind.h"
#include <string.h>
#include <netdir.h>
#include <unistd.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <inet/ip.h>
#include <inet/ip6.h>
#include <netinet/ip6.h>
#include <sys/utsname.h>
#define YPSLEEPTIME 5
#define TIMEOUT 30
#define INTER_TRY 10
static int translate = TRUE;
static int dodump = FALSE;
static char *domain = NULL;
static char default_domain_name[YPMAXDOMAIN];
static char *host = NULL;
static int vers = YPBINDVERS;
static char default_host_name[256];
static bool get_master = FALSE;
static bool get_server = FALSE;
static char *map = NULL;
static char nm[YPMAXMAP+1];
static struct timeval timeout = {
TIMEOUT,
0
};
static char nullstring[] = "\000";
static char err_usage[] =
"Usage:\n\
ypwhich [-d domain] [[-t] -m [mname] | [-Vn] host]\n\
ypwhich -x\n\
where\n\
mname may be either a mapname or a nickname for a map.\n\
host if specified, is the machine whose NIS server is to be found.\n\
-t inhibits map nickname translation.\n\
-Vn version of ypbind, V3 is default.\n\
-x dumps the map nickname translation table.\n";
static char err_bad_args[] =
"ypwhich: %s argument is bad.\n";
static char err_cant_get_kname[] =
"ypwhich: can't get %s back from system call.\n";
static char err_null_kname[] =
"ypwhich: the %s hasn't been set on this machine.\n";
static char err_bad_mapname[] = "mapname";
static char err_bad_domainname[] = "domainname";
static char err_bad_hostname[] = "hostname";
static void get_command_line_args();
static void getdomain();
static void getlochost();
static void get_server_name();
static int call_binder();
static void get_map_master();
extern void maketable();
extern int getmapname();
#ifdef DEBUG
static void dump_response();
#endif
static void dump_ypmaps();
static void dumpmaps();
static bool xdr_yp_inaddr();
static bool xdr_old_ypbind_resp();
static bool xdr_old_yp_binding();
static int old_call_binder();
static void print_server();
struct old_ypbind_binding {
struct in_addr ypbind_binding_addr;
unsigned short int ypbind_binding_port;
};
struct old_ypbind_resp {
enum ypbind_resptype ypbind_status;
union {
unsigned long ypbind_error;
struct old_ypbind_binding ypbind_bindinfo;
} ypbind_respbody;
};
int
main(int argc, char **argv)
{
get_command_line_args(argc, argv);
if (dodump) {
maketable(dodump);
exit(0);
}
if (!domain) {
getdomain();
}
if (map && translate && (strchr(map, '.') == NULL) &&
(getmapname(map, nm))) {
map = nm;
}
if (get_server) {
if (!host)
getlochost();
get_server_name();
} else {
if (map)
get_map_master();
else
dump_ypmaps();
}
return (0);
}
static void
get_command_line_args(argc, argv)
int argc;
char **argv;
{
argv++;
if (argc == 1) {
get_server = TRUE;
return;
}
while (--argc) {
if ((*argv)[0] == '-') {
switch ((*argv)[1]) {
case 'V':
vers = atoi(argv[0]+2);
if (vers < 1) {
(void) fprintf(stderr, err_usage);
exit(1);
}
argv++;
break;
case 'm':
get_master = TRUE;
argv++;
if (argc > 1) {
if ((*(argv))[0] == '-') {
break;
}
argc--;
map = *argv;
argv++;
if ((int)strlen(map) > YPMAXMAP) {
(void) fprintf(stderr, err_bad_args,
err_bad_mapname);
exit(1);
}
}
break;
case 'd':
if (argc > 1) {
argv++;
argc--;
domain = *argv;
argv++;
if ((int)strlen(domain) > YPMAXDOMAIN) {
(void) fprintf(stderr, err_bad_args,
err_bad_domainname);
exit(1);
}
} else {
(void) fprintf(stderr, err_usage);
exit(1);
}
break;
case 't':
translate = FALSE;
argv++;
break;
case 'x':
dodump = TRUE;
argv++;
break;
default:
(void) fprintf(stderr, err_usage);
exit(1);
}
} else {
if (get_server) {
(void) fprintf(stderr, err_usage);
exit(1);
}
get_server = TRUE;
host = *argv;
argv++;
if ((int)strlen(host) > 256) {
(void) fprintf(stderr,
err_bad_args, err_bad_hostname);
exit(1);
}
}
}
if (get_master && get_server) {
(void) fprintf(stderr, err_usage);
exit(1);
}
if (!get_master && !get_server) {
get_server = TRUE;
}
}
static void
getdomain()
{
if (!getdomainname(default_domain_name, YPMAXDOMAIN)) {
domain = default_domain_name;
} else {
(void) fprintf(stderr, err_cant_get_kname, err_bad_domainname);
exit(1);
}
if ((int)strlen(domain) == 0) {
(void) fprintf(stderr, err_null_kname, err_bad_domainname);
exit(1);
}
}
static void
getlochost()
{
struct utsname utsname;
if (uname(&utsname) != -1) {
strcpy(default_host_name, utsname.nodename);
host = default_host_name;
} else {
(void) fprintf(stderr, err_cant_get_kname, err_bad_hostname);
exit(1);
}
}
static void
get_server_name()
{
char *notbound = "Domain %s not bound on %s.\n";
if (vers >= 3) {
if (!call_binder(vers))
(void) fprintf(stderr, notbound, domain, host);
} else {
if (!old_call_binder(vers))
(void) fprintf(stderr, notbound, domain, host);
}
}
extern CLIENT *__clnt_create_loopback();
static int
call_binder(vers)
int vers;
{
CLIENT *client;
struct ypbind_resp *response;
struct ypbind_domain ypbd;
char errstring[256];
extern struct rpc_createerr rpc_createerr;
int yperr = 0;
struct utsname utsname;
const char *str;
if ((uname(&utsname) != -1) &&
(strcmp(host, utsname.nodename) == 0))
client = __clnt_create_loopback(YPBINDPROG, vers, &yperr);
else
client = clnt_create(host, YPBINDPROG, vers, "netpath");
if (client == NULL) {
if (yperr)
(void) fprintf(stderr,
"ypwhich: %s\n", yperr_string(yperr));
else {
if (rpc_createerr.cf_stat == RPC_PROGNOTREGISTERED ||
rpc_createerr.cf_stat == RPC_PROGUNAVAIL) {
(void) fprintf(stderr,
"ypwhich: %s is not running ypbind\n", host);
} else if (rpc_createerr.cf_stat == RPC_PMAPFAILURE) {
(void) fprintf(stderr,
"ypwhich: %s is not running rpcbind\n",
host);
} else
(void) clnt_pcreateerror("ypwhich: \
clnt_create error");
}
exit(1);
}
ypbd.ypbind_domainname = domain;
ypbd.ypbind_vers = vers;
response = ypbindproc_domain_3(&ypbd, client);
if (response == NULL) {
(void) sprintf(errstring,
"ypwhich: can't call ypbind on %s", host);
(void) clnt_perror(client, errstring);
exit(1);
}
clnt_destroy(client);
if (response->ypbind_status != YPBIND_SUCC_VAL) {
return (FALSE);
}
if (response->ypbind_resp_u.ypbind_bindinfo) {
char *server =
response->ypbind_resp_u.ypbind_bindinfo->ypbind_servername;
if (strcmp(server, nullstring) == 0) {
struct nd_hostservlist *nhs = NULL;
struct netconfig *nconf =
response->ypbind_resp_u.ypbind_bindinfo->ypbind_nconf;
struct netbuf *svcaddr =
response->ypbind_resp_u.ypbind_bindinfo->ypbind_svcaddr;
if (netdir_getbyaddr(nconf, &nhs, svcaddr) != ND_OK) {
struct sockaddr_in *sa4;
struct sockaddr_in6 *sa6;
char buf[INET6_ADDRSTRLEN];
char xbuf[IPV6_ADDR_LEN];
int af;
void *addr;
XDR xdrs;
sa4 = (struct sockaddr_in *)svcaddr->buf;
af = ntohs(sa4->sin_family);
if (af != sa4->sin_family) {
xdrmem_create(&xdrs,
(caddr_t)xbuf, IPV6_ADDR_LEN,
XDR_DECODE);
if (af == AF_INET6) {
xdr_opaque(&xdrs,
(caddr_t)svcaddr->buf,
IPV6_ADDR_LEN);
sa6 = (struct sockaddr_in6 *)
xbuf;
addr = &sa6->sin6_addr;
} else {
xdr_opaque(&xdrs,
(caddr_t)svcaddr->buf,
IPV4_ADDR_LEN);
sa4 = (struct sockaddr_in *)
xbuf;
addr = &sa4->sin_addr;
}
} else {
if (af == AF_INET6) {
sa6 = (struct sockaddr_in6 *)
svcaddr->buf;
addr = &sa6->sin6_addr;
} else {
addr = &sa4->sin_addr;
}
}
str = inet_ntop(af, addr, buf, sizeof (buf));
if (str == NULL)
perror("inet_ntop");
else
fprintf(stdout, "%s\n", str);
} else {
str = nhs->h_hostservs->h_host;
if (str == NULL)
str = "<unknown>";
fprintf(stdout, "%s\n", str);
}
netdir_free((char *)nhs, ND_HOSTSERVLIST);
} else {
fprintf(stdout, "%s\n", server);
}
}
#ifdef DEBUG
dump_response(response);
#endif
return (TRUE);
}
static bool xdr_yp_inaddr(xdrs, ps)
XDR * xdrs;
struct in_addr *ps;
{
return (xdr_opaque(xdrs, (caddr_t)&ps->s_addr, 4));
}
static bool xdr_old_yp_binding(xdrs, ps)
XDR * xdrs;
struct old_ypbind_binding *ps;
{
return (xdr_yp_inaddr(xdrs, &ps->ypbind_binding_addr) &&
xdr_opaque(xdrs, (caddr_t)&ps->ypbind_binding_port, 2));
}
static bool xdr_old_ypbind_resp(xdrs, ps)
XDR * xdrs;
struct old_ypbind_resp *ps;
{
if (!xdr_enum(xdrs, (enum_t *)&ps->ypbind_status)) {
return (FALSE);
}
switch (ps->ypbind_status) {
case YPBIND_SUCC_VAL:
return (xdr_old_yp_binding(xdrs,
&ps->ypbind_respbody.ypbind_bindinfo));
case YPBIND_FAIL_VAL:
return (xdr_u_long(xdrs,
&ps->ypbind_respbody.ypbind_error));
}
return (FALSE);
}
static int old_call_binder(vers)
int vers;
{
CLIENT *client;
struct hostent *hp;
int sock = RPC_ANYSOCK;
enum clnt_stat rpc_stat;
struct old_ypbind_resp response;
char errstring[256];
extern struct rpc_createerr rpc_createerr;
struct in_addr *server;
if ((client = clnt_create(host, YPBINDPROG, vers, "udp")) == NULL) {
if (rpc_createerr.cf_stat == RPC_PROGNOTREGISTERED) {
(void) printf("ypwhich: %s is not running ypbind\n",
host);
exit(1);
}
if (rpc_createerr.cf_stat == RPC_PMAPFAILURE) {
(void) printf("ypwhich: %s is not running port mapper\n",
host);
exit(1);
}
(void) clnt_pcreateerror("ypwhich: clnt_create error");
exit(1);
}
rpc_stat = clnt_call(client, YPBINDPROC_DOMAIN,
(xdrproc_t)xdr_ypdomain_wrap_string, (caddr_t)&domain,
(xdrproc_t)xdr_old_ypbind_resp, (caddr_t)&response,
timeout);
if ((rpc_stat != RPC_SUCCESS) &&
(rpc_stat != RPC_PROGVERSMISMATCH)) {
(void) sprintf(errstring,
"ypwhich: can't call ypbind on %s", host);
(void) clnt_perror(client, errstring);
exit(1);
}
clnt_destroy(client);
close(sock);
if ((rpc_stat != RPC_SUCCESS) ||
(response.ypbind_status != YPBIND_SUCC_VAL)) {
return (FALSE);
}
server = &response.ypbind_respbody.ypbind_bindinfo.ypbind_binding_addr;
print_server (server);
return (TRUE);
}
static void print_server(server)
struct in_addr *server;
{
char buf[256];
struct hostent *hp;
strcpy(buf, inet_ntoa(*server));
hp = gethostbyaddr((char *)&server->s_addr,
sizeof (struct in_addr), AF_INET);
printf("%s\n", hp ? hp->h_name : buf);
}
#ifdef DEBUG
static void
dump_response(which)
ypbind_resp * which;
{
struct netconfig *nc;
struct netbuf *ua;
ypbind_binding * b;
int i;
{
b = which->ypbind_resp_u.ypbind_bindinfo;
if (b == NULL)
(void) fprintf(stderr, "???NO Binding information\n");
else {
(void) fprintf(stderr,
"server=%s lovers=%ld hivers=%ld\n",
b->ypbind_servername,
b->ypbind_lo_vers, b->ypbind_hi_vers);
nc = b->ypbind_nconf;
ua = b->ypbind_svcaddr;
if (nc == NULL)
(void) fprintf(stderr,
"ypwhich: NO netconfig information\n");
else {
(void) fprintf(stderr,
"ypwhich: id %s device %s flag %x protofmly %s proto %s\n",
nc->nc_netid, nc->nc_device,
(int)nc->nc_flag, nc->nc_protofmly,
nc->nc_proto);
}
if (ua == NULL)
(void) fprintf(stderr,
"ypwhich: NO netbuf information available from binder\n");
else {
(void) fprintf(stderr,
"maxlen=%d len=%d\naddr=", ua->maxlen, ua->len);
for (i = 0; i < ua->len; i++) {
if (i != (ua->len - 1))
(void) fprintf(stderr,
"%d.", ua->buf[i]);
else
(void) fprintf(stderr,
"%d\n", ua->buf[i]);
}
}
}
}
}
#endif
static void
get_map_master()
{
int err;
char *master;
err = __yp_master_rsvdport(domain, map, &master);
if (err) {
(void) fprintf(stderr,
"ypwhich: Can't find the master of %s. Reason: %s.\n",
map, yperr_string(err));
exit(1);
} else {
(void) printf("%s\n", master);
}
}
static void
dump_ypmaps()
{
int err;
struct dom_binding *binding;
if (err = __yp_dobind(domain, &binding)) {
(void) fprintf(stderr,
"dump_ypmaps: Can't bind for domain %s. Reason: %s\n",
domain, yperr_string(err));
return;
}
if (binding->dom_binding->ypbind_hi_vers >= YPVERS) {
dumpmaps(binding);
}
}
static void
dumpmaps(binding)
struct dom_binding *binding;
{
enum clnt_stat rpc_stat;
int err;
char *master;
struct ypmaplist *pmpl;
struct ypresp_maplist maplist;
maplist.list = (struct ypmaplist *)NULL;
rpc_stat = clnt_call(binding->dom_client, YPPROC_MAPLIST,
(xdrproc_t)xdr_ypdomain_wrap_string, (caddr_t)&domain,
(xdrproc_t)xdr_ypresp_maplist, (caddr_t)&maplist,
timeout);
if (rpc_stat != RPC_SUCCESS) {
(void) clnt_perror(binding->dom_client,
"ypwhich(dumpmaps): can't get maplist");
__yp_rel_binding(binding);
exit(1);
}
if (maplist.status != YP_TRUE) {
(void) fprintf(stderr,
"ypwhich: Can't get maplist. Reason: %s.\n",
yperr_string(ypprot_err(maplist.status)));
exit(1);
}
__yp_rel_binding(binding);
for (pmpl = maplist.list; pmpl; pmpl = pmpl->ypml_next) {
(void) printf("%s ", pmpl->ypml_name);
err = __yp_master_rsvdport(domain, pmpl->ypml_name, &master);
if (err) {
(void) printf("????????\n");
(void) fprintf(stderr,
"ypwhich: Can't find the master of %s. Reason: %s.\n",
pmpl->ypml_name, yperr_string(err));
} else {
(void) printf("%s\n", master);
}
}
}