Поиск открытых файловых дескрипторов для процесса Linux (код C)? - PullRequest
25 голосов
/ 05 июля 2011

Я хотел найти все открытые fds для процесса в linux.

Могу ли я сделать это с помощью функций библиотеки glib?

Ответы [ 6 ]

27 голосов
/ 17 марта 2012

Вот некоторый код, который я использовал, я не знал о / proc / self (спасибо Донал!), Но этот способ, вероятно, в любом случае более общий. Я включил необходимые функции для всех функций вверху.

#include <string.h>
#include <stdio.h>
#include <dirent.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <sys/resource.h>

#ifndef FALSE
#define FALSE (0)
#endif
#ifndef TRUE
#define TRUE (!FALSE)
#endif

/* implementation of Donal Fellows method */ 
int get_num_fds()
{
     int fd_count;
     char buf[64];
     struct dirent *dp;

     snprintf(buf, 64, "/proc/%i/fd/", getpid());

     fd_count = 0;
     DIR *dir = opendir(buf);
     while ((dp = readdir(dir)) != NULL) {
          fd_count++;
     }
     closedir(dir);
     return fd_count;
}

Однажды я столкнулся с очень плохой проблемой утечки файловых дескрипторов, и оказалось, что я на самом деле закодировал решение, которое Том Х. предложил:

/* check whether a file-descriptor is valid */
int pth_util_fd_valid(int fd)
{
     if (fd < 3 || fd >= FD_SETSIZE)
          return FALSE;
     if (fcntl(fd, F_GETFL) == -1 && errno == EBADF)
          return FALSE;
     return TRUE;
}

/* check first 1024 (usual size of FD_SESIZE) file handles */
int test_fds()
{
     int i;
     int fd_dup;
     char errst[64];
     for (i = 0; i < FD_SETSIZE; i++) {
          *errst = 0;
          fd_dup = dup(i);
          if (fd_dup == -1) {
                strcpy(errst, strerror(errno));
                // EBADF  oldfd isn’t an open file descriptor, or newfd is out of the allowed range for file descriptors.
                // EBUSY  (Linux only) This may be returned by dup2() during a race condition with open(2) and dup().
                // EINTR  The dup2() call was interrupted by a signal; see signal(7).
                // EMFILE The process already has the maximum number of file descriptors open and tried to open a new one.
          } else {
                close(fd_dup);
                strcpy(errst, "dup() ok");
          }
          printf("%4i: %5i %24s %s\n", i, fcntl(i, F_GETOWN), fd_info(i), errst);
     }
     return 0;
}

Вы, вероятно, захотите это тоже, чтобы удовлетворить последний printf выше ...

char *fcntl_flags(int flags)
{
    static char output[128];
    *output = 0;

    if (flags & O_RDONLY)
        strcat(output, "O_RDONLY ");
    if (flags & O_WRONLY)
        strcat(output, "O_WRONLY ");
    if (flags & O_RDWR)
        strcat(output, "O_RDWR ");
    if (flags & O_CREAT)
        strcat(output, "O_CREAT ");
    if (flags & O_EXCL)
        strcat(output, "O_EXCL ");
    if (flags & O_NOCTTY)
        strcat(output, "O_NOCTTY ");
    if (flags & O_TRUNC)
        strcat(output, "O_TRUNC ");
    if (flags & O_APPEND)
        strcat(output, "O_APPEND ");
    if (flags & O_NONBLOCK)
        strcat(output, "O_NONBLOCK ");
    if (flags & O_SYNC)
        strcat(output, "O_SYNC ");
    if (flags & O_ASYNC)
        strcat(output, "O_ASYNC ");

    return output;
}

char *fd_info(int fd)
{
    if (fd < 0 || fd >= FD_SETSIZE)
        return FALSE;
    // if (fcntl(fd, F_GETFL) == -1 && errno == EBADF)
    int rv = fcntl(fd, F_GETFL);
    return (rv == -1) ? strerror(errno) : fcntl_flags(rv);
}

