usr/src/lib/libc/port/gen/closefrom.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
3636
1f5a3ccc94df 6521689 closefrom() does not close all file descriptors
raf
parents: 0
diff changeset
     5
 * Common Development and Distribution License (the "License").
1f5a3ccc94df 6521689 closefrom() does not close all file descriptors
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
 */
3636
1f5a3ccc94df 6521689 closefrom() does not close all file descriptors
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: 3636
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
#pragma ident	"%Z%%M%	%I%	%E% SMI"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3636
diff changeset
    29
#pragma weak _closefrom = closefrom
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3636
diff changeset
    30
#pragma weak _fdwalk = fdwalk
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3636
diff changeset
    32
#include "lint.h"
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
#include <ctype.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
#include <stdlib.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
#include <unistd.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
#include <string.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
#include <fcntl.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
#include <dirent.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
#include <limits.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
#include <errno.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
#include <alloca.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
#include <sys/resource.h>
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
/* Initial size of the open file descriptor array */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
#define	FDS_SZ	(1024 * sizeof (int))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
 * Iterate over all open file descriptors, calling 'func' on each one.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
 * Terminate the iteration when 'func' returns non-zero or when all
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
 * open file descriptors have been processed.  Return the value of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
 * the last non-zero return from 'func' or zero.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
fdwalk(int (*func)(void *, int), void *cd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
	int err = errno;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
	int rv = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
	int max_fds = INT_MAX;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
	struct rlimit rl;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
	DIR *dirp;
3636
1f5a3ccc94df 6521689 closefrom() does not close all file descriptors
raf
parents: 0
diff changeset
    61
	struct dirent64 *dp;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
	int *fds;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
	size_t fds_sz;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
	int nfds;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
	int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
	nfds = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
	fds = alloca(FDS_SZ);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
	fds_sz = FDS_SZ;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
	if ((dirp = opendir("/proc/self/fd")) != NULL) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
		 * Collect all of the open file descriptors and close
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
		 * the directory before calling 'func' on any of them.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
		 */
3636
1f5a3ccc94df 6521689 closefrom() does not close all file descriptors
raf
parents: 0
diff changeset
    75
		while ((dp = readdir64(dirp)) != NULL) {
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
			/* skip '.', '..' and the opendir() fd */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
			if (!isdigit(dp->d_name[0]) ||
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
			    (i = atoi(dp->d_name)) == dirp->dd_fd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
			if (fds_sz <= nfds * sizeof (int)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
				fds = memcpy(alloca(fds_sz * 2), fds, fds_sz);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
				fds_sz *= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
			fds[nfds++] = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
		(void) closedir(dirp);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
	} else {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
		/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
		 * We could not open the /proc file descriptor directory.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
		 * We have to do it the hard way.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
		 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
		if (getrlimit(RLIMIT_NOFILE, &rl) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
			max_fds = (rl.rlim_max == RLIM_INFINITY)?
6812
febeba71273d PSARC 2008/309 expunge synonyms.h
raf
parents: 3636
diff changeset
    94
			    INT_MAX : rl.rlim_max;
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
		for (i = 0; i < max_fds; i++) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
			if (fcntl(i, F_GETFD) < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
				continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
			if (fds_sz <= nfds * sizeof (int)) {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
				fds = memcpy(alloca(fds_sz * 2), fds, fds_sz);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
				fds_sz *= 2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
			}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
			fds[nfds++] = i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
	 * Restore the original value of errno so that
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
	 * the caller sees only the value of errno set
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
	 * by the callback function.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
	errno = err;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
	/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
	 * Perform the callbacks on all of the open files.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
	for (i = 0; i < nfds; i++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
		if ((rv = func(cd, fds[i])) != 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
			break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
	return (rv);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
 * Call-back function for closefrom(), below.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
static int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
void_close(void *lowp, int fd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
	if (fd >= *(int *)lowp)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
		(void) close(fd);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
	return (0);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
 * Close all open file descriptors greater than or equal to lowfd.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
void
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
closefrom(int lowfd)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
	int low = (lowfd < 0)? 0 : lowfd;
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
	 * Close lowfd right away as a hedge against failing
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   144
	 * to open the /proc file descriptor directory due
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   145
	 * all file descriptors being currently used up.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   146
	 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   147
	(void) close(low++);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   148
	(void) fdwalk(void_close, &low);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   149
}