usr/src/lib/libc/port/gen/readdir.c
author Jon Tibble <meths@btinternet.com>
Thu, 09 Dec 2010 22:32:39 +0100
changeset 13255 4afa820d78b9
parent 6812 febeba71273d
permissions -rw-r--r--
298 SPARC build fails in smt_pause.o 478 Build needs fixing for pkgdepend flag day Reviewed by: [email protected] Reviewed by: [email protected] Reviewed by: [email protected] Approved by: [email protected]
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 * CDDL HEADER START
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     3
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 * The contents of this file are subject to the terms of the
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
     5
 * Common Development and Distribution License (the "License").
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
     6
 * You may not use this file except in compliance with the License.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 * or http://www.opensolaris.org/os/licensing.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 * See the License for the specific language governing permissions
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 * and limitations under the License.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 * When distributing Covered Code, include this CDDL HEADER in each
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 * If applicable, add the following below this CDDL HEADER, with the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 * fields enclosed by brackets "[]" replaced with your own identifying
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 * information: Portions Copyright [yyyy] [name of copyright owner]
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
 * CDDL HEADER END
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
 */
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    21
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
/*
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3453
diff changeset
    23
 * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
 * Use is subject to license terms.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
/*	Copyright (c) 1988 AT&T	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
/*	  All Rights Reserved  	*/
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
 * University Copyright- Copyright (c) 1982, 1986, 1988
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
 * The Regents of the University of California
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
 * All Rights Reserved
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
 * University Acknowledgment- Portions of this document are derived from
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
 * software developed by the University of California, Berkeley, and its
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
 * contributors.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
 * readdir -- C library extension routine
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
#include	<sys/feature_tests.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
#if !defined(_LP64)
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3453
diff changeset
    49
#pragma weak _readdir64 = readdir64
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
#endif
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3453
diff changeset
    51
#pragma weak _readdir = readdir
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3453
diff changeset
    53
#include "lint.h"
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    54
#include <dirent.h>
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    55
#include <limits.h>
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    56
#include <errno.h>
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    57
#include "libc.h"
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
#ifdef _LP64
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    61
dirent_t *
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
readdir(DIR *dirp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
{
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    64
	dirent_t *dp;	/* -> directory data */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
	int saveloc = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
	if (dirp->dd_size != 0) {
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    68
		dp = (dirent_t *)(uintptr_t)&dirp->dd_buf[dirp->dd_loc];
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    69
		saveloc = dirp->dd_loc;		/* save for possible EOF */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
		dirp->dd_loc += (int)dp->d_reclen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
	if (dirp->dd_loc >= dirp->dd_size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
		dirp->dd_loc = dirp->dd_size = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
	if (dirp->dd_size == 0 && 	/* refill buffer */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
	    (dirp->dd_size = getdents(dirp->dd_fd,
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    77
	    (dirent_t *)(uintptr_t)dirp->dd_buf, DIRBUF)) <= 0) {
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    78
		if (dirp->dd_size == 0)		/* This means EOF */
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    79
			dirp->dd_loc = saveloc;	/* so save for telldir */
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    80
		return (NULL);		/* error or EOF */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    83
	return ((dirent_t *)(uintptr_t)&dirp->dd_buf[dirp->dd_loc]);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
#else	/* _LP64 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
 * Welcome to the complicated world of large files on a small system.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    92
dirent64_t *
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
readdir64(DIR *dirp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
{
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    95
	dirent64_t *dp64;	/* -> directory data */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
	int saveloc = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
	if (dirp->dd_size != 0) {
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
    99
		dp64 = (dirent64_t *)(uintptr_t)&dirp->dd_buf[dirp->dd_loc];
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
		/* was converted by readdir and needs to be reversed */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
		if (dp64->d_ino == (ino64_t)-1) {
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   102
			dirent_t *dp32;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   104
			dp32 = (dirent_t *)(&dp64->d_off);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
			dp64->d_ino = (ino64_t)dp32->d_ino;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
			dp64->d_off = (off64_t)dp32->d_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
			dp64->d_reclen = (unsigned short)(dp32->d_reclen +
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   108
			    ((char *)&dp64->d_off - (char *)dp64));
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
		}
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   110
		saveloc = dirp->dd_loc;		/* save for possible EOF */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
		dirp->dd_loc += (int)dp64->d_reclen;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
	if (dirp->dd_loc >= dirp->dd_size)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
		dirp->dd_loc = dirp->dd_size = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
	if (dirp->dd_size == 0 && 	/* refill buffer */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
	    (dirp->dd_size = getdents64(dirp->dd_fd,
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   118
	    (dirent64_t *)(uintptr_t)dirp->dd_buf, DIRBUF)) <= 0) {
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   119
		if (dirp->dd_size == 0)		/* This means EOF */
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   120
			dirp->dd_loc = saveloc;	/* so save for telldir */
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   121
		return (NULL);		/* error or EOF */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   124
	dp64 = (dirent64_t *)(uintptr_t)&dirp->dd_buf[dirp->dd_loc];
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
	return (dp64);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
 * readdir now does translation of dirent64 entries into dirent entries.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
 * We rely on the fact that dirents are smaller than dirent64s and we
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
 * reuse the space accordingly.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
 */
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   133
dirent_t *
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
readdir(DIR *dirp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
{
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   136
	dirent64_t *dp64;	/* -> directory data */
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   137
	dirent_t *dp32;		/* -> directory data */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
	if ((dp64 = readdir64(dirp)) == NULL)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
		return (NULL);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   141
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   142
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   143
	 * Make sure that the offset fits in 32 bits.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   144
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   145
	if (((off_t)dp64->d_off != dp64->d_off &&
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   146
	    (uint64_t)dp64->d_off > (uint64_t)UINT32_MAX) ||
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   147
	    dp64->d_ino > SIZE_MAX) {
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   148
		errno = EOVERFLOW;
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   149
		return (NULL);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   150
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   151
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   152
	dp32 = (dirent_t *)(&dp64->d_off);
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   153
	dp32->d_off = (off_t)dp64->d_off;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   154
	dp32->d_ino = (ino_t)dp64->d_ino;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   155
	dp32->d_reclen = (unsigned short)(dp64->d_reclen -
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   156
	    ((char *)&dp64->d_off - (char *)dp64));
3453
bc3a776612a0 6513772 readdir_r() and associates should use per-DIR locking
raf
parents: 0
diff changeset
   157
	dp64->d_ino = (ino64_t)-1;	/* flag as converted for readdir64 */
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   158
	/* d_name d_reclen should not move */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
	return (dp32);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   160
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   161
#endif	/* _LP64 */