FD_SETSIZE обычно составляет 1024, а максимальное количество файлов на процесс обычно составляет 1024. Если вы хотите быть уверенным, вы можете заменить его вызовом этой функции, как описано TomH.

#include <sys/time.h>
#include <sys/resource.h>

rlim_t get_rlimit_files()
{
    struct rlimit rlim;
    getrlimit(RLIMIT_NOFILE, &rlim);
    return rlim.rlim_cur;
}   

Если вы соберете все это в один файл (что я и сделал, просто чтобы проверить), вы можете создать вывод, похожий на этот, чтобы подтвердить, что он работает так, как объявлено:

0:     0                  O_RDWR  dup() ok
1:     0                O_WRONLY  dup() ok
2:     0                  O_RDWR  dup() ok
3:     0              O_NONBLOCK  dup() ok
4:     0     O_WRONLY O_NONBLOCK  dup() ok
5:    -1      Bad file descriptor Bad file descriptor
6:    -1      Bad file descriptor Bad file descriptor
7:    -1      Bad file descriptor Bad file descriptor
8:    -1      Bad file descriptor Bad file descriptor
9:    -1      Bad file descriptor Bad file descriptor

Я надеюсь, что ответит на все ваши вопросы, и, если вам интересно, я действительно пришел сюда, чтобы найти ответ на вопрос, заданный ОП, и, прочитав ответ, помните, что я уже написал код много лет назад. Наслаждайтесь.

26 голосов
/ 05 июля 2011

Поскольку вы работаете в Linux, у вас (почти наверняка) смонтирована файловая система /proc. Это означает, что самый простой способ - получить список содержимого /proc/self/fd; каждый файл там назван в честь FD. (Конечно, используйте g_dir_open, g_dir_read_name и g_dir_close для составления списка.)

В противном случае получение информации умеренно неудобно (например, нет полезного API POSIX; это область, которая не была стандартизирована).

6 голосов
/ 05 июля 2011

Если вы можете идентифицировать процесс с помощью pid, вы можете просто сделать

ls -l /proc/<pid>/fd | wc - l

В C вы можете передавать все и повторно использовать либо вывод, либо вы можете сами подсчитать файлы в вышеупомянутой директории (countнапример, здесь Подсчет количества файлов в каталоге с использованием C )

3 голосов
/ 31 октября 2014

Иногда C ++ - вариант, решение Донала, использующее boost :: filesystem:

#include <iostream>
#include <string>
#include <boost/filesystem.hpp>
#include <unistd.h>

namespace fs = boost::filesystem;

int main()
{
    std::string path = "/proc/" + std::to_string(::getpid()) + "/fd/";
    unsigned count = std::distance(fs::directory_iterator(path),
                                   fs::directory_iterator());
    std::cout << "Number of opened FDs: " << count << std::endl;
}
2 голосов
/ 05 июля 2011

Если вы имеете в виду, как вы можете сделать это программно изнутри процесса, тогда обычный (если немного неприятный) метод - это сделать что-то вроде зацикливания всех возможных дескрипторов (используйте getrlimit(), чтобы прочитать RLIMIT_NOFILE, чтобы найти диапазон) вызывая что-то вроде fcntl(fd, F_GETFD, 0) для каждого и проверяя ответы EBADF, чтобы увидеть, какие из них не открыты.

Если вы имеете в виду, что хотите узнать из оболочки, какие файлы открыты у процесса, тогда lsof -p <pid> - это то, что вам нужно.

1 голос
/ 05 июля 2011

Команда fstat выводит список всех запущенных процессов системы и их открытых дескрипторов, кроме того, она указывает, какой это тип дескриптора (файл, сокет, канал и т. Д.), И пытается дать подсказку о том, что дескриптор читает или пишет на такомкакая файловая система и какой номер индекса в этой файловой системе

...