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