19e7c127fSCraig Rodrigues /* $OpenBSD: yp.c,v 1.14 2015/02/11 01:26:00 pelikan Exp $ */
29e7c127fSCraig Rodrigues /*
39e7c127fSCraig Rodrigues * Copyright (c) 2008 Pierre-Yves Ritschard <pyr@openbsd.org>
49e7c127fSCraig Rodrigues *
59e7c127fSCraig Rodrigues * Permission to use, copy, modify, and distribute this software for any
69e7c127fSCraig Rodrigues * purpose with or without fee is hereby granted, provided that the above
79e7c127fSCraig Rodrigues * copyright notice and this permission notice appear in all copies.
89e7c127fSCraig Rodrigues *
99e7c127fSCraig Rodrigues * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
109e7c127fSCraig Rodrigues * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
119e7c127fSCraig Rodrigues * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
129e7c127fSCraig Rodrigues * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
139e7c127fSCraig Rodrigues * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
149e7c127fSCraig Rodrigues * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
159e7c127fSCraig Rodrigues * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
169e7c127fSCraig Rodrigues */
179e7c127fSCraig Rodrigues
189e7c127fSCraig Rodrigues #include <sys/types.h>
1966b5c05dSCraig Rodrigues #include <sys/param.h>
209e7c127fSCraig Rodrigues #include <sys/queue.h>
219e7c127fSCraig Rodrigues #include <sys/socket.h>
229e7c127fSCraig Rodrigues #include <sys/select.h>
239e7c127fSCraig Rodrigues #include <sys/tree.h>
249e7c127fSCraig Rodrigues
259e7c127fSCraig Rodrigues #include <netinet/in.h>
269e7c127fSCraig Rodrigues #include <arpa/inet.h>
279e7c127fSCraig Rodrigues
289e7c127fSCraig Rodrigues #include <errno.h>
299e7c127fSCraig Rodrigues #include <event.h>
309e7c127fSCraig Rodrigues #include <fcntl.h>
319e7c127fSCraig Rodrigues #include <unistd.h>
329e7c127fSCraig Rodrigues #include <pwd.h>
339e7c127fSCraig Rodrigues #include <stdio.h>
349e7c127fSCraig Rodrigues #include <stdlib.h>
359e7c127fSCraig Rodrigues #include <string.h>
369e7c127fSCraig Rodrigues #include <limits.h>
379e7c127fSCraig Rodrigues
389e7c127fSCraig Rodrigues #include <rpc/rpc.h>
399e7c127fSCraig Rodrigues #include <rpc/xdr.h>
409e7c127fSCraig Rodrigues #include <rpc/pmap_clnt.h>
419e7c127fSCraig Rodrigues #include <rpc/pmap_prot.h>
429e7c127fSCraig Rodrigues #include <rpc/pmap_rmt.h>
439e7c127fSCraig Rodrigues #include <rpcsvc/yp.h>
449e7c127fSCraig Rodrigues #include <rpcsvc/ypclnt.h>
459e7c127fSCraig Rodrigues
469e7c127fSCraig Rodrigues #include "ypldap.h"
479e7c127fSCraig Rodrigues
489e7c127fSCraig Rodrigues void yp_dispatch(struct svc_req *, SVCXPRT *);
499e7c127fSCraig Rodrigues void yp_disable_events(void);
509e7c127fSCraig Rodrigues void yp_fd_event(int, short, void *);
519e7c127fSCraig Rodrigues int yp_check(struct svc_req *);
529e7c127fSCraig Rodrigues int yp_valid_domain(char *, struct ypresp_val *);
539e7c127fSCraig Rodrigues void yp_make_val(struct ypresp_val *, char *, int);
549e7c127fSCraig Rodrigues void yp_make_keyval(struct ypresp_key_val *, char *, char *);
559e7c127fSCraig Rodrigues
569e7c127fSCraig Rodrigues static struct env *env;
579e7c127fSCraig Rodrigues
589e7c127fSCraig Rodrigues struct yp_event {
599e7c127fSCraig Rodrigues TAILQ_ENTRY(yp_event) ye_entry;
609e7c127fSCraig Rodrigues struct event ye_event;
619e7c127fSCraig Rodrigues };
629e7c127fSCraig Rodrigues
639e7c127fSCraig Rodrigues struct yp_data {
649e7c127fSCraig Rodrigues SVCXPRT *yp_trans_udp;
659e7c127fSCraig Rodrigues SVCXPRT *yp_trans_tcp;
669e7c127fSCraig Rodrigues TAILQ_HEAD(, yp_event) yd_events;
679e7c127fSCraig Rodrigues };
689e7c127fSCraig Rodrigues
699e7c127fSCraig Rodrigues void
yp_disable_events(void)709e7c127fSCraig Rodrigues yp_disable_events(void)
719e7c127fSCraig Rodrigues {
729e7c127fSCraig Rodrigues struct yp_event *ye;
739e7c127fSCraig Rodrigues
749e7c127fSCraig Rodrigues while ((ye = TAILQ_FIRST(&env->sc_yp->yd_events)) != NULL) {
759e7c127fSCraig Rodrigues TAILQ_REMOVE(&env->sc_yp->yd_events, ye, ye_entry);
769e7c127fSCraig Rodrigues event_del(&ye->ye_event);
779e7c127fSCraig Rodrigues free(ye);
789e7c127fSCraig Rodrigues }
799e7c127fSCraig Rodrigues }
809e7c127fSCraig Rodrigues
819e7c127fSCraig Rodrigues void
yp_enable_events(void)829e7c127fSCraig Rodrigues yp_enable_events(void)
839e7c127fSCraig Rodrigues {
849e7c127fSCraig Rodrigues int i;
859e7c127fSCraig Rodrigues struct yp_event *ye;
869e7c127fSCraig Rodrigues
87db891e21SMarcelo Araujo for (i = 0; i < getdtablesize(); i++) {
889e7c127fSCraig Rodrigues if ((ye = calloc(1, sizeof(*ye))) == NULL)
899e7c127fSCraig Rodrigues fatal(NULL);
909e7c127fSCraig Rodrigues event_set(&ye->ye_event, i, EV_READ, yp_fd_event, NULL);
919e7c127fSCraig Rodrigues event_add(&ye->ye_event, NULL);
929e7c127fSCraig Rodrigues TAILQ_INSERT_TAIL(&env->sc_yp->yd_events, ye, ye_entry);
939e7c127fSCraig Rodrigues }
949e7c127fSCraig Rodrigues }
959e7c127fSCraig Rodrigues
969e7c127fSCraig Rodrigues void
yp_fd_event(int fd,short event,void * p)979e7c127fSCraig Rodrigues yp_fd_event(int fd, short event, void *p)
989e7c127fSCraig Rodrigues {
999e7c127fSCraig Rodrigues svc_getreq_common(fd);
1009e7c127fSCraig Rodrigues yp_disable_events();
1019e7c127fSCraig Rodrigues yp_enable_events();
1029e7c127fSCraig Rodrigues }
1039e7c127fSCraig Rodrigues
1049e7c127fSCraig Rodrigues void
yp_init(struct env * x_env)1059e7c127fSCraig Rodrigues yp_init(struct env *x_env)
1069e7c127fSCraig Rodrigues {
1079e7c127fSCraig Rodrigues struct yp_data *yp;
1089e7c127fSCraig Rodrigues
1099e7c127fSCraig Rodrigues if ((yp = calloc(1, sizeof(*yp))) == NULL)
1109e7c127fSCraig Rodrigues fatal(NULL);
1119e7c127fSCraig Rodrigues TAILQ_INIT(&yp->yd_events);
1129e7c127fSCraig Rodrigues
1139e7c127fSCraig Rodrigues env = x_env;
1149e7c127fSCraig Rodrigues env->sc_yp = yp;
1159e7c127fSCraig Rodrigues
1169e7c127fSCraig Rodrigues (void)pmap_unset(YPPROG, YPVERS);
1179e7c127fSCraig Rodrigues
1189e7c127fSCraig Rodrigues if ((yp->yp_trans_udp = svcudp_create(RPC_ANYSOCK)) == NULL)
1199e7c127fSCraig Rodrigues fatal("cannot create udp service");
1209e7c127fSCraig Rodrigues if ((yp->yp_trans_tcp = svctcp_create(RPC_ANYSOCK, 0, 0)) == NULL)
1219e7c127fSCraig Rodrigues fatal("cannot create tcp service");
1229e7c127fSCraig Rodrigues
1239e7c127fSCraig Rodrigues if (!svc_register(yp->yp_trans_udp, YPPROG, YPVERS,
1249e7c127fSCraig Rodrigues yp_dispatch, IPPROTO_UDP)) {
1259e7c127fSCraig Rodrigues fatal("unable to register (YPPROG, YPVERS, udp)");
1269e7c127fSCraig Rodrigues }
1279e7c127fSCraig Rodrigues if (!svc_register(yp->yp_trans_tcp, YPPROG, YPVERS,
1289e7c127fSCraig Rodrigues yp_dispatch, IPPROTO_TCP)) {
1299e7c127fSCraig Rodrigues fatal("unable to register (YPPROG, YPVERS, tcp)");
1309e7c127fSCraig Rodrigues }
1319e7c127fSCraig Rodrigues }
1329e7c127fSCraig Rodrigues
1339e7c127fSCraig Rodrigues /*
1349e7c127fSCraig Rodrigues * lots of inspiration from ypserv by Mats O Jansson
1359e7c127fSCraig Rodrigues */
1369e7c127fSCraig Rodrigues void
yp_dispatch(struct svc_req * req,SVCXPRT * trans)1379e7c127fSCraig Rodrigues yp_dispatch(struct svc_req *req, SVCXPRT *trans)
1389e7c127fSCraig Rodrigues {
1399e7c127fSCraig Rodrigues xdrproc_t xdr_argument;
1409e7c127fSCraig Rodrigues xdrproc_t xdr_result;
1419e7c127fSCraig Rodrigues char *result;
1429e7c127fSCraig Rodrigues char *(*cb)(char *, struct svc_req *);
1439e7c127fSCraig Rodrigues union {
1449e7c127fSCraig Rodrigues domainname ypproc_domain_2_arg;
1459e7c127fSCraig Rodrigues domainname ypproc_domain_nonack_2_arg;
1469e7c127fSCraig Rodrigues ypreq_key ypproc_match_2_arg;
1479e7c127fSCraig Rodrigues ypreq_nokey ypproc_first_2_arg;
1489e7c127fSCraig Rodrigues ypreq_key ypproc_next_2_arg;
1499e7c127fSCraig Rodrigues ypreq_xfr ypproc_xfr_2_arg;
1509e7c127fSCraig Rodrigues ypreq_nokey ypproc_all_2_arg;
1519e7c127fSCraig Rodrigues ypreq_nokey ypproc_master_2_arg;
1529e7c127fSCraig Rodrigues ypreq_nokey ypproc_order_2_arg;
1539e7c127fSCraig Rodrigues domainname ypproc_maplist_2_arg;
1549e7c127fSCraig Rodrigues } argument;
1559e7c127fSCraig Rodrigues
1569e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_void;
1579e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_void;
1589e7c127fSCraig Rodrigues cb = NULL;
1599e7c127fSCraig Rodrigues switch (req->rq_proc) {
1609e7c127fSCraig Rodrigues case YPPROC_NULL:
1619e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_void;
1629e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_void;
1639e7c127fSCraig Rodrigues if (yp_check(req) == -1)
1649e7c127fSCraig Rodrigues return;
1659e7c127fSCraig Rodrigues result = NULL;
1669e7c127fSCraig Rodrigues if (!svc_sendreply(trans, (xdrproc_t) xdr_void,
1679e7c127fSCraig Rodrigues (void *)&result))
1689e7c127fSCraig Rodrigues svcerr_systemerr(trans);
1699e7c127fSCraig Rodrigues return;
1709e7c127fSCraig Rodrigues case YPPROC_DOMAIN:
1719e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_domainname;
1729e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_bool;
1739e7c127fSCraig Rodrigues if (yp_check(req) == -1)
1749e7c127fSCraig Rodrigues return;
1759e7c127fSCraig Rodrigues cb = (void *)ypproc_domain_2_svc;
1769e7c127fSCraig Rodrigues break;
1779e7c127fSCraig Rodrigues case YPPROC_DOMAIN_NONACK:
1789e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_domainname;
1799e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_bool;
1809e7c127fSCraig Rodrigues if (yp_check(req) == -1)
1819e7c127fSCraig Rodrigues return;
1829e7c127fSCraig Rodrigues cb = (void *)ypproc_domain_nonack_2_svc;
1839e7c127fSCraig Rodrigues break;
1849e7c127fSCraig Rodrigues case YPPROC_MATCH:
1859e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_ypreq_key;
1869e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_ypresp_val;
1879e7c127fSCraig Rodrigues if (yp_check(req) == -1)
1889e7c127fSCraig Rodrigues return;
1899e7c127fSCraig Rodrigues cb = (void *)ypproc_match_2_svc;
1909e7c127fSCraig Rodrigues break;
1919e7c127fSCraig Rodrigues case YPPROC_FIRST:
1929e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_ypreq_nokey;
1939e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_ypresp_key_val;
1949e7c127fSCraig Rodrigues if (yp_check(req) == -1)
1959e7c127fSCraig Rodrigues return;
1969e7c127fSCraig Rodrigues cb = (void *)ypproc_first_2_svc;
1979e7c127fSCraig Rodrigues break;
1989e7c127fSCraig Rodrigues case YPPROC_NEXT:
1999e7c127fSCraig Rodrigues xdr_argument = (xdrproc_t) xdr_ypreq_key;
2009e7c127fSCraig Rodrigues xdr_result = (xdrproc_t) xdr_ypresp_key_val;
2019e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2029e7c127fSCraig Rodrigues return;
2039e7c127fSCraig Rodrigues cb = (void *)ypproc_next_2_svc;
2049e7c127fSCraig Rodrigues break;
2059e7c127fSCraig Rodrigues case YPPROC_XFR:
2069e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2079e7c127fSCraig Rodrigues return;
2089e7c127fSCraig Rodrigues svcerr_noproc(trans);
2099e7c127fSCraig Rodrigues return;
2109e7c127fSCraig Rodrigues case YPPROC_CLEAR:
2119e7c127fSCraig Rodrigues log_debug("ypproc_clear");
2129e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2139e7c127fSCraig Rodrigues return;
2149e7c127fSCraig Rodrigues svcerr_noproc(trans);
2159e7c127fSCraig Rodrigues return;
2169e7c127fSCraig Rodrigues case YPPROC_ALL:
2179e7c127fSCraig Rodrigues log_debug("ypproc_all");
21857074683SMarcelo Araujo xdr_argument = (xdrproc_t) xdr_ypreq_nokey;
21957074683SMarcelo Araujo xdr_result = (xdrproc_t) xdr_ypresp_all;
2209e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2219e7c127fSCraig Rodrigues return;
2229e7c127fSCraig Rodrigues cb = (void *)ypproc_all_2_svc;
2239e7c127fSCraig Rodrigues break;
2249e7c127fSCraig Rodrigues case YPPROC_MASTER:
2259e7c127fSCraig Rodrigues log_debug("ypproc_master");
226619d20a1SMarcelo Araujo xdr_argument = (xdrproc_t) xdr_ypreq_nokey;
227619d20a1SMarcelo Araujo xdr_result = (xdrproc_t) xdr_ypresp_master;
2289e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2299e7c127fSCraig Rodrigues return;
2309e7c127fSCraig Rodrigues cb = (void *)ypproc_master_2_svc;
2319e7c127fSCraig Rodrigues break;
2329e7c127fSCraig Rodrigues case YPPROC_ORDER:
2339e7c127fSCraig Rodrigues log_debug("ypproc_order");
2349e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2359e7c127fSCraig Rodrigues return;
2369e7c127fSCraig Rodrigues svcerr_noproc(trans);
2379e7c127fSCraig Rodrigues return;
2389e7c127fSCraig Rodrigues case YPPROC_MAPLIST:
2399e7c127fSCraig Rodrigues log_debug("ypproc_maplist");
24057074683SMarcelo Araujo xdr_argument = (xdrproc_t) xdr_domainname;
24157074683SMarcelo Araujo xdr_result = (xdrproc_t) xdr_ypresp_maplist;
2429e7c127fSCraig Rodrigues if (yp_check(req) == -1)
2439e7c127fSCraig Rodrigues return;
2449e7c127fSCraig Rodrigues cb = (void *)ypproc_maplist_2_svc;
2459e7c127fSCraig Rodrigues break;
2469e7c127fSCraig Rodrigues default:
2479e7c127fSCraig Rodrigues svcerr_noproc(trans);
2489e7c127fSCraig Rodrigues return;
2499e7c127fSCraig Rodrigues }
2509e7c127fSCraig Rodrigues (void)memset(&argument, 0, sizeof(argument));
2519e7c127fSCraig Rodrigues
2529e7c127fSCraig Rodrigues if (!svc_getargs(trans, xdr_argument, (caddr_t)&argument)) {
2539e7c127fSCraig Rodrigues svcerr_decode(trans);
2549e7c127fSCraig Rodrigues return;
2559e7c127fSCraig Rodrigues }
2569e7c127fSCraig Rodrigues result = (*cb)((char *)&argument, req);
2579e7c127fSCraig Rodrigues if (result != NULL && !svc_sendreply(trans, xdr_result, result))
2589e7c127fSCraig Rodrigues svcerr_systemerr(trans);
2599e7c127fSCraig Rodrigues if (!svc_freeargs(trans, xdr_argument, (caddr_t)&argument)) {
2609e7c127fSCraig Rodrigues /*
2619e7c127fSCraig Rodrigues * ypserv does it too.
2629e7c127fSCraig Rodrigues */
2639e7c127fSCraig Rodrigues fatal("unable to free arguments");
2649e7c127fSCraig Rodrigues }
2659e7c127fSCraig Rodrigues }
2669e7c127fSCraig Rodrigues
2679e7c127fSCraig Rodrigues int
yp_check(struct svc_req * req)2689e7c127fSCraig Rodrigues yp_check(struct svc_req *req)
2699e7c127fSCraig Rodrigues {
2709e7c127fSCraig Rodrigues /*
2719e7c127fSCraig Rodrigues * We might want to know who we allow here.
2729e7c127fSCraig Rodrigues */
2739e7c127fSCraig Rodrigues return (0);
2749e7c127fSCraig Rodrigues }
2759e7c127fSCraig Rodrigues
2769e7c127fSCraig Rodrigues int
yp_valid_domain(char * domain,struct ypresp_val * res)2779e7c127fSCraig Rodrigues yp_valid_domain(char *domain, struct ypresp_val *res)
2789e7c127fSCraig Rodrigues {
2799e7c127fSCraig Rodrigues if (domain == NULL) {
2809e7c127fSCraig Rodrigues log_debug("NULL domain !");
2819e7c127fSCraig Rodrigues return (-1);
2829e7c127fSCraig Rodrigues }
2839e7c127fSCraig Rodrigues if (strcmp(domain, env->sc_domainname) != 0) {
2849e7c127fSCraig Rodrigues res->stat = YP_NODOM;
2859e7c127fSCraig Rodrigues return (-1);
2869e7c127fSCraig Rodrigues }
2879e7c127fSCraig Rodrigues return (0);
2889e7c127fSCraig Rodrigues }
2899e7c127fSCraig Rodrigues
2909e7c127fSCraig Rodrigues bool_t *
ypproc_domain_2_svc(domainname * arg,struct svc_req * req)2919e7c127fSCraig Rodrigues ypproc_domain_2_svc(domainname *arg, struct svc_req *req)
2929e7c127fSCraig Rodrigues {
2939e7c127fSCraig Rodrigues static bool_t res;
2949e7c127fSCraig Rodrigues
2959e7c127fSCraig Rodrigues res = (bool_t)1;
2969e7c127fSCraig Rodrigues if (strcmp(*arg, env->sc_domainname) != 0)
2979e7c127fSCraig Rodrigues res = (bool_t)0;
2989e7c127fSCraig Rodrigues return (&res);
2999e7c127fSCraig Rodrigues }
3009e7c127fSCraig Rodrigues
3019e7c127fSCraig Rodrigues bool_t *
ypproc_domain_nonack_2_svc(domainname * arg,struct svc_req * req)3029e7c127fSCraig Rodrigues ypproc_domain_nonack_2_svc(domainname *arg, struct svc_req *req)
3039e7c127fSCraig Rodrigues {
3049e7c127fSCraig Rodrigues static bool_t res;
3059e7c127fSCraig Rodrigues
3069e7c127fSCraig Rodrigues if (strcmp(*arg, env->sc_domainname) != 0)
3079e7c127fSCraig Rodrigues return NULL;
3089e7c127fSCraig Rodrigues res = (bool_t)1;
3099e7c127fSCraig Rodrigues return (&res);
3109e7c127fSCraig Rodrigues }
3119e7c127fSCraig Rodrigues
3129e7c127fSCraig Rodrigues ypresp_val *
ypproc_match_2_svc(ypreq_key * arg,struct svc_req * req)3139e7c127fSCraig Rodrigues ypproc_match_2_svc(ypreq_key *arg, struct svc_req *req)
3149e7c127fSCraig Rodrigues {
3159e7c127fSCraig Rodrigues struct userent ukey;
3169e7c127fSCraig Rodrigues struct userent *ue;
3179e7c127fSCraig Rodrigues struct groupent gkey;
3189e7c127fSCraig Rodrigues struct groupent *ge;
3199e7c127fSCraig Rodrigues static struct ypresp_val res;
3209e7c127fSCraig Rodrigues const char *estr;
3219e7c127fSCraig Rodrigues char *bp, *cp;
322*6b462d27SKonstantin Belousov char *key;
3239e7c127fSCraig Rodrigues
3249e7c127fSCraig Rodrigues log_debug("matching '%.*s' in map %s", arg->key.keydat_len,
3259e7c127fSCraig Rodrigues arg->key.keydat_val, arg->map);
3269e7c127fSCraig Rodrigues
3279e7c127fSCraig Rodrigues if (yp_valid_domain(arg->domain, (struct ypresp_val *)&res) == -1)
3289e7c127fSCraig Rodrigues return (&res);
3299e7c127fSCraig Rodrigues
3309e7c127fSCraig Rodrigues if (env->sc_user_names == NULL) {
3319e7c127fSCraig Rodrigues /*
3329e7c127fSCraig Rodrigues * tree not ready.
3339e7c127fSCraig Rodrigues */
3349e7c127fSCraig Rodrigues return (NULL);
3359e7c127fSCraig Rodrigues }
3369e7c127fSCraig Rodrigues
3379e7c127fSCraig Rodrigues if (arg->key.keydat_len > YPMAXRECORD) {
3389e7c127fSCraig Rodrigues log_debug("argument too long");
3399e7c127fSCraig Rodrigues return (NULL);
3409e7c127fSCraig Rodrigues }
341*6b462d27SKonstantin Belousov key = calloc(arg->key.keydat_len + 1, 1);
342*6b462d27SKonstantin Belousov if (key == NULL)
343*6b462d27SKonstantin Belousov return (NULL);
3449e7c127fSCraig Rodrigues (void)strncpy(key, arg->key.keydat_val, arg->key.keydat_len);
3459e7c127fSCraig Rodrigues
3469e7c127fSCraig Rodrigues if (strcmp(arg->map, "passwd.byname") == 0 ||
3479e7c127fSCraig Rodrigues strcmp(arg->map, "master.passwd.byname") == 0) {
3489e7c127fSCraig Rodrigues ukey.ue_line = key;
3499e7c127fSCraig Rodrigues if ((ue = RB_FIND(user_name_tree, env->sc_user_names,
3509e7c127fSCraig Rodrigues &ukey)) == NULL) {
3519e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
352*6b462d27SKonstantin Belousov goto out;
3539e7c127fSCraig Rodrigues }
3549e7c127fSCraig Rodrigues
3559e7c127fSCraig Rodrigues yp_make_val(&res, ue->ue_line, 1);
356*6b462d27SKonstantin Belousov goto out;
3579e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "passwd.byuid") == 0 ||
3589e7c127fSCraig Rodrigues strcmp(arg->map, "master.passwd.byuid") == 0) {
3599e7c127fSCraig Rodrigues ukey.ue_uid = strtonum(key, 0, UID_MAX, &estr);
3609e7c127fSCraig Rodrigues if (estr) {
3619e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
362*6b462d27SKonstantin Belousov goto out;
3639e7c127fSCraig Rodrigues }
3649e7c127fSCraig Rodrigues
3659e7c127fSCraig Rodrigues if ((ue = RB_FIND(user_uid_tree, &env->sc_user_uids,
3669e7c127fSCraig Rodrigues &ukey)) == NULL) {
3679e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
368*6b462d27SKonstantin Belousov goto out;
3699e7c127fSCraig Rodrigues }
3709e7c127fSCraig Rodrigues
3719e7c127fSCraig Rodrigues yp_make_val(&res, ue->ue_line, 1);
3729e7c127fSCraig Rodrigues return (&res);
3739e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "group.bygid") == 0) {
3749e7c127fSCraig Rodrigues gkey.ge_gid = strtonum(key, 0, GID_MAX, &estr);
3759e7c127fSCraig Rodrigues if (estr) {
3769e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
377*6b462d27SKonstantin Belousov goto out;
3789e7c127fSCraig Rodrigues }
3799e7c127fSCraig Rodrigues if ((ge = RB_FIND(group_gid_tree, &env->sc_group_gids,
3809e7c127fSCraig Rodrigues &gkey)) == NULL) {
3819e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
382*6b462d27SKonstantin Belousov goto out;
3839e7c127fSCraig Rodrigues }
3849e7c127fSCraig Rodrigues
3859e7c127fSCraig Rodrigues yp_make_val(&res, ge->ge_line, 1);
3869e7c127fSCraig Rodrigues return (&res);
3879e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "group.byname") == 0) {
3889e7c127fSCraig Rodrigues gkey.ge_line = key;
3899e7c127fSCraig Rodrigues if ((ge = RB_FIND(group_name_tree, env->sc_group_names,
3909e7c127fSCraig Rodrigues &gkey)) == NULL) {
3919e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
392*6b462d27SKonstantin Belousov goto out;
3939e7c127fSCraig Rodrigues }
3949e7c127fSCraig Rodrigues
3959e7c127fSCraig Rodrigues yp_make_val(&res, ge->ge_line, 1);
3969e7c127fSCraig Rodrigues return (&res);
3979e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "netid.byname") == 0) {
3989e7c127fSCraig Rodrigues bp = cp = key;
3999e7c127fSCraig Rodrigues
4009e7c127fSCraig Rodrigues if (strncmp(bp, "unix.", strlen("unix.")) != 0) {
4019e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
402*6b462d27SKonstantin Belousov goto out;
4039e7c127fSCraig Rodrigues }
4049e7c127fSCraig Rodrigues
4059e7c127fSCraig Rodrigues bp += strlen("unix.");
4069e7c127fSCraig Rodrigues
4079e7c127fSCraig Rodrigues if (*bp == '\0') {
4089e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
409*6b462d27SKonstantin Belousov goto out;
4109e7c127fSCraig Rodrigues }
4119e7c127fSCraig Rodrigues
4129e7c127fSCraig Rodrigues if (!(cp = strsep(&bp, "@"))) {
4139e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
414*6b462d27SKonstantin Belousov goto out;
4159e7c127fSCraig Rodrigues }
4169e7c127fSCraig Rodrigues
4179e7c127fSCraig Rodrigues if (strcmp(bp, arg->domain) != 0) {
4189e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
419*6b462d27SKonstantin Belousov goto out;
4209e7c127fSCraig Rodrigues }
4219e7c127fSCraig Rodrigues
4229e7c127fSCraig Rodrigues ukey.ue_uid = strtonum(cp, 0, UID_MAX, &estr);
4239e7c127fSCraig Rodrigues if (estr) {
4249e7c127fSCraig Rodrigues res.stat = YP_BADARGS;
425*6b462d27SKonstantin Belousov goto out;
4269e7c127fSCraig Rodrigues }
4279e7c127fSCraig Rodrigues
4289e7c127fSCraig Rodrigues if ((ue = RB_FIND(user_uid_tree, &env->sc_user_uids,
4299e7c127fSCraig Rodrigues &ukey)) == NULL) {
4309e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
431*6b462d27SKonstantin Belousov goto out;
4329e7c127fSCraig Rodrigues }
4339e7c127fSCraig Rodrigues
4349e7c127fSCraig Rodrigues yp_make_val(&res, ue->ue_netid_line, 0);
435*6b462d27SKonstantin Belousov goto out;
4369e7c127fSCraig Rodrigues
4379e7c127fSCraig Rodrigues } else {
4389e7c127fSCraig Rodrigues log_debug("unknown map %s", arg->map);
4399e7c127fSCraig Rodrigues res.stat = YP_NOMAP;
440*6b462d27SKonstantin Belousov goto out;
4419e7c127fSCraig Rodrigues }
442*6b462d27SKonstantin Belousov out:
443*6b462d27SKonstantin Belousov free(key);
444*6b462d27SKonstantin Belousov return (&res);
4459e7c127fSCraig Rodrigues }
4469e7c127fSCraig Rodrigues
4479e7c127fSCraig Rodrigues ypresp_key_val *
ypproc_first_2_svc(ypreq_nokey * arg,struct svc_req * req)4489e7c127fSCraig Rodrigues ypproc_first_2_svc(ypreq_nokey *arg, struct svc_req *req)
4499e7c127fSCraig Rodrigues {
4509e7c127fSCraig Rodrigues static struct ypresp_key_val res;
4519e7c127fSCraig Rodrigues
4529e7c127fSCraig Rodrigues if (yp_valid_domain(arg->domain, (struct ypresp_val *)&res) == -1)
4539e7c127fSCraig Rodrigues return (&res);
4549e7c127fSCraig Rodrigues
4559e7c127fSCraig Rodrigues if (strcmp(arg->map, "passwd.byname") == 0 ||
4569e7c127fSCraig Rodrigues strcmp(arg->map, "master.passwd.byname") == 0) {
4579e7c127fSCraig Rodrigues if (env->sc_user_lines == NULL)
4589e7c127fSCraig Rodrigues return (NULL);
4599e7c127fSCraig Rodrigues
4609e7c127fSCraig Rodrigues yp_make_keyval(&res, env->sc_user_lines, env->sc_user_lines);
4619e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "group.byname") == 0) {
4629e7c127fSCraig Rodrigues if (env->sc_group_lines == NULL)
4639e7c127fSCraig Rodrigues return (NULL);
4649e7c127fSCraig Rodrigues
4659e7c127fSCraig Rodrigues yp_make_keyval(&res, env->sc_group_lines, env->sc_group_lines);
4669e7c127fSCraig Rodrigues } else {
4679e7c127fSCraig Rodrigues log_debug("unknown map %s", arg->map);
4689e7c127fSCraig Rodrigues res.stat = YP_NOMAP;
4699e7c127fSCraig Rodrigues }
4709e7c127fSCraig Rodrigues
4719e7c127fSCraig Rodrigues return (&res);
4729e7c127fSCraig Rodrigues }
4739e7c127fSCraig Rodrigues
4749e7c127fSCraig Rodrigues ypresp_key_val *
ypproc_next_2_svc(ypreq_key * arg,struct svc_req * req)4759e7c127fSCraig Rodrigues ypproc_next_2_svc(ypreq_key *arg, struct svc_req *req)
4769e7c127fSCraig Rodrigues {
4779e7c127fSCraig Rodrigues struct userent ukey;
4789e7c127fSCraig Rodrigues struct userent *ue;
4799e7c127fSCraig Rodrigues struct groupent gkey;
4809e7c127fSCraig Rodrigues struct groupent *ge;
4819e7c127fSCraig Rodrigues char *line;
4829e7c127fSCraig Rodrigues static struct ypresp_key_val res;
483*6b462d27SKonstantin Belousov char *key;
4849e7c127fSCraig Rodrigues
4859e7c127fSCraig Rodrigues if (yp_valid_domain(arg->domain, (struct ypresp_val *)&res) == -1)
4869e7c127fSCraig Rodrigues return (&res);
4879e7c127fSCraig Rodrigues
488*6b462d27SKonstantin Belousov key = NULL;
4899e7c127fSCraig Rodrigues if (strcmp(arg->map, "passwd.byname") == 0 ||
4909e7c127fSCraig Rodrigues strcmp(arg->map, "master.passwd.byname") == 0) {
491*6b462d27SKonstantin Belousov key = calloc(arg->key.keydat_len + 1, 1);
492*6b462d27SKonstantin Belousov if (key == NULL) {
493*6b462d27SKonstantin Belousov res.stat = YP_YPERR;
494*6b462d27SKonstantin Belousov return (&res);
495*6b462d27SKonstantin Belousov }
4969e7c127fSCraig Rodrigues (void)strncpy(key, arg->key.keydat_val,
4979e7c127fSCraig Rodrigues arg->key.keydat_len);
4989e7c127fSCraig Rodrigues ukey.ue_line = key;
4999e7c127fSCraig Rodrigues if ((ue = RB_FIND(user_name_tree, env->sc_user_names,
5009e7c127fSCraig Rodrigues &ukey)) == NULL) {
5019e7c127fSCraig Rodrigues /*
5029e7c127fSCraig Rodrigues * canacar's trick:
5039e7c127fSCraig Rodrigues * the user might have been deleted in between calls
5049e7c127fSCraig Rodrigues * to next since the tree may be modified by a reload.
5059e7c127fSCraig Rodrigues * next should still return the next user in
5069e7c127fSCraig Rodrigues * lexicographical order, hence insert the search key
5079e7c127fSCraig Rodrigues * and look up the next field, then remove it again.
5089e7c127fSCraig Rodrigues */
5099e7c127fSCraig Rodrigues RB_INSERT(user_name_tree, env->sc_user_names, &ukey);
5109e7c127fSCraig Rodrigues if ((ue = RB_NEXT(user_name_tree, &env->sc_user_names,
5119e7c127fSCraig Rodrigues &ukey)) == NULL) {
5129e7c127fSCraig Rodrigues RB_REMOVE(user_name_tree, env->sc_user_names,
5139e7c127fSCraig Rodrigues &ukey);
5149e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
515*6b462d27SKonstantin Belousov free(key);
5169e7c127fSCraig Rodrigues return (&res);
5179e7c127fSCraig Rodrigues }
5189e7c127fSCraig Rodrigues RB_REMOVE(user_name_tree, env->sc_user_names, &ukey);
5199e7c127fSCraig Rodrigues }
5209e7c127fSCraig Rodrigues line = ue->ue_line + (strlen(ue->ue_line) + 1);
5219e7c127fSCraig Rodrigues line = line + (strlen(line) + 1);
5229e7c127fSCraig Rodrigues yp_make_keyval(&res, line, line);
523*6b462d27SKonstantin Belousov free(key);
5249e7c127fSCraig Rodrigues return (&res);
5259e7c127fSCraig Rodrigues
5269e7c127fSCraig Rodrigues
5279e7c127fSCraig Rodrigues } else if (strcmp(arg->map, "group.byname") == 0) {
528*6b462d27SKonstantin Belousov key = calloc(arg->key.keydat_len + 1, 1);
529*6b462d27SKonstantin Belousov if (key == NULL) {
530*6b462d27SKonstantin Belousov res.stat = YP_YPERR;
531*6b462d27SKonstantin Belousov return (&res);
532*6b462d27SKonstantin Belousov }
5339e7c127fSCraig Rodrigues (void)strncpy(key, arg->key.keydat_val,
5349e7c127fSCraig Rodrigues arg->key.keydat_len);
5359e7c127fSCraig Rodrigues
5369e7c127fSCraig Rodrigues gkey.ge_line = key;
5379e7c127fSCraig Rodrigues if ((ge = RB_FIND(group_name_tree, env->sc_group_names,
5389e7c127fSCraig Rodrigues &gkey)) == NULL) {
5399e7c127fSCraig Rodrigues /*
5409e7c127fSCraig Rodrigues * canacar's trick reloaded.
5419e7c127fSCraig Rodrigues */
5429e7c127fSCraig Rodrigues RB_INSERT(group_name_tree, env->sc_group_names, &gkey);
5439e7c127fSCraig Rodrigues if ((ge = RB_NEXT(group_name_tree, &env->sc_group_names,
5449e7c127fSCraig Rodrigues &gkey)) == NULL) {
5459e7c127fSCraig Rodrigues RB_REMOVE(group_name_tree, env->sc_group_names,
5469e7c127fSCraig Rodrigues &gkey);
5479e7c127fSCraig Rodrigues res.stat = YP_NOKEY;
548*6b462d27SKonstantin Belousov free(key);
5499e7c127fSCraig Rodrigues return (&res);
5509e7c127fSCraig Rodrigues }
5519e7c127fSCraig Rodrigues RB_REMOVE(group_name_tree, env->sc_group_names, &gkey);
5529e7c127fSCraig Rodrigues }
5539e7c127fSCraig Rodrigues
5549e7c127fSCraig Rodrigues line = ge->ge_line + (strlen(ge->ge_line) + 1);
5559e7c127fSCraig Rodrigues line = line + (strlen(line) + 1);
5569e7c127fSCraig Rodrigues yp_make_keyval(&res, line, line);
557*6b462d27SKonstantin Belousov free(key);
5589e7c127fSCraig Rodrigues return (&res);
5599e7c127fSCraig Rodrigues } else {
5609e7c127fSCraig Rodrigues log_debug("unknown map %s", arg->map);
5619e7c127fSCraig Rodrigues res.stat = YP_NOMAP;
5629e7c127fSCraig Rodrigues return (&res);
5639e7c127fSCraig Rodrigues }
5649e7c127fSCraig Rodrigues }
5659e7c127fSCraig Rodrigues
5669e7c127fSCraig Rodrigues ypresp_all *
ypproc_all_2_svc(ypreq_nokey * arg,struct svc_req * req)5679e7c127fSCraig Rodrigues ypproc_all_2_svc(ypreq_nokey *arg, struct svc_req *req)
5689e7c127fSCraig Rodrigues {
5699e7c127fSCraig Rodrigues static struct ypresp_all res;
5709e7c127fSCraig Rodrigues
5719e7c127fSCraig Rodrigues if (yp_valid_domain(arg->domain, (struct ypresp_val *)&res) == -1)
5729e7c127fSCraig Rodrigues return (&res);
5739e7c127fSCraig Rodrigues
5749e7c127fSCraig Rodrigues svcerr_auth(req->rq_xprt, AUTH_FAILED);
5759e7c127fSCraig Rodrigues return (NULL);
5769e7c127fSCraig Rodrigues }
5779e7c127fSCraig Rodrigues
5789e7c127fSCraig Rodrigues ypresp_master *
ypproc_master_2_svc(ypreq_nokey * arg,struct svc_req * req)5799e7c127fSCraig Rodrigues ypproc_master_2_svc(ypreq_nokey *arg, struct svc_req *req)
5809e7c127fSCraig Rodrigues {
5819e7c127fSCraig Rodrigues static struct ypresp_master res;
582619d20a1SMarcelo Araujo static char master[YPMAXPEER + 1];
5839e7c127fSCraig Rodrigues
584bbdfc8f1SMarcelo Araujo memset(&res, 0, sizeof(res));
5859e7c127fSCraig Rodrigues if (yp_valid_domain(arg->domain, (struct ypresp_val *)&res) == -1)
5869e7c127fSCraig Rodrigues return (&res);
5879e7c127fSCraig Rodrigues
588619d20a1SMarcelo Araujo if (gethostname(master, sizeof(master)) == 0) {
589619d20a1SMarcelo Araujo res.peer = (peername)master;
590619d20a1SMarcelo Araujo res.stat = YP_TRUE;
591619d20a1SMarcelo Araujo } else
592619d20a1SMarcelo Araujo res.stat = YP_NOKEY;
593619d20a1SMarcelo Araujo
5949e7c127fSCraig Rodrigues return (&res);
5959e7c127fSCraig Rodrigues }
5969e7c127fSCraig Rodrigues
5979e7c127fSCraig Rodrigues ypresp_maplist *
ypproc_maplist_2_svc(domainname * arg,struct svc_req * req)5989e7c127fSCraig Rodrigues ypproc_maplist_2_svc(domainname *arg, struct svc_req *req)
5999e7c127fSCraig Rodrigues {
6009e7c127fSCraig Rodrigues size_t i;
6019e7c127fSCraig Rodrigues static struct {
6029e7c127fSCraig Rodrigues char *name;
6039e7c127fSCraig Rodrigues int cond;
6049e7c127fSCraig Rodrigues } mapnames[] = {
6059e7c127fSCraig Rodrigues { "passwd.byname", YPMAP_PASSWD_BYNAME },
6069e7c127fSCraig Rodrigues { "passwd.byuid", YPMAP_PASSWD_BYUID },
6079e7c127fSCraig Rodrigues { "master.passwd.byname", YPMAP_MASTER_PASSWD_BYNAME },
6089e7c127fSCraig Rodrigues { "master.passwd.byuid", YPMAP_MASTER_PASSWD_BYUID },
6099e7c127fSCraig Rodrigues { "group.byname", YPMAP_GROUP_BYNAME },
6109e7c127fSCraig Rodrigues { "group.bygid", YPMAP_GROUP_BYGID },
6119e7c127fSCraig Rodrigues { "netid.byname", YPMAP_NETID_BYNAME },
6129e7c127fSCraig Rodrigues };
6139e7c127fSCraig Rodrigues static ypresp_maplist res;
6146a35cae5SMarcelo Araujo static struct ypmaplist maps[nitems(mapnames)];
6159e7c127fSCraig Rodrigues
6169e7c127fSCraig Rodrigues if (yp_valid_domain(*arg, (struct ypresp_val *)&res) == -1)
6179e7c127fSCraig Rodrigues return (&res);
6189e7c127fSCraig Rodrigues
6199e7c127fSCraig Rodrigues res.stat = YP_TRUE;
6209e7c127fSCraig Rodrigues res.maps = NULL;
6216a35cae5SMarcelo Araujo for (i = 0; i < nitems(mapnames); i++) {
6229e7c127fSCraig Rodrigues if (!(env->sc_flags & mapnames[i].cond))
6239e7c127fSCraig Rodrigues continue;
6249e7c127fSCraig Rodrigues maps[i].map = mapnames[i].name;
6259e7c127fSCraig Rodrigues maps[i].next = res.maps;
6269e7c127fSCraig Rodrigues res.maps = &maps[i];
6279e7c127fSCraig Rodrigues }
6289e7c127fSCraig Rodrigues
6299e7c127fSCraig Rodrigues return (&res);
6309e7c127fSCraig Rodrigues }
6319e7c127fSCraig Rodrigues
6329e7c127fSCraig Rodrigues void
yp_make_val(struct ypresp_val * res,char * line,int replacecolon)6339e7c127fSCraig Rodrigues yp_make_val(struct ypresp_val *res, char *line, int replacecolon)
6349e7c127fSCraig Rodrigues {
6359e7c127fSCraig Rodrigues static char buf[LINE_WIDTH];
6369e7c127fSCraig Rodrigues
637bbdfc8f1SMarcelo Araujo memset(buf, 0, sizeof(buf));
6389e7c127fSCraig Rodrigues
6399e7c127fSCraig Rodrigues if (replacecolon)
6409e7c127fSCraig Rodrigues line[strlen(line)] = ':';
6419e7c127fSCraig Rodrigues (void)strlcpy(buf, line, sizeof(buf));
6429e7c127fSCraig Rodrigues if (replacecolon)
6439e7c127fSCraig Rodrigues line[strcspn(line, ":")] = '\0';
6449e7c127fSCraig Rodrigues log_debug("sending out %s", buf);
6459e7c127fSCraig Rodrigues
6469e7c127fSCraig Rodrigues res->stat = YP_TRUE;
6479e7c127fSCraig Rodrigues res->val.valdat_len = strlen(buf);
6489e7c127fSCraig Rodrigues res->val.valdat_val = buf;
6499e7c127fSCraig Rodrigues }
6509e7c127fSCraig Rodrigues
6519e7c127fSCraig Rodrigues void
yp_make_keyval(struct ypresp_key_val * res,char * key,char * line)6529e7c127fSCraig Rodrigues yp_make_keyval(struct ypresp_key_val *res, char *key, char *line)
6539e7c127fSCraig Rodrigues {
6549e7c127fSCraig Rodrigues static char keybuf[YPMAXRECORD+1];
6559e7c127fSCraig Rodrigues static char buf[LINE_WIDTH];
6569e7c127fSCraig Rodrigues
657bbdfc8f1SMarcelo Araujo memset(keybuf, 0, sizeof(keybuf));
658bbdfc8f1SMarcelo Araujo memset(buf, 0, sizeof(buf));
6599e7c127fSCraig Rodrigues
6609e7c127fSCraig Rodrigues (void)strlcpy(keybuf, key, sizeof(keybuf));
6619e7c127fSCraig Rodrigues res->key.keydat_len = strlen(keybuf);
6629e7c127fSCraig Rodrigues res->key.keydat_val = keybuf;
6639e7c127fSCraig Rodrigues
6649e7c127fSCraig Rodrigues if (*line == '\0') {
6659e7c127fSCraig Rodrigues res->stat = YP_NOMORE;
6669e7c127fSCraig Rodrigues return;
6679e7c127fSCraig Rodrigues }
6689e7c127fSCraig Rodrigues res->stat = YP_TRUE;
6699e7c127fSCraig Rodrigues line[strlen(line)] = ':';
6709e7c127fSCraig Rodrigues (void)strlcpy(buf, line, sizeof(buf));
6719e7c127fSCraig Rodrigues line[strcspn(line, ":")] = '\0';
6729e7c127fSCraig Rodrigues log_debug("sending out %s => %s", keybuf, buf);
6739e7c127fSCraig Rodrigues
6749e7c127fSCraig Rodrigues res->val.valdat_len = strlen(buf);
6759e7c127fSCraig Rodrigues res->val.valdat_val = buf;
6769e7c127fSCraig Rodrigues }
677