xref: /freebsd/lib/libc/net/nss_compat.c (revision 559a218c9b257775fb249b67945fe4a05b7a6b9f)
146d93063SJacques Vidrine /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3d915a14eSPedro F. Giffuni  *
446d93063SJacques Vidrine  * Copyright (c) 2003 Networks Associates Technology, Inc.
546d93063SJacques Vidrine  * All rights reserved.
646d93063SJacques Vidrine  *
746d93063SJacques Vidrine  * This software was developed for the FreeBSD Project by
846d93063SJacques Vidrine  * Jacques A. Vidrine, Safeport Network Services, and Network
946d93063SJacques Vidrine  * Associates Laboratories, the Security Research Division of Network
1046d93063SJacques Vidrine  * Associates, Inc. under DARPA/SPAWAR contract N66001-01-C-8035
1146d93063SJacques Vidrine  * ("CBOSS"), as part of the DARPA CHATS research program.
1246d93063SJacques Vidrine  *
1346d93063SJacques Vidrine  * Redistribution and use in source and binary forms, with or without
1446d93063SJacques Vidrine  * modification, are permitted provided that the following conditions
1546d93063SJacques Vidrine  * are met:
1646d93063SJacques Vidrine  * 1. Redistributions of source code must retain the above copyright
1746d93063SJacques Vidrine  *    notice, this list of conditions and the following disclaimer.
1846d93063SJacques Vidrine  * 2. Redistributions in binary form must reproduce the above copyright
1946d93063SJacques Vidrine  *    notice, this list of conditions and the following disclaimer in the
2046d93063SJacques Vidrine  *    documentation and/or other materials provided with the distribution.
2146d93063SJacques Vidrine  *
2246d93063SJacques Vidrine  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
2346d93063SJacques Vidrine  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2446d93063SJacques Vidrine  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2546d93063SJacques Vidrine  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2646d93063SJacques Vidrine  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2746d93063SJacques Vidrine  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2846d93063SJacques Vidrine  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2946d93063SJacques Vidrine  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3046d93063SJacques Vidrine  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3146d93063SJacques Vidrine  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3246d93063SJacques Vidrine  * SUCH DAMAGE.
3346d93063SJacques Vidrine  *
3446d93063SJacques Vidrine  * Compatibility shims for the GNU C Library-style nsswitch interface.
3546d93063SJacques Vidrine  */
3646d93063SJacques Vidrine #include "namespace.h"
3746d93063SJacques Vidrine #include <sys/param.h>
38c849849dSJacques Vidrine #include <errno.h>
3946d93063SJacques Vidrine #include <nss.h>
4046d93063SJacques Vidrine #include <pthread.h>
4146d93063SJacques Vidrine #include <pthread_np.h>
4246d93063SJacques Vidrine #include "un-namespace.h"
43a03fd3b6SJacques Vidrine #include "libc_private.h"
4446d93063SJacques Vidrine 
4546d93063SJacques Vidrine 
4646d93063SJacques Vidrine struct group;
4746d93063SJacques Vidrine struct passwd;
4846d93063SJacques Vidrine 
4946d93063SJacques Vidrine static int	terminator;
5046d93063SJacques Vidrine 
5146d93063SJacques Vidrine #define DECLARE_TERMINATOR(x)					\
5246d93063SJacques Vidrine static pthread_key_t	 _term_key_##x;				\
5346d93063SJacques Vidrine static void							\
5446d93063SJacques Vidrine _term_create_##x(void)						\
5546d93063SJacques Vidrine {								\
5646d93063SJacques Vidrine 	(void)_pthread_key_create(&_term_key_##x, NULL);	\
5746d93063SJacques Vidrine }								\
5846d93063SJacques Vidrine static void		*_term_main_##x;			\
5946d93063SJacques Vidrine static pthread_once_t	 _term_once_##x = PTHREAD_ONCE_INIT
6046d93063SJacques Vidrine 
6146d93063SJacques Vidrine #define SET_TERMINATOR(x, y)						\
6246d93063SJacques Vidrine do {									\
63a03fd3b6SJacques Vidrine 	if (!__isthreaded || _pthread_main_np())			\
6446d93063SJacques Vidrine 		_term_main_##x = (y);					\
6546d93063SJacques Vidrine 	else {								\
6646d93063SJacques Vidrine 		(void)_pthread_once(&_term_once_##x, _term_create_##x);	\
6746d93063SJacques Vidrine 		(void)_pthread_setspecific(_term_key_##x, y);		\
6846d93063SJacques Vidrine 	}								\
6946d93063SJacques Vidrine } while (0)
7046d93063SJacques Vidrine 
7146d93063SJacques Vidrine #define CHECK_TERMINATOR(x)					\
72a03fd3b6SJacques Vidrine (!__isthreaded || _pthread_main_np() ?				\
7346d93063SJacques Vidrine     (_term_main_##x) :						\
7446d93063SJacques Vidrine     ((void)_pthread_once(&_term_once_##x, _term_create_##x),	\
7546d93063SJacques Vidrine     _pthread_getspecific(_term_key_##x)))
7646d93063SJacques Vidrine 
7746d93063SJacques Vidrine 
7846d93063SJacques Vidrine 
7946d93063SJacques Vidrine DECLARE_TERMINATOR(group);
8046d93063SJacques Vidrine 
81bf1aa507SCraig Rodrigues int __nss_compat_getgrnam_r(void *retval, void *mdata, va_list ap);
82bf1aa507SCraig Rodrigues int __nss_compat_getgrgid_r(void *retval, void *mdata, va_list ap);
83bf1aa507SCraig Rodrigues int __nss_compat_getgrent_r(void *retval, void *mdata, va_list ap);
84bf1aa507SCraig Rodrigues int __nss_compat_setgrent(void *retval, void *mdata, va_list ap);
85bf1aa507SCraig Rodrigues int __nss_compat_endgrent(void *retval, void *mdata, va_list ap);
86bf1aa507SCraig Rodrigues int __nss_compat_getpwnam_r(void *retval, void *mdata, va_list ap);
87bf1aa507SCraig Rodrigues int __nss_compat_getpwuid_r(void *retval, void *mdata, va_list ap);
88bf1aa507SCraig Rodrigues int __nss_compat_getpwent_r(void *retval, void *mdata, va_list ap);
89bf1aa507SCraig Rodrigues int __nss_compat_setpwent(void *retval, void *mdata, va_list ap);
90bf1aa507SCraig Rodrigues int __nss_compat_endpwent(void *retval, void *mdata, va_list ap);
9146d93063SJacques Vidrine 
9246d93063SJacques Vidrine int
__nss_compat_getgrnam_r(void * retval,void * mdata,va_list ap)9346d93063SJacques Vidrine __nss_compat_getgrnam_r(void *retval, void *mdata, va_list ap)
9446d93063SJacques Vidrine {
9546d93063SJacques Vidrine 	int (*fn)(const char *, struct group *, char *, size_t, int *);
9646d93063SJacques Vidrine 	const char	*name;
9746d93063SJacques Vidrine 	struct group	*grp;
9846d93063SJacques Vidrine 	char		*buffer;
993bb0c17dSBrooks Davis 	int		*errnop, ns_status;
10046d93063SJacques Vidrine 	size_t		 bufsize;
1013bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
10246d93063SJacques Vidrine 
10346d93063SJacques Vidrine 	fn = mdata;
10446d93063SJacques Vidrine 	name = va_arg(ap, const char *);
10546d93063SJacques Vidrine 	grp = va_arg(ap, struct group *);
10646d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
10746d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
10846d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
1093bb0c17dSBrooks Davis 	nss_status = fn(name, grp, buffer, bufsize, errnop);
1103bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
1113bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
11246d93063SJacques Vidrine 		*(struct group **)retval = grp;
1133bb0c17dSBrooks Davis 	return (ns_status);
11446d93063SJacques Vidrine }
11546d93063SJacques Vidrine 
11646d93063SJacques Vidrine 
11746d93063SJacques Vidrine int
__nss_compat_getgrgid_r(void * retval,void * mdata,va_list ap)11846d93063SJacques Vidrine __nss_compat_getgrgid_r(void *retval, void *mdata, va_list ap)
11946d93063SJacques Vidrine {
12046d93063SJacques Vidrine 	int (*fn)(gid_t, struct group *, char *, size_t, int *);
12146d93063SJacques Vidrine 	gid_t		 gid;
12246d93063SJacques Vidrine 	struct group	*grp;
12346d93063SJacques Vidrine 	char		*buffer;
1243bb0c17dSBrooks Davis 	int		*errnop, ns_status;
12546d93063SJacques Vidrine 	size_t		 bufsize;
1263bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
12746d93063SJacques Vidrine 
12846d93063SJacques Vidrine 	fn = mdata;
12946d93063SJacques Vidrine 	gid = va_arg(ap, gid_t);
13046d93063SJacques Vidrine 	grp = va_arg(ap, struct group *);
13146d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
13246d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
13346d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
1343bb0c17dSBrooks Davis 	nss_status = fn(gid, grp, buffer, bufsize, errnop);
1353bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
1363bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
13746d93063SJacques Vidrine 		*(struct group **)retval = grp;
1383bb0c17dSBrooks Davis 	return (ns_status);
13946d93063SJacques Vidrine }
14046d93063SJacques Vidrine 
14146d93063SJacques Vidrine 
14246d93063SJacques Vidrine int
__nss_compat_getgrent_r(void * retval,void * mdata,va_list ap)14346d93063SJacques Vidrine __nss_compat_getgrent_r(void *retval, void *mdata, va_list ap)
14446d93063SJacques Vidrine {
14546d93063SJacques Vidrine 	int (*fn)(struct group *, char *, size_t, int *);
14646d93063SJacques Vidrine 	struct group	*grp;
14746d93063SJacques Vidrine 	char		*buffer;
1483bb0c17dSBrooks Davis 	int		*errnop, ns_status;
14946d93063SJacques Vidrine 	size_t		 bufsize;
1503bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
15146d93063SJacques Vidrine 
15246d93063SJacques Vidrine 	if (CHECK_TERMINATOR(group))
15346d93063SJacques Vidrine 		return (NS_NOTFOUND);
15446d93063SJacques Vidrine 	fn = mdata;
15546d93063SJacques Vidrine 	grp = va_arg(ap, struct group *);
15646d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
15746d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
15846d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
1593bb0c17dSBrooks Davis 	nss_status = fn(grp, buffer, bufsize, errnop);
1603bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
1613bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
16246d93063SJacques Vidrine 		*(struct group **)retval = grp;
1633bb0c17dSBrooks Davis 	else if (ns_status != NS_RETURN)
16446d93063SJacques Vidrine 		SET_TERMINATOR(group, &terminator);
1653bb0c17dSBrooks Davis 	return (ns_status);
16646d93063SJacques Vidrine }
16746d93063SJacques Vidrine 
16846d93063SJacques Vidrine 
16946d93063SJacques Vidrine int
__nss_compat_setgrent(void * retval,void * mdata,va_list ap)17046d93063SJacques Vidrine __nss_compat_setgrent(void *retval, void *mdata, va_list ap)
17146d93063SJacques Vidrine {
17246d93063SJacques Vidrine 
17346d93063SJacques Vidrine 	SET_TERMINATOR(group, NULL);
17446d93063SJacques Vidrine 	((int (*)(void))mdata)();
17546d93063SJacques Vidrine 	return (NS_UNAVAIL);
17646d93063SJacques Vidrine }
17746d93063SJacques Vidrine 
17846d93063SJacques Vidrine 
17946d93063SJacques Vidrine int
__nss_compat_endgrent(void * retval,void * mdata,va_list ap)18046d93063SJacques Vidrine __nss_compat_endgrent(void *retval, void *mdata, va_list ap)
18146d93063SJacques Vidrine {
18246d93063SJacques Vidrine 
18346d93063SJacques Vidrine 	SET_TERMINATOR(group, NULL);
18446d93063SJacques Vidrine 	((int (*)(void))mdata)();
18546d93063SJacques Vidrine 	return (NS_UNAVAIL);
18646d93063SJacques Vidrine }
18746d93063SJacques Vidrine 
18846d93063SJacques Vidrine 
18946d93063SJacques Vidrine 
19046d93063SJacques Vidrine DECLARE_TERMINATOR(passwd);
19146d93063SJacques Vidrine 
19246d93063SJacques Vidrine 
19346d93063SJacques Vidrine int
__nss_compat_getpwnam_r(void * retval,void * mdata,va_list ap)19446d93063SJacques Vidrine __nss_compat_getpwnam_r(void *retval, void *mdata, va_list ap)
19546d93063SJacques Vidrine {
19646d93063SJacques Vidrine 	int (*fn)(const char *, struct passwd *, char *, size_t, int *);
19746d93063SJacques Vidrine 	const char	*name;
19846d93063SJacques Vidrine 	struct passwd	*pwd;
19946d93063SJacques Vidrine 	char		*buffer;
2003bb0c17dSBrooks Davis 	int		*errnop, ns_status;
20146d93063SJacques Vidrine 	size_t		 bufsize;
2023bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
20346d93063SJacques Vidrine 
20446d93063SJacques Vidrine 	fn = mdata;
20546d93063SJacques Vidrine 	name = va_arg(ap, const char *);
20646d93063SJacques Vidrine 	pwd = va_arg(ap, struct passwd *);
20746d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
20846d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
20946d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
2103bb0c17dSBrooks Davis 	nss_status = fn(name, pwd, buffer, bufsize, errnop);
2113bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
2123bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
21346d93063SJacques Vidrine 		*(struct passwd **)retval = pwd;
2143bb0c17dSBrooks Davis 	return (ns_status);
21546d93063SJacques Vidrine }
21646d93063SJacques Vidrine 
21746d93063SJacques Vidrine 
21846d93063SJacques Vidrine int
__nss_compat_getpwuid_r(void * retval,void * mdata,va_list ap)21946d93063SJacques Vidrine __nss_compat_getpwuid_r(void *retval, void *mdata, va_list ap)
22046d93063SJacques Vidrine {
22146d93063SJacques Vidrine 	int (*fn)(uid_t, struct passwd *, char *, size_t, int *);
22246d93063SJacques Vidrine 	uid_t		 uid;
22346d93063SJacques Vidrine 	struct passwd	*pwd;
22446d93063SJacques Vidrine 	char		*buffer;
2253bb0c17dSBrooks Davis 	int		*errnop, ns_status;
22646d93063SJacques Vidrine 	size_t		 bufsize;
2273bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
22846d93063SJacques Vidrine 
22946d93063SJacques Vidrine 	fn = mdata;
23046d93063SJacques Vidrine 	uid = va_arg(ap, uid_t);
23146d93063SJacques Vidrine 	pwd = va_arg(ap, struct passwd *);
23246d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
23346d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
23446d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
2353bb0c17dSBrooks Davis 	nss_status = fn(uid, pwd, buffer, bufsize, errnop);
2363bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
2373bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
23846d93063SJacques Vidrine 		*(struct passwd **)retval = pwd;
2393bb0c17dSBrooks Davis 	return (ns_status);
24046d93063SJacques Vidrine }
24146d93063SJacques Vidrine 
24246d93063SJacques Vidrine 
24346d93063SJacques Vidrine int
__nss_compat_getpwent_r(void * retval,void * mdata,va_list ap)24446d93063SJacques Vidrine __nss_compat_getpwent_r(void *retval, void *mdata, va_list ap)
24546d93063SJacques Vidrine {
24646d93063SJacques Vidrine 	int (*fn)(struct passwd *, char *, size_t, int *);
24746d93063SJacques Vidrine 	struct passwd	*pwd;
24846d93063SJacques Vidrine 	char		*buffer;
2493bb0c17dSBrooks Davis 	int		*errnop, ns_status;
25046d93063SJacques Vidrine 	size_t		 bufsize;
2513bb0c17dSBrooks Davis 	enum nss_status	 nss_status;
25246d93063SJacques Vidrine 
25346d93063SJacques Vidrine 	if (CHECK_TERMINATOR(passwd))
25446d93063SJacques Vidrine 		return (NS_NOTFOUND);
25546d93063SJacques Vidrine 	fn = mdata;
25646d93063SJacques Vidrine 	pwd = va_arg(ap, struct passwd *);
25746d93063SJacques Vidrine 	buffer = va_arg(ap, char *);
25846d93063SJacques Vidrine 	bufsize = va_arg(ap, size_t);
25946d93063SJacques Vidrine 	errnop = va_arg(ap, int *);
2603bb0c17dSBrooks Davis 	nss_status = fn(pwd, buffer, bufsize, errnop);
2613bb0c17dSBrooks Davis 	ns_status = __nss_compat_result(nss_status, *errnop);
2623bb0c17dSBrooks Davis 	if (ns_status == NS_SUCCESS)
26346d93063SJacques Vidrine 		*(struct passwd **)retval = pwd;
2643bb0c17dSBrooks Davis 	else if (ns_status != NS_RETURN)
26546d93063SJacques Vidrine 		SET_TERMINATOR(passwd, &terminator);
2663bb0c17dSBrooks Davis 	return (ns_status);
26746d93063SJacques Vidrine }
26846d93063SJacques Vidrine 
26946d93063SJacques Vidrine 
27046d93063SJacques Vidrine int
__nss_compat_setpwent(void * retval,void * mdata,va_list ap)27146d93063SJacques Vidrine __nss_compat_setpwent(void *retval, void *mdata, va_list ap)
27246d93063SJacques Vidrine {
27346d93063SJacques Vidrine 
27446d93063SJacques Vidrine 	SET_TERMINATOR(passwd, NULL);
27546d93063SJacques Vidrine 	((int (*)(void))mdata)();
27646d93063SJacques Vidrine 	return (NS_UNAVAIL);
27746d93063SJacques Vidrine }
27846d93063SJacques Vidrine 
27946d93063SJacques Vidrine 
28046d93063SJacques Vidrine int
__nss_compat_endpwent(void * retval,void * mdata,va_list ap)28146d93063SJacques Vidrine __nss_compat_endpwent(void *retval, void *mdata, va_list ap)
28246d93063SJacques Vidrine {
28346d93063SJacques Vidrine 
28446d93063SJacques Vidrine 	SET_TERMINATOR(passwd, NULL);
28546d93063SJacques Vidrine 	((int (*)(void))mdata)();
28646d93063SJacques Vidrine 	return (NS_UNAVAIL);
28746d93063SJacques Vidrine }
288