1 1.6 joerg /* $NetBSD: compat_getdents.c,v 1.6 2012/03/13 22:24:48 joerg Exp $ */ 2 1.1 christos 3 1.1 christos /*- 4 1.1 christos * Copyright (c) 2005 The NetBSD Foundation, Inc. 5 1.1 christos * All rights reserved. 6 1.1 christos * 7 1.1 christos * This code is derived from software contributed to The NetBSD Foundation 8 1.1 christos * by Christos Zoulas. 9 1.1 christos * 10 1.1 christos * Redistribution and use in source and binary forms, with or without 11 1.1 christos * modification, are permitted provided that the following conditions 12 1.1 christos * are met: 13 1.1 christos * 1. Redistributions of source code must retain the above copyright 14 1.1 christos * notice, this list of conditions and the following disclaimer. 15 1.1 christos * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 christos * notice, this list of conditions and the following disclaimer in the 17 1.1 christos * documentation and/or other materials provided with the distribution. 18 1.1 christos * 19 1.1 christos * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 christos * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 christos * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 christos * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 christos * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 christos * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 christos * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 christos * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 christos * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 christos * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 christos * POSSIBILITY OF SUCH DAMAGE. 30 1.1 christos */ 31 1.1 christos 32 1.1 christos #include <sys/cdefs.h> 33 1.1 christos #if defined(LIBC_SCCS) && !defined(lint) 34 1.6 joerg __RCSID("$NetBSD: compat_getdents.c,v 1.6 2012/03/13 22:24:48 joerg Exp $"); 35 1.1 christos #endif /* LIBC_SCCS and not lint */ 36 1.1 christos 37 1.1 christos #define __LIBC12_SOURCE__ 38 1.1 christos 39 1.1 christos #include "namespace.h" 40 1.1 christos #include <sys/types.h> 41 1.6 joerg #include <assert.h> 42 1.1 christos #include <dirent.h> 43 1.6 joerg #include <stddef.h> 44 1.1 christos #include <compat/include/dirent.h> 45 1.1 christos #include <string.h> 46 1.1 christos 47 1.1 christos /* 48 1.1 christos * libc12 compatible getdents routine. 49 1.1 christos */ 50 1.1 christos int 51 1.1 christos getdents(int fd, char *buf, size_t nbytes) 52 1.1 christos { 53 1.1 christos struct dirent *ndp, *nndp, *endp; 54 1.1 christos struct dirent12 *odp; 55 1.4 mrg ino_t ino; 56 1.1 christos int rv; 57 1.5 christos size_t len; 58 1.1 christos 59 1.1 christos if ((rv = __getdents30(fd, buf, nbytes)) == -1) 60 1.1 christos return rv; 61 1.1 christos 62 1.1 christos odp = (struct dirent12 *)(void *)buf; 63 1.1 christos ndp = (struct dirent *)(void *)buf; 64 1.1 christos endp = (struct dirent *)(void *)&buf[rv]; 65 1.1 christos 66 1.1 christos /* 67 1.1 christos * In-place conversion. This works because odp 68 1.1 christos * is smaller than ndp, but it has to be done 69 1.1 christos * in the right sequence. 70 1.1 christos */ 71 1.1 christos for (; ndp < endp; ndp = nndp) { 72 1.1 christos nndp = _DIRENT_NEXT(ndp); 73 1.2 nakayama /* XXX: avoid unaligned 64-bit access on sparc64 */ 74 1.4 mrg /* XXX: does this work? */ 75 1.4 mrg memcpy(&ino, &ndp->d_ino, sizeof(ino_t)); 76 1.4 mrg odp->d_ino = (uint32_t)ino; 77 1.1 christos if (ndp->d_namlen >= sizeof(odp->d_name)) 78 1.1 christos odp->d_namlen = sizeof(odp->d_name) - 1; 79 1.1 christos else 80 1.1 christos odp->d_namlen = (u_int8_t)ndp->d_namlen; 81 1.1 christos odp->d_type = ndp->d_type; 82 1.1 christos (void)memcpy(odp->d_name, ndp->d_name, (size_t)odp->d_namlen); 83 1.1 christos odp->d_name[odp->d_namlen] = '\0'; 84 1.5 christos len = _DIRENT_SIZE(odp); 85 1.5 christos _DIAGASSERT(__type_fit(uint16_t, len)); 86 1.5 christos odp->d_reclen = (uint16_t)len; 87 1.1 christos odp = _DIRENT_NEXT(odp); 88 1.1 christos } 89 1.5 christos ptrdiff_t td = (((char *)(void *)odp) - buf); 90 1.5 christos _DIAGASSERT(__type_fit(int, td)); 91 1.5 christos return (int)td; 92 1.1 christos } 93