Archived
1
0
Fork 0
This repository has been archived on 2026-05-10. You can view files and clone it, but you cannot make any changes to it's state, such as pushing and creating new issues, pull requests or comments.
archived/src/inotify-backend.c
Henrik Hautakoski f46ae1970b Change indentation to follow the updated standard.
Alot of mixed indentation. use 4 chars wide soft tabs.
2011-03-19 12:28:48 +02:00

135 lines
3 KiB
C

/* inotify-backend.c
*
* Copyright (C) 2010-2011 Henrik Hautakoski <henrik@fiktivkod.org>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*/
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <time.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include "inotify-backend.h"
#include "inotify-syscalls.h"
#include "log.h"
#define IN_ONLYDIR 0x01000000
#define WATCH_MASK (IN_MOVE | IN_CREATE | IN_DELETE | IN_ONLYDIR)
#define PROCFS_PATH "/proc/sys/fs/inotify/"
#define PROCFS_QSIZE PROCFS_PATH "max_queued_events"
#define PROCFS_MAX_WATCHES PROCFS_PATH "max_user_watches"
static int fd = -1;
static unsigned inotify_qsize = 256;
static unsigned inotify_max_watches = 8192;
void read_unsigned_int(const char *filename, unsigned *val) {
int fd;
char buf[16];
fd = open(filename, O_RDONLY);
if (fd < 0)
return;
if (read(fd, buf, 15) > 0 && *buf) {
unsigned long tmp;
errno = 0;
tmp = strtoul(buf, NULL, 10);
if (!errno && tmp <= (unsigned)-1)
*val = tmp;
}
close(fd);
}
int inotify_backend_init(void) {
if (fd >= 0)
close(fd);
fd = inotify_init();
if (fd < 0)
return -errno;
read_unsigned_int(PROCFS_QSIZE, &inotify_qsize);
read_unsigned_int(PROCFS_MAX_WATCHES, &inotify_max_watches);
if (inotify_max_watches < 16384)
logmsg(LOG_WARN,
"inotify: the maximum number of inotify"
" watches is quite low (%u)."
" the application may not be able to track all files",
inotify_max_watches);
return fd;
}
void inotify_backend_exit(void) {
if (fd >= 0)
close(fd);
fd = -1;
}
int inotify_backend_watch(const char *path) {
int wd = inotify_add_watch(fd, path, WATCH_MASK);
if (wd < 0) {
if (errno != EACCES && errno != ENOTDIR)
logerrno(LOG_CRIT, "inotify_watch", errno);
return -errno;
}
return wd;
}
int inotify_backend_ignore(int wd) {
if (inotify_rm_watch(fd, wd) < 0) {
logerrno(LOG_CRIT, "intotify_ignore", errno);
return -errno;
}
return 0;
}
#define MAX_PENDING 5
#define THROTTLE_SLEEP_NS 2000000
#define UPPER_BOUND(size) ((unsigned) (size) >> 1)
int inotify_backend_read(void *buf, size_t size) {
struct timespec tres = { 0, THROTTLE_SLEEP_NS };
unsigned short tcount;
unsigned int ioready = 0;
for(tcount = 0; tcount < MAX_PENDING; tcount++) {
unsigned int events;
if (ioctl(fd, FIONREAD, &ioready) < 0)
break;
events = ioready / IN_EVENT_SIZE;
if (events > UPPER_BOUND(inotify_qsize))
goto do_read;
nanosleep(&tres, NULL);
}
if (ioready)
goto do_read;
return 0;
do_read:
return read(fd, buf, size);
}