xref: /openbsd-src/lib/libc/yp/ypmatch_cache.c (revision 2d62dde8310c0f11b45f14adaba36cf1950be6bf)
1*2d62dde8Sderaadt /*	$OpenBSD: ypmatch_cache.c,v 1.18 2022/08/02 16:59:30 deraadt Exp $ */
218128546Sderaadt /*
318128546Sderaadt  * Copyright (c) 1992, 1993 Theo de Raadt <deraadt@theos.com>
418128546Sderaadt  * All rights reserved.
518128546Sderaadt  *
618128546Sderaadt  * Redistribution and use in source and binary forms, with or without
718128546Sderaadt  * modification, are permitted provided that the following conditions
818128546Sderaadt  * are met:
918128546Sderaadt  * 1. Redistributions of source code must retain the above copyright
1018128546Sderaadt  *    notice, this list of conditions and the following disclaimer.
1118128546Sderaadt  * 2. Redistributions in binary form must reproduce the above copyright
1218128546Sderaadt  *    notice, this list of conditions and the following disclaimer in the
1318128546Sderaadt  *    documentation and/or other materials provided with the distribution.
1418128546Sderaadt  *
1518128546Sderaadt  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
1618128546Sderaadt  * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
1718128546Sderaadt  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1818128546Sderaadt  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
1918128546Sderaadt  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2018128546Sderaadt  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2118128546Sderaadt  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2218128546Sderaadt  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2318128546Sderaadt  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2418128546Sderaadt  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2518128546Sderaadt  * SUCH DAMAGE.
2618128546Sderaadt  */
2718128546Sderaadt 
2818128546Sderaadt #include <sys/types.h>
2918128546Sderaadt #include <stdlib.h>
3018128546Sderaadt #include <string.h>
31aea60beeSderaadt #include <limits.h>
3218128546Sderaadt #include <rpc/rpc.h>
3318128546Sderaadt #include <rpc/xdr.h>
3418128546Sderaadt #include <rpcsvc/yp.h>
3518128546Sderaadt #include <rpcsvc/ypclnt.h>
3618128546Sderaadt #include "ypinternal.h"
3718128546Sderaadt 
38951548d6Sespie #ifdef YPMATCHCACHE
39372ff852Smarc static bool_t ypmatch_add(const char *, const char *, u_int, char *, u_int);
40372ff852Smarc static bool_t ypmatch_find(const char *, const char *, u_int, char **, u_int *);
41372ff852Smarc 
42372ff852Smarc static struct ypmatch_ent {
43372ff852Smarc 	struct ypmatch_ent	*next;
44372ff852Smarc 	char			*map, *key;
45372ff852Smarc 	char			*val;
46372ff852Smarc 	int			 keylen, vallen;
47372ff852Smarc 	time_t			 expire_t;
48372ff852Smarc } *ypmc;
49372ff852Smarc 
5018128546Sderaadt int _yplib_cache = 5;
5118128546Sderaadt 
5218128546Sderaadt static bool_t
ypmatch_add(const char * map,const char * key,u_int keylen,char * val,u_int vallen)5311e5d692Sderaadt ypmatch_add(const char *map, const char *key, u_int keylen, char *val,
5411e5d692Sderaadt     u_int vallen)
5518128546Sderaadt {
5618128546Sderaadt 	struct ypmatch_ent *ep;
57740a91e6Sschwarze 	char *newmap = NULL, *newkey = NULL, *newval = NULL;
5818128546Sderaadt 	time_t t;
5918128546Sderaadt 
600ffb34fdSderaadt 	if (keylen == 0 || vallen == 0)
610ffb34fdSderaadt 		return (0);
620ffb34fdSderaadt 
6318128546Sderaadt 	(void)time(&t);
6418128546Sderaadt 
65740a91e6Sschwarze 	/* Allocate all required memory first. */
66740a91e6Sschwarze 	if ((newmap = strdup(map)) == NULL ||
67740a91e6Sschwarze 	    (newkey = malloc(keylen)) == NULL ||
68740a91e6Sschwarze 	    (newval = malloc(vallen)) == NULL) {
69740a91e6Sschwarze 		free(newkey);
70740a91e6Sschwarze 		free(newmap);
71740a91e6Sschwarze 		return 0;
72740a91e6Sschwarze 	}
73740a91e6Sschwarze 
7418128546Sderaadt 	for (ep = ypmc; ep; ep = ep->next)
7518128546Sderaadt 		if (ep->expire_t < t)
7618128546Sderaadt 			break;
77740a91e6Sschwarze 
7818128546Sderaadt 	if (ep == NULL) {
79740a91e6Sschwarze 		/* No expired node, create a new one. */
80740a91e6Sschwarze 		if ((ep = malloc(sizeof *ep)) == NULL) {
81740a91e6Sschwarze 			free(newval);
82740a91e6Sschwarze 			free(newkey);
83740a91e6Sschwarze 			free(newmap);
8418128546Sderaadt 			return 0;
85740a91e6Sschwarze 		}
8618128546Sderaadt 		ep->next = ypmc;
8718128546Sderaadt 		ypmc = ep;
88740a91e6Sschwarze 	} else {
89740a91e6Sschwarze 		/* Reuse the first expired node from the list. */
9018128546Sderaadt 		free(ep->val);
9118128546Sderaadt 		free(ep->key);
92740a91e6Sschwarze 		free(ep->map);
9318128546Sderaadt 	}
9418128546Sderaadt 
95740a91e6Sschwarze 	/* Now we have all the memory we need, copy the data in. */
96740a91e6Sschwarze 	(void)memcpy(newkey, key, keylen);
97740a91e6Sschwarze 	(void)memcpy(newval, val, vallen);
98740a91e6Sschwarze 	ep->map = newmap;
99740a91e6Sschwarze 	ep->key = newkey;
100740a91e6Sschwarze 	ep->val = newval;
10118128546Sderaadt 	ep->keylen = keylen;
10218128546Sderaadt 	ep->vallen = vallen;
10318128546Sderaadt 	ep->expire_t = t + _yplib_cache;
10418128546Sderaadt 	return 1;
10518128546Sderaadt }
10618128546Sderaadt 
10718128546Sderaadt static bool_t
ypmatch_find(const char * map,const char * key,u_int keylen,char ** val,u_int * vallen)10811e5d692Sderaadt ypmatch_find(const char *map, const char *key, u_int keylen, char **val,
10911e5d692Sderaadt     u_int *vallen)
11018128546Sderaadt {
11118128546Sderaadt 	struct ypmatch_ent *ep;
11218128546Sderaadt 	time_t          t;
11318128546Sderaadt 
11418128546Sderaadt 	if (ypmc == NULL)
11518128546Sderaadt 		return 0;
11618128546Sderaadt 
11718128546Sderaadt 	(void) time(&t);
11818128546Sderaadt 
11918128546Sderaadt 	for (ep = ypmc; ep; ep = ep->next) {
12018128546Sderaadt 		if (ep->keylen != keylen)
12118128546Sderaadt 			continue;
12218128546Sderaadt 		if (strcmp(ep->map, map))
12318128546Sderaadt 			continue;
12418128546Sderaadt 		if (memcmp(ep->key, key, keylen))
12518128546Sderaadt 			continue;
12618128546Sderaadt 		if (t > ep->expire_t)
12718128546Sderaadt 			continue;
12818128546Sderaadt 
12918128546Sderaadt 		*val = ep->val;
13018128546Sderaadt 		*vallen = ep->vallen;
13118128546Sderaadt 		return 1;
13218128546Sderaadt 	}
13318128546Sderaadt 	return 0;
13418128546Sderaadt }
135951548d6Sespie #endif
13618128546Sderaadt 
13718128546Sderaadt int
yp_match(const char * indomain,const char * inmap,const char * inkey,int inkeylen,char ** outval,int * outvallen)13811e5d692Sderaadt yp_match(const char *indomain, const char *inmap, const char *inkey,
13911e5d692Sderaadt     int inkeylen, char **outval, int *outvallen)
14018128546Sderaadt {
14118128546Sderaadt 	struct dom_binding *ysd;
14218128546Sderaadt 	struct ypresp_val yprv;
14318128546Sderaadt 	struct timeval  tv;
14418128546Sderaadt 	struct ypreq_key yprk;
14577124a50Sderaadt 	int tries = 0, r;
14618128546Sderaadt 
147a69cfd83Sderaadt 	if (indomain == NULL || *indomain == '\0' ||
148a69cfd83Sderaadt 	    strlen(indomain) > YPMAXDOMAIN || inmap == NULL ||
149a69cfd83Sderaadt 	    *inmap == '\0' || strlen(inmap) > YPMAXMAP ||
15084c5bc33Sderaadt 	    inkey == NULL || inkeylen == 0 || inkeylen >= YPMAXRECORD)
151a69cfd83Sderaadt 		return YPERR_BADARGS;
152a69cfd83Sderaadt 
15318128546Sderaadt 	*outval = NULL;
15418128546Sderaadt 	*outvallen = 0;
15518128546Sderaadt 
15618128546Sderaadt again:
15718128546Sderaadt 	if (_yp_dobind(indomain, &ysd) != 0)
15818128546Sderaadt 		return YPERR_DOMAIN;
15918128546Sderaadt 
16018128546Sderaadt #ifdef YPMATCHCACHE
16118128546Sderaadt 	if (!strcmp(_yp_domain, indomain) && ypmatch_find(inmap, inkey,
16218128546Sderaadt 	    inkeylen, &yprv.val.valdat_val, &yprv.val.valdat_len)) {
16318128546Sderaadt 		*outvallen = yprv.val.valdat_len;
16418128546Sderaadt 		if ((*outval = malloc(*outvallen + 1)) == NULL) {
16518128546Sderaadt 			_yp_unbind(ysd);
1664e2fcec2Sschwarze 			return YPERR_RESRC;
16718128546Sderaadt 		}
16818128546Sderaadt 		(void)memcpy(*outval, yprv.val.valdat_val, *outvallen);
16918128546Sderaadt 		(*outval)[*outvallen] = '\0';
17018128546Sderaadt 		_yp_unbind(ysd);
17118128546Sderaadt 		return 0;
17218128546Sderaadt 	}
17318128546Sderaadt #endif
17418128546Sderaadt 
17518128546Sderaadt 	tv.tv_sec = _yplib_timeout;
17618128546Sderaadt 	tv.tv_usec = 0;
17718128546Sderaadt 
17818128546Sderaadt 	yprk.domain = (char *)indomain;
17918128546Sderaadt 	yprk.map = (char *)inmap;
18018128546Sderaadt 	yprk.key.keydat_val = (char *) inkey;
18118128546Sderaadt 	yprk.key.keydat_len = inkeylen;
18218128546Sderaadt 
18318128546Sderaadt 	memset(&yprv, 0, sizeof yprv);
18418128546Sderaadt 
18518128546Sderaadt 	r = clnt_call(ysd->dom_client, YPPROC_MATCH,
18618128546Sderaadt 	    xdr_ypreq_key, &yprk, xdr_ypresp_val, &yprv, tv);
18718128546Sderaadt 	if (r != RPC_SUCCESS) {
18877124a50Sderaadt 		if (tries++)
18918128546Sderaadt 			clnt_perror(ysd->dom_client, "yp_match: clnt_call");
190*2d62dde8Sderaadt 		_yp_unbind(ysd);
19118128546Sderaadt 		goto again;
19218128546Sderaadt 	}
19318128546Sderaadt 	if (!(r = ypprot_err(yprv.stat))) {
19418128546Sderaadt 		*outvallen = yprv.val.valdat_len;
19518128546Sderaadt 		if ((*outval = malloc(*outvallen + 1)) == NULL) {
1964e2fcec2Sschwarze 			r = YPERR_RESRC;
19718128546Sderaadt 			goto out;
19818128546Sderaadt 		}
19918128546Sderaadt 		(void)memcpy(*outval, yprv.val.valdat_val, *outvallen);
20018128546Sderaadt 		(*outval)[*outvallen] = '\0';
20118128546Sderaadt #ifdef YPMATCHCACHE
20218128546Sderaadt 		if (strcmp(_yp_domain, indomain) == 0)
2034e0c9721Sschwarze 			(void)ypmatch_add(inmap, inkey, inkeylen,
2044e0c9721Sschwarze 			    *outval, *outvallen);
20518128546Sderaadt #endif
20618128546Sderaadt 	}
20718128546Sderaadt out:
20818128546Sderaadt 	xdr_free(xdr_ypresp_val, (char *) &yprv);
20918128546Sderaadt 	_yp_unbind(ysd);
21018128546Sderaadt 	return r;
21118128546Sderaadt }
212ba364befSguenther DEF_WEAK(yp_match);
21318128546Sderaadt 
21418128546Sderaadt int
yp_next(const char * indomain,const char * inmap,const char * inkey,int inkeylen,char ** outkey,int * outkeylen,char ** outval,int * outvallen)21511e5d692Sderaadt yp_next(const char *indomain, const char *inmap, const char *inkey,
21611e5d692Sderaadt     int inkeylen, char **outkey, int *outkeylen, char **outval, int *outvallen)
21718128546Sderaadt {
21818128546Sderaadt 	struct ypresp_key_val yprkv;
21918128546Sderaadt 	struct ypreq_key yprk;
22018128546Sderaadt 	struct dom_binding *ysd;
22118128546Sderaadt 	struct timeval  tv;
22277124a50Sderaadt 	int tries = 0, r;
22318128546Sderaadt 
224a69cfd83Sderaadt 	if (indomain == NULL || *indomain == '\0' ||
225a69cfd83Sderaadt 	    strlen(indomain) > YPMAXDOMAIN || inmap == NULL ||
22684c5bc33Sderaadt 	    *inmap == '\0' || strlen(inmap) > YPMAXMAP ||
22784c5bc33Sderaadt 	    inkeylen == 0 || inkeylen >= YPMAXRECORD)
228a69cfd83Sderaadt 		return YPERR_BADARGS;
229a69cfd83Sderaadt 
23018128546Sderaadt 	*outkey = *outval = NULL;
23118128546Sderaadt 	*outkeylen = *outvallen = 0;
23218128546Sderaadt 
23318128546Sderaadt again:
23418128546Sderaadt 	if (_yp_dobind(indomain, &ysd) != 0)
23518128546Sderaadt 		return YPERR_DOMAIN;
23618128546Sderaadt 
23718128546Sderaadt 	tv.tv_sec = _yplib_timeout;
23818128546Sderaadt 	tv.tv_usec = 0;
23918128546Sderaadt 
24018128546Sderaadt 	yprk.domain = (char *)indomain;
24118128546Sderaadt 	yprk.map = (char *)inmap;
24218128546Sderaadt 	yprk.key.keydat_val = (char *)inkey;
24318128546Sderaadt 	yprk.key.keydat_len = inkeylen;
24418128546Sderaadt 	(void)memset(&yprkv, 0, sizeof yprkv);
24518128546Sderaadt 
24618128546Sderaadt 	r = clnt_call(ysd->dom_client, YPPROC_NEXT,
24718128546Sderaadt 	    xdr_ypreq_key, &yprk, xdr_ypresp_key_val, &yprkv, tv);
24818128546Sderaadt 	if (r != RPC_SUCCESS) {
24977124a50Sderaadt 		if (tries++)
25018128546Sderaadt 			clnt_perror(ysd->dom_client, "yp_next: clnt_call");
251*2d62dde8Sderaadt 		_yp_unbind(ysd);
25218128546Sderaadt 		goto again;
25318128546Sderaadt 	}
25418128546Sderaadt 	if (!(r = ypprot_err(yprkv.stat))) {
25518128546Sderaadt 		*outkeylen = yprkv.key.keydat_len;
2564e0c9721Sschwarze 		*outvallen = yprkv.val.valdat_len;
2574e0c9721Sschwarze 		if ((*outkey = malloc(*outkeylen + 1)) == NULL ||
2584e0c9721Sschwarze 		    (*outval = malloc(*outvallen + 1)) == NULL) {
2594e0c9721Sschwarze 			free(*outkey);
260a69cfd83Sderaadt 			r = YPERR_RESRC;
2614e0c9721Sschwarze 		} else {
26218128546Sderaadt 			(void)memcpy(*outkey, yprkv.key.keydat_val, *outkeylen);
26318128546Sderaadt 			(*outkey)[*outkeylen] = '\0';
26418128546Sderaadt 			(void)memcpy(*outval, yprkv.val.valdat_val, *outvallen);
26518128546Sderaadt 			(*outval)[*outvallen] = '\0';
26618128546Sderaadt 		}
26718128546Sderaadt 	}
26818128546Sderaadt 	xdr_free(xdr_ypresp_key_val, (char *) &yprkv);
26918128546Sderaadt 	_yp_unbind(ysd);
27018128546Sderaadt 	return r;
27118128546Sderaadt }
272ba364befSguenther DEF_WEAK(yp_next);
273