xref: /freebsd/lib/libc/gen/telldir.h (revision d40daefca64750c1076822bdbd3c409a9519f513)
18a16b7a1SPedro F. Giffuni /*-
28a16b7a1SPedro F. Giffuni  * SPDX-License-Identifier: BSD-3-Clause
38a16b7a1SPedro F. Giffuni  *
410d1cba0SDaniel Eischen  * Copyright (c) 1983, 1993
510d1cba0SDaniel Eischen  *	The Regents of the University of California.  All rights reserved.
610d1cba0SDaniel Eischen  *
710d1cba0SDaniel Eischen  * Copyright (c) 2000
810d1cba0SDaniel Eischen  * 	Daniel Eischen.  All rights reserved.
910d1cba0SDaniel Eischen  *
1010d1cba0SDaniel Eischen  * Redistribution and use in source and binary forms, with or without
1110d1cba0SDaniel Eischen  * modification, are permitted provided that the following conditions
1210d1cba0SDaniel Eischen  * are met:
1310d1cba0SDaniel Eischen  * 1. Redistributions of source code must retain the above copyright
1410d1cba0SDaniel Eischen  *    notice, this list of conditions and the following disclaimer.
1510d1cba0SDaniel Eischen  * 2. Redistributions in binary form must reproduce the above copyright
1610d1cba0SDaniel Eischen  *    notice, this list of conditions and the following disclaimer in the
1710d1cba0SDaniel Eischen  *    documentation and/or other materials provided with the distribution.
1810d1cba0SDaniel Eischen  * 3. Neither the name of the University nor the names of its contributors
1910d1cba0SDaniel Eischen  *    may be used to endorse or promote products derived from this software
2010d1cba0SDaniel Eischen  *    without specific prior written permission.
2110d1cba0SDaniel Eischen  *
2210d1cba0SDaniel Eischen  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2310d1cba0SDaniel Eischen  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2410d1cba0SDaniel Eischen  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2510d1cba0SDaniel Eischen  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2610d1cba0SDaniel Eischen  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2710d1cba0SDaniel Eischen  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2810d1cba0SDaniel Eischen  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2910d1cba0SDaniel Eischen  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3010d1cba0SDaniel Eischen  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3110d1cba0SDaniel Eischen  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3210d1cba0SDaniel Eischen  * SUCH DAMAGE.
3310d1cba0SDaniel Eischen  */
3410d1cba0SDaniel Eischen 
3510d1cba0SDaniel Eischen #ifndef _TELLDIR_H_
3610d1cba0SDaniel Eischen #define	_TELLDIR_H_
3710d1cba0SDaniel Eischen 
3810d1cba0SDaniel Eischen #include <sys/queue.h>
399f72c032SJohn Baldwin #include <stdbool.h>
4010d1cba0SDaniel Eischen 
4110d1cba0SDaniel Eischen /*
4282241ed5SAlan Somers  * telldir will malloc one of these to describe the current directory position,
4382241ed5SAlan Somers  * if it can't fit that information into the packed structure below.  It
4482241ed5SAlan Somers  * records the current magic cookie returned by getdirentries and the offset
4582241ed5SAlan Somers  * within the buffer associated with that return value.
4610d1cba0SDaniel Eischen  */
4782241ed5SAlan Somers struct ddloc_mem {
4882241ed5SAlan Somers 	LIST_ENTRY(ddloc_mem) loc_lqe; /* entry in list */
4910d1cba0SDaniel Eischen 	long	loc_index;	/* key associated with structure */
5069921123SKonstantin Belousov 	off_t	loc_seek;	/* magic cookie returned by getdirentries */
5110d1cba0SDaniel Eischen 	long	loc_loc;	/* offset of entry in buffer */
5210d1cba0SDaniel Eischen };
5310d1cba0SDaniel Eischen 
5482241ed5SAlan Somers #ifdef __LP64__
5582241ed5SAlan Somers #define DD_LOC_BITS	31
5682241ed5SAlan Somers #define DD_SEEK_BITS	32
5782241ed5SAlan Somers #define DD_INDEX_BITS	63
5882241ed5SAlan Somers #else
5982241ed5SAlan Somers #define DD_LOC_BITS	12
6082241ed5SAlan Somers #define DD_SEEK_BITS	19
6182241ed5SAlan Somers #define DD_INDEX_BITS	31
6282241ed5SAlan Somers #endif
6382241ed5SAlan Somers 
6482241ed5SAlan Somers /*
6582241ed5SAlan Somers  * This is the real type returned by telldir.  telldir will prefer to return
6682241ed5SAlan Somers  * the packed type, if possible, or the malloced type otherwise.  For msdosfs,
6782241ed5SAlan Somers  * UFS, and NFS, directory positions usually fit within the packed type.  For
6882241ed5SAlan Somers  * ZFS and tmpfs, they usually fit within the packed type on 64-bit
6982241ed5SAlan Somers  * architectures.
7082241ed5SAlan Somers  */
7182241ed5SAlan Somers union ddloc_packed {
7282241ed5SAlan Somers 	long	l;		/* Opaque type returned by telldir(3) */
7382241ed5SAlan Somers 	struct {
7482241ed5SAlan Somers 		/* Identifies union type.  Must be 0. */
7582241ed5SAlan Somers 		unsigned long is_packed:1;
7682241ed5SAlan Somers 		/* Index into directory's linked list of ddloc_mem */
7782241ed5SAlan Somers 		unsigned long index:DD_INDEX_BITS;
7882241ed5SAlan Somers 	} __packed i;
7982241ed5SAlan Somers 	struct {
8082241ed5SAlan Somers 		/* Identifies union type.  Must be 1. */
8182241ed5SAlan Somers 		unsigned long is_packed:1;
8282241ed5SAlan Somers 		/* offset of entry in buffer*/
8382241ed5SAlan Somers 		unsigned long loc:DD_LOC_BITS;
8482241ed5SAlan Somers 		/* magic cookie returned by getdirentries */
8582241ed5SAlan Somers 		unsigned long seek:DD_SEEK_BITS;
8682241ed5SAlan Somers 	} __packed s;
8782241ed5SAlan Somers };
8882241ed5SAlan Somers 
8982241ed5SAlan Somers _Static_assert(sizeof(long) == sizeof(union ddloc_packed),
9082241ed5SAlan Somers     "packed telldir size mismatch!");
9182241ed5SAlan Somers 
9210d1cba0SDaniel Eischen /*
9310d1cba0SDaniel Eischen  * One of these structures is malloced for each DIR to record telldir
9410d1cba0SDaniel Eischen  * positions.
9510d1cba0SDaniel Eischen  */
9610d1cba0SDaniel Eischen struct _telldir {
9782241ed5SAlan Somers 	LIST_HEAD(, ddloc_mem) td_locq; /* list of locations */
9810d1cba0SDaniel Eischen 	long	td_loccnt;	/* index of entry for sequential readdir's */
9910d1cba0SDaniel Eischen };
10010d1cba0SDaniel Eischen 
1019f72c032SJohn Baldwin bool		_filldir(DIR *, bool);
10290c68c17SKonstantin Belousov struct dirent	*_readdir_unlocked(DIR *, int);
103d201fe46SDaniel Eischen void 		_reclaim_telldir(DIR *);
104d201fe46SDaniel Eischen void 		_seekdir(DIR *, long);
105af069b06SJulian Elischer void		_fixtelldir(DIR *dirp, long oldseek, long oldloc);
106*d40daefcSKonstantin Belousov DIR		*__opendir_common(int, int, bool);
10710d1cba0SDaniel Eischen 
10869921123SKonstantin Belousov #define	RDU_SKIP	0x0001
10969921123SKonstantin Belousov #define	RDU_SHORT	0x0002
11069921123SKonstantin Belousov 
11110d1cba0SDaniel Eischen #endif
112