This commit is contained in:
parent
ce0078081a
commit
45f4acd152
218
src/utils/closefrom.cpp
Normal file
218
src/utils/closefrom.cpp
Normal file
@ -0,0 +1,218 @@
|
||||
#ifndef lint
|
||||
static char rcsid[] = "@(#$Id: $ (C) 2009 J.F.Dockes";
|
||||
#endif
|
||||
/*
|
||||
* Close all file descriptors above a given value.
|
||||
*
|
||||
* A Unix execXX() call used to execute another program does not close open
|
||||
* file descriptors by default.
|
||||
*
|
||||
* The only descriptors closed are those on which the FD_CLOEXEC flag was
|
||||
* set. FD_CLOEXEC is not easily usable on files opened by external
|
||||
* libraries.
|
||||
*
|
||||
* There are many reasons for closing file descriptors before
|
||||
* an exec (security, pipe control, the possibility that a bug will trigger
|
||||
* an unwanted write, etc.)
|
||||
*
|
||||
* A process has currently no POSIX way to determine the list of open file
|
||||
* descriptors or at least the highest value. Closing all files (except a few),
|
||||
* thus implies performing a close() system call on each entry up to the
|
||||
* maximum, which can be both relatively difficult to determine, and quite
|
||||
* high (ie: several thousands), incurring a non-negligible cost.
|
||||
*
|
||||
* A number of systems have non-portable support for mitigating or solving
|
||||
* this problem.
|
||||
*
|
||||
* This module supplies a portable interface to this functionality.
|
||||
*
|
||||
* The initial data on system interfaces was obtained from:
|
||||
* http://stackoverflow.com/questions/899038/\
|
||||
* getting-the-highest-allocated-file-descriptor
|
||||
*
|
||||
* System interfaces:
|
||||
* FreeBSD:
|
||||
* - Has a closefrom() system call as of release 7.x around Sep 2009
|
||||
* - Has a /dev/fd, directory which lists the current process' open
|
||||
* descriptors. Only descriptors 0, 1, 2 are shown except if
|
||||
* fdescfs is mounted which it is not by default
|
||||
*
|
||||
* Interface:
|
||||
* int libclf_closefrom(fd)
|
||||
* @param fd All open file descriptors with equal or higher numeric
|
||||
* values will be closed. fd needs not be a valid descriptor.
|
||||
* @return 0 for success, -1 for error.
|
||||
*/
|
||||
#ifndef TEST_CLOSEFROM
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/param.h>
|
||||
|
||||
/* #define DEBUG_CLOSEFROM*/
|
||||
#ifdef DEBUG_CLOSEFROM
|
||||
#define DPRINT(X) fprintf X
|
||||
#else
|
||||
#define DPRINT(X)
|
||||
#endif
|
||||
|
||||
/* Note: sudo has a closefrom implementation, needs a lot of autoconfig, but
|
||||
* we could use it instead. It's quite close to this though */
|
||||
|
||||
/*************************************************************************/
|
||||
|
||||
/* closefrom() exists on Solaris, netbsd and openbsd, but someone will
|
||||
* have to provide me the appropriate macro to test */
|
||||
#if (defined(__FreeBSD__) && __FreeBSD_version >= 702104)
|
||||
/* Use closefrom() system call */
|
||||
int libclf_closefrom(int fd0)
|
||||
{
|
||||
DPRINT((stderr, "libclf_closefrom: using closefrom(2)\n"));
|
||||
closefrom(fd0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*************************************************************************/
|
||||
#elif defined(F_CLOSEM)
|
||||
|
||||
/* Use fcntl(fd, F_CLOSEM) */
|
||||
|
||||
int libclf_closefrom(int fd0)
|
||||
{
|
||||
DPRINT((stderr, "libclf_closefrom: using fcntl(F_CLOSEM)\n"));
|
||||
// We need a valid descriptor for this to work. Try to dup stdin, else
|
||||
// go wild
|
||||
if (fcntl(0, F_GETFL) != -1) {
|
||||
if (fd0 != 0)
|
||||
dup2(0, fd0);
|
||||
} else {
|
||||
int fd = open("/etc/group", 0); // yes i am a unix man
|
||||
if (fd >= 0 && fd != fd0) {
|
||||
dup2(fd, fd0);
|
||||
close(fd);
|
||||
}
|
||||
}
|
||||
return fcntl(fd0, F_CLOSEM, 0);
|
||||
}
|
||||
|
||||
/*************************************************************************/
|
||||
#elif defined(linux)
|
||||
|
||||
/* Use /proc/self/fd directory */
|
||||
#include <sys/types.h>
|
||||
#include <dirent.h>
|
||||
|
||||
int libclf_closefrom(int fd0)
|
||||
{
|
||||
DIR *dirp;
|
||||
struct dirent *ent;
|
||||
|
||||
DPRINT((stderr, "libclf_closefrom: using /proc\n"));
|
||||
dirp = opendir("/proc/self/fd");
|
||||
if (dirp == 0)
|
||||
return -1;
|
||||
|
||||
while ((ent = readdir(dirp)) != 0) {
|
||||
int fd;
|
||||
if (!strcmp(ent->d_name, ".") || !strcmp(ent->d_name, ".."))
|
||||
continue;
|
||||
|
||||
if (sscanf(ent->d_name, "%d", &fd) == 1 && fd >= fd0 &&
|
||||
fd != dirfd(dirp)) {
|
||||
close(fd);
|
||||
}
|
||||
}
|
||||
closedir(dirp);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*************************************************************************/
|
||||
#else
|
||||
/* System has no native support for this functionality whatsoever.
|
||||
*
|
||||
* Close all descriptors up to compiled/configured maximum.
|
||||
* The caller will usually have an idea of a reasonable maximum, else
|
||||
* we retrieve a value from the system.
|
||||
*/
|
||||
|
||||
static int closefrom_maxfd = -1;
|
||||
|
||||
void libclf_setmaxfd(int max)
|
||||
{
|
||||
closefrom_maxfd = max;
|
||||
}
|
||||
|
||||
int libclf_closefrom(int fd0)
|
||||
{
|
||||
int i, maxfd = closefrom_maxfd;
|
||||
|
||||
if (maxfd < 0) {
|
||||
#ifdef _SC_OPEN_MAX
|
||||
maxfd = sysconf(_SC_OPEN_MAX);
|
||||
DPRINT((stderr, "Maxfd is %d after sysconf()\n", maxfd));
|
||||
#else
|
||||
maxfd = getdtablesize();
|
||||
DPRINT((stderr, "Maxfd is %d after getdtablesize()\n", maxfd));
|
||||
#endif
|
||||
}
|
||||
if (maxfd < 0)
|
||||
maxfd = OPEN_MAX;
|
||||
|
||||
DPRINT((stderr, "libclf_closefrom: using loop to %d\n", maxfd));
|
||||
|
||||
for (i = fd0; i < maxfd; i++) {
|
||||
(void)close(i);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
#else /* TEST_CLOSEFROM */
|
||||
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "closefrom.h"
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
int i;
|
||||
|
||||
int fd0 = open("/etc/group", 0);
|
||||
if (fd0 < 0) {
|
||||
perror("open /etc/group");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (dup2(fd0, 11) < 0) {
|
||||
perror("dup2->11");
|
||||
exit(1);
|
||||
}
|
||||
if (dup2(fd0, 19) < 0) {
|
||||
perror("dup2->19");
|
||||
exit(1);
|
||||
}
|
||||
if (dup2(fd0, 99)< 0) {
|
||||
perror("dup2->99 (ok)");
|
||||
}
|
||||
if (dup2(fd0, 999) < 0) {
|
||||
perror("dup3->999 (ok)");
|
||||
}
|
||||
|
||||
libclf_closefrom(11);
|
||||
for (i = 0; i < 10000; i++) {
|
||||
if (fcntl(i, F_GETFL) != -1) {
|
||||
fprintf(stderr, "Descriptor %d is still open", i);
|
||||
if (i < 11)
|
||||
fprintf(stderr, " (OK)\n");
|
||||
else
|
||||
fprintf(stderr, " (BAD)\n");
|
||||
}
|
||||
}
|
||||
exit(0);
|
||||
}
|
||||
|
||||
#endif
|
||||
8
src/utils/closefrom.h
Normal file
8
src/utils/closefrom.h
Normal file
@ -0,0 +1,8 @@
|
||||
#ifndef _closefrom_h_included_
|
||||
#define _closefrom_h_included_
|
||||
/* @(#$Id: $ (C) 2004 J.F.Dockes */
|
||||
|
||||
/* Close all descriptors >= fd */
|
||||
extern int libclf_closefrom(int fd);
|
||||
|
||||
#endif /* _closefrom_h_included_ */
|
||||
Loading…
x
Reference in New Issue
Block a user