xref: /freebsd/lib/libc/gen/readdir-compat11.c (revision dc36d6f9bb1753f3808552f3afd30eda9a7b206a)
169921123SKonstantin Belousov /*
269921123SKonstantin Belousov  * Copyright (c) 1983, 1993
369921123SKonstantin Belousov  *	The Regents of the University of California.  All rights reserved.
469921123SKonstantin Belousov  *
569921123SKonstantin Belousov  * Redistribution and use in source and binary forms, with or without
669921123SKonstantin Belousov  * modification, are permitted provided that the following conditions
769921123SKonstantin Belousov  * are met:
869921123SKonstantin Belousov  * 1. Redistributions of source code must retain the above copyright
969921123SKonstantin Belousov  *    notice, this list of conditions and the following disclaimer.
1069921123SKonstantin Belousov  * 2. Redistributions in binary form must reproduce the above copyright
1169921123SKonstantin Belousov  *    notice, this list of conditions and the following disclaimer in the
1269921123SKonstantin Belousov  *    documentation and/or other materials provided with the distribution.
1369921123SKonstantin Belousov  * 3. Neither the name of the University nor the names of its contributors
1469921123SKonstantin Belousov  *    may be used to endorse or promote products derived from this software
1569921123SKonstantin Belousov  *    without specific prior written permission.
1669921123SKonstantin Belousov  *
1769921123SKonstantin Belousov  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
1869921123SKonstantin Belousov  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1969921123SKonstantin Belousov  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2069921123SKonstantin Belousov  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2169921123SKonstantin Belousov  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2269921123SKonstantin Belousov  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2369921123SKonstantin Belousov  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2469921123SKonstantin Belousov  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2569921123SKonstantin Belousov  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2669921123SKonstantin Belousov  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2769921123SKonstantin Belousov  * SUCH DAMAGE.
28*c1920558SJohn Baldwin  * From: FreeBSD: head/lib/libc/gen/readdir.c 314436 2017-02-28 23:42:47Z imp
2969921123SKonstantin Belousov  */
3069921123SKonstantin Belousov 
3169921123SKonstantin Belousov #include "namespace.h"
3269921123SKonstantin Belousov #include <sys/param.h>
3369921123SKonstantin Belousov #define	_WANT_FREEBSD11_DIRENT
3469921123SKonstantin Belousov #include <dirent.h>
3569921123SKonstantin Belousov #include <errno.h>
3669921123SKonstantin Belousov #include <stdbool.h>
3769921123SKonstantin Belousov #include <stddef.h>
3869921123SKonstantin Belousov #include <stdlib.h>
3969921123SKonstantin Belousov #include <string.h>
4069921123SKonstantin Belousov #include <pthread.h>
4169921123SKonstantin Belousov #include "un-namespace.h"
4269921123SKonstantin Belousov 
4369921123SKonstantin Belousov #include "libc_private.h"
4469921123SKonstantin Belousov #include "gen-private.h"
4569921123SKonstantin Belousov #include "telldir.h"
4669921123SKonstantin Belousov 
4769921123SKonstantin Belousov #include "gen-compat.h"
4869921123SKonstantin Belousov 
4969921123SKonstantin Belousov static bool
freebsd11_cvtdirent(struct freebsd11_dirent * dstdp,struct dirent * srcdp)5069921123SKonstantin Belousov freebsd11_cvtdirent(struct freebsd11_dirent *dstdp, struct dirent *srcdp)
5169921123SKonstantin Belousov {
5269921123SKonstantin Belousov 
5369921123SKonstantin Belousov 	if (srcdp->d_namlen >= sizeof(dstdp->d_name))
5469921123SKonstantin Belousov 		return (false);
5569921123SKonstantin Belousov 	dstdp->d_type = srcdp->d_type;
5669921123SKonstantin Belousov 	dstdp->d_namlen = srcdp->d_namlen;
5769921123SKonstantin Belousov 	dstdp->d_fileno = srcdp->d_fileno;		/* truncate */
5869921123SKonstantin Belousov 	dstdp->d_reclen = FREEBSD11_DIRSIZ(dstdp);
5969921123SKonstantin Belousov 	bcopy(srcdp->d_name, dstdp->d_name, dstdp->d_namlen);
6069921123SKonstantin Belousov 	bzero(dstdp->d_name + dstdp->d_namlen,
6169921123SKonstantin Belousov 	    dstdp->d_reclen - offsetof(struct freebsd11_dirent, d_name) -
6269921123SKonstantin Belousov 	    dstdp->d_namlen);
6369921123SKonstantin Belousov 	return (true);
6469921123SKonstantin Belousov }
6569921123SKonstantin Belousov 
6669921123SKonstantin Belousov struct freebsd11_dirent *
freebsd11_readdir(DIR * dirp)6769921123SKonstantin Belousov freebsd11_readdir(DIR *dirp)
6869921123SKonstantin Belousov {
6969921123SKonstantin Belousov 	struct freebsd11_dirent *dstdp;
7069921123SKonstantin Belousov 	struct dirent *dp;
7169921123SKonstantin Belousov 
7269921123SKonstantin Belousov 	if (__isthreaded)
7369921123SKonstantin Belousov 		_pthread_mutex_lock(&dirp->dd_lock);
7469921123SKonstantin Belousov 	dp = _readdir_unlocked(dirp, RDU_SKIP);
7569921123SKonstantin Belousov 	if (dp != NULL) {
7669921123SKonstantin Belousov 		if (dirp->dd_compat_de == NULL)
7769921123SKonstantin Belousov 			dirp->dd_compat_de = malloc(sizeof(struct
7869921123SKonstantin Belousov 			    freebsd11_dirent));
7969921123SKonstantin Belousov 		if (freebsd11_cvtdirent(dirp->dd_compat_de, dp))
8069921123SKonstantin Belousov 			dstdp = dirp->dd_compat_de;
8169921123SKonstantin Belousov 		else
8269921123SKonstantin Belousov 			dstdp = NULL;
8369921123SKonstantin Belousov 	} else
8469921123SKonstantin Belousov 		dstdp = NULL;
8569921123SKonstantin Belousov 	if (__isthreaded)
8669921123SKonstantin Belousov 		_pthread_mutex_unlock(&dirp->dd_lock);
8769921123SKonstantin Belousov 
8869921123SKonstantin Belousov 	return (dstdp);
8969921123SKonstantin Belousov }
9069921123SKonstantin Belousov 
9169921123SKonstantin Belousov int
freebsd11_readdir_r(DIR * dirp,struct freebsd11_dirent * entry,struct freebsd11_dirent ** result)9269921123SKonstantin Belousov freebsd11_readdir_r(DIR *dirp, struct freebsd11_dirent *entry,
9369921123SKonstantin Belousov     struct freebsd11_dirent **result)
9469921123SKonstantin Belousov {
9569921123SKonstantin Belousov 	struct dirent xentry, *xresult;
9669921123SKonstantin Belousov 	int error;
9769921123SKonstantin Belousov 
9869921123SKonstantin Belousov 	error = __readdir_r(dirp, &xentry, &xresult);
9969921123SKonstantin Belousov 	if (error != 0)
10069921123SKonstantin Belousov 		return (error);
10169921123SKonstantin Belousov 	if (xresult != NULL) {
10269921123SKonstantin Belousov 		if (freebsd11_cvtdirent(entry, &xentry))
10369921123SKonstantin Belousov 			*result = entry;
10469921123SKonstantin Belousov 		else /* should not happen due to RDU_SHORT */
10569921123SKonstantin Belousov 			*result = NULL;
10669921123SKonstantin Belousov 	} else
10769921123SKonstantin Belousov 		*result = NULL;
10869921123SKonstantin Belousov 	return (0);
10969921123SKonstantin Belousov }
11069921123SKonstantin Belousov 
11169921123SKonstantin Belousov __sym_compat(readdir, freebsd11_readdir, FBSD_1.0);
11269921123SKonstantin Belousov __sym_compat(readdir_r, freebsd11_readdir_r, FBSD_1.0);
113