inotify

作者: 如果听见下雨的声音 | 来源:发表于2020-04-21 14:40 被阅读0次

    监听目录状态变化

    #include <errno.h>
    #include <poll.h>
    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/inotify.h>
    #include <unistd.h>
    
    static void
    handle_events(int fd, int *wd, int argc, char *argv[])
    {
        /* Some systems cannot read integer variables if they are not
           properly aligned. On other systems, incorrect alignment may
           decrease performance. Hence, the buffer used for reading from
           the inotify file descriptor should have the same alignment as
           struct inotify_event. */
    
        char buf[4096]
                __attribute__ ((aligned(__alignof__(struct inotify_event))));
        const struct inotify_event *event;
        int i;
        ssize_t len;
        char *ptr;
    
        /* Loop while events can be read from inotify file descriptor. */
    
        for (;;)
        {
    
            /* Read some events. */
    
            len = read(fd, buf, sizeof buf);
            if (len == -1 && errno != EAGAIN)
            {
                perror("read");
                exit(EXIT_FAILURE);
            }
    
            /* If the nonblocking read() found no events to read, then
               it returns -1 with errno set to EAGAIN. In that case,
               we exit the loop. */
    
            if (len <= 0)
                break;
    
            /* Loop over all events in the buffer */
    
            for (ptr = buf; ptr < buf + len;
                 ptr += sizeof(struct inotify_event) + event->len)
            {
    
                event = (const struct inotify_event *) ptr;
    
                /* Print event type */
    
                if (event->mask & IN_OPEN)
                    printf("IN_OPEN: ");
                if (event->mask & IN_CLOSE_NOWRITE)
                    printf("IN_CLOSE_NOWRITE: ");
                if (event->mask & IN_CLOSE_WRITE)
                    printf("IN_CLOSE_WRITE: ");
    
                /* Print the name of the watched directory */
    
                for (i = 1; i < argc; ++i)
                {
                    if (wd[i] == event->wd)
                    {
                        printf("%s/", argv[i]);
                        break;
                    }
                }
    
                /* Print the name of the file */
    
                if (event->len)
                    printf("%s", event->name);
    
                /* Print type of filesystem object */
    
                if (event->mask & IN_ISDIR)
                    printf(" [directory]\n");
                else
                    printf(" [file]\n");
            }
        }
    }
    
    
    void inotify_with_c(int argc, char *argv[])
    {
        char buf;
        int fd, i, poll_num;
        int *wd;
        nfds_t nfds;
        struct pollfd fds[2];
    
        if (argc < 2)
        {
            printf("Usage: %s PATH [PATH ...]\n", argv[0]);
            exit(EXIT_FAILURE);
        }
    
        printf("Press ENTER key to terminate.\n");
    
        /* Create the file descriptor for accessing the inotify API */
    
        fd = inotify_init1(IN_NONBLOCK);
        if (fd == -1)
        {
            perror("inotify_init1");
            exit(EXIT_FAILURE);
        }
    
        /* Allocate memory for watch descriptors */
    
        wd = static_cast<int *>(calloc(argc, sizeof(int)));
        if (wd == NULL)
        {
            perror("calloc");
            exit(EXIT_FAILURE);
        }
    
        /* Mark directories for events
           - file was opened
           - file was closed */
    
        for (i = 1; i < argc; i++)
        {
            wd[i] = inotify_add_watch(fd, argv[i],
                                      IN_OPEN | IN_CLOSE);
            if (wd[i] == -1)
            {
                fprintf(stderr, "Cannot watch '%s'\n", argv[i]);
                perror("inotify_add_watch");
                exit(EXIT_FAILURE);
            }
        }
    
        /* Prepare for polling */
    
        nfds = 2;
    
        /* Console input */
    
        fds[0].fd = STDIN_FILENO;
        fds[0].events = POLLIN;
    
        /* Inotify input */
    
        fds[1].fd = fd;
        fds[1].events = POLLIN;
    
        /* Wait for events and/or terminal input */
    
        printf("Listening for events.\n");
        while (1)
        {
            poll_num = poll(fds, nfds, -1);
            if (poll_num == -1)
            {
                if (errno == EINTR)
                    continue;
                perror("poll");
                exit(EXIT_FAILURE);
            }
    
            if (poll_num > 0)
            {
    
                if (fds[0].revents & POLLIN)
                {
    
                    /* Console input is available. Empty stdin and quit */
    
                    while (read(STDIN_FILENO, &buf, 1) > 0 && buf != '\n')
    
                        continue;
                    break;
                }
    
                if (fds[1].revents & POLLIN)
                {
    
                    /* Inotify events are available */
    
                    handle_events(fd, wd, argc, argv);
                }
            }
        }
    
        printf("Listening for events stopped.\n");
    
        /* Close inotify file descriptor */
    
        close(fd);
    
        free(wd);
        exit(EXIT_SUCCESS);
    
    }
    
    int main()
    {
        const char *argv[] = {"test", "/tmp"};
    
        inotify_with_c(sizeof argv / sizeof argv[0], const_cast<char **>(argv));
    
        return 0;
    }
    

    相关文章

      网友评论

          本文标题:inotify

          本文链接:https://www.haomeiwen.com/subject/pvvuihtx.html