You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							1006 lines
						
					
					
						
							27 KiB
						
					
					
				
			
		
		
	
	
							1006 lines
						
					
					
						
							27 KiB
						
					
					
				| # --- SDE-COPYRIGHT-NOTE-BEGIN --- | |
| # This copyright note is auto-generated by ./scripts/Create-CopyPatch. | |
| # | |
| # Filename: package/.../fam/fam-2.7.0-dnotify-1.patch | |
| # Copyright (C) 2010 The OpenSDE Project | |
| # | |
| # More information can be found in the files COPYING and README. | |
| # | |
| # This patch file is dual-licensed. It is available under the license the | |
| # patched project is licensed under, as long as it is an OpenSource license | |
| # as defined at http://www.opensource.org/ (e.g. BSD, X11) or under the terms | |
| # of the GNU General Public License as published by the Free Software | |
| # Foundation; either version 2 of the License, or (at your option) any later | |
| # version. | |
| # --- SDE-COPYRIGHT-NOTE-END --- | |
|  | |
| Submitted By: Jim Gifford (jim at linuxfromscratch dot org) | |
| Date: 2004-03-15 | |
| Initial Package Version: Jim Gifford | |
| Origin: Debian Maintainer of FAM | |
| Description: Dnotify Patch for FAM 2.7.0 | |
|   | |
| diff -Naur fam-2.7.0.orig/config.h.in fam-2.7.0/config.h.in | |
| --- fam-2.7.0.orig/config.h.in	2003-01-20 00:40:15.000000000 +0000 | |
| +++ fam-2.7.0/config.h.in	2004-03-15 21:31:34.553059530 +0000 | |
| @@ -180,3 +180,6 @@ | |
|   | |
|  /* Define to `int' if <sys/types.h> doesn't define. */ | |
|  #undef uid_t | |
| + | |
| +/* Define to 1 if you have F_NOTIFY fcntl */ | |
| +#undef USE_DNOTIFY | |
| diff -Naur fam-2.7.0.orig/configure.ac fam-2.7.0/configure.ac | |
| --- fam-2.7.0.orig/configure.ac	2003-11-26 19:47:59.000000000 +0000 | |
| +++ fam-2.7.0/configure.ac	2004-03-15 21:31:34.556058981 +0000 | |
| @@ -33,7 +33,26 @@ | |
|  AC_HEADER_DIRENT | |
|  AC_CHECK_HEADERS([fcntl.h limits.h linux/imon.h netinet/in.h rpc/rpc.h rpcsvc/mount.h stddef.h stdlib.h string.h syslog.h sys/imon.h sys/param.h sys/select.h sys/statvfs.h sys/syssgi.h sys/time.h sys/types.h sys/un.h unistd.h]) | |
|   | |
| -if test "$have_sys_imon_h"; then | |
| +# Test for the linux dnotify fcntl | |
| +AC_MSG_CHECKING([for dnotify fcntl support]) | |
| +AC_TRY_COMPILE([ | |
| +#define _GNU_SOURCE   | |
| +#include <fcntl.h> | |
| +#include <unistd.h> | |
| +], | |
| +[ int fd = 1; | |
| +  fcntl (fd, F_NOTIFY, (DN_MODIFY|DN_CREATE|DN_DELETE|DN_RENAME|DN_ATTRIB) | |
| +                       |DN_MULTISHOT); | |
| +], have_dnotify=yes, have_dnotify=no) | |
| + | |
| +use_dnotify=false | |
| +AC_MSG_RESULT($have_dnotify) | |
| + | |
| +if test "$have_dnotify"; then | |
| +        MONITOR_FUNCS=IMonNone | |
| +        AC_DEFINE([USE_DNOTIFY], [], [Use dnotify]) | |
| +	use_dnotify=true | |
| +elif test "$have_sys_imon_h"; then | |
|  	MONITOR_FUNCS=IMonIRIX | |
|  elif test "$have_linux_imon_h"; then | |
|  	MONITOR_FUNCS=IMonLinux | |
| @@ -41,6 +60,7 @@ | |
|  	MONITOR_FUNCS=IMonNone | |
|  fi | |
|  AC_SUBST(MONITOR_FUNCS) | |
| +AM_CONDITIONAL(USE_DNOTIFY, $use_dnotify) | |
|   | |
|  # Checks for typedefs, structures, and compiler characteristics. | |
|  AC_HEADER_STDBOOL | |
| diff -Naur fam-2.7.0.orig/src/DNotify.c++ fam-2.7.0/src/DNotify.c++ | |
| --- fam-2.7.0.orig/src/DNotify.c++	1970-01-01 00:00:00.000000000 +0000 | |
| +++ fam-2.7.0/src/DNotify.c++	2004-03-15 21:31:34.542061543 +0000 | |
| @@ -0,0 +1,582 @@ | |
| +//  Copyright (C) 2001 Red Hat, Inc.  All Rights Reserved. | |
| +//  Copyright (C) 1999 Silicon Graphics, Inc.  All Rights Reserved. | |
| +//   | |
| +//  This program is free software; you can redistribute it and/or modify it | |
| +//  under the terms of version 2 of the GNU General Public License as | |
| +//  published by the Free Software Foundation. | |
| +// | |
| +//  This program is distributed in the hope that it would be useful, but | |
| +//  WITHOUT ANY WARRANTY; without even the implied warranty of | |
| +//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  Further, any | |
| +//  license provided herein, whether implied or otherwise, is limited to | |
| +//  this program in accordance with the express provisions of the GNU | |
| +//  General Public License.  Patent licenses, if any, provided herein do not | |
| +//  apply to combinations of this program with other product or programs, or | |
| +//  any other product whatsoever.  This program is distributed without any | |
| +//  warranty that the program is delivered free of the rightful claim of any | |
| +//  third person by way of infringement or the like.  See the GNU General | |
| +//  Public License for more details. | |
| +// | |
| +//  You should have received a copy of the GNU General Public License along | |
| +//  with this program; if not, write the Free Software Foundation, Inc., 59 | |
| +//  Temple Place - Suite 330, Boston MA 02111-1307, USA. | |
| + | |
| +#define _GNU_SOURCE   | |
| +#include <fcntl.h> | |
| + | |
| +#include <string.h> | |
| +#include <signal.h> | |
| +#include <stdio.h> | |
| +#include <unistd.h> | |
| +#include <sys/types.h> | |
| +#include <sys/stat.h> | |
| +#include <libgen.h> | |
| + | |
| +#include "DNotify.h" | |
| + | |
| +#include "Interest.h" | |
| +#include "Log.h" | |
| +#include "Scheduler.h" | |
| +#include <memory> | |
| + | |
| + | |
| +int DNotify::pipe_write_fd = -2; | |
| +int DNotify::pipe_read_fd = -2; | |
| +volatile sig_atomic_t DNotify::queue_overflowed = 0; | |
| +volatile sig_atomic_t DNotify::queue_changed = 0; | |
| +int DNotify::change_queue[QUEUESIZE]; | |
| +volatile int DNotify::queue_head = 0; // Only modified by read handler | |
| +volatile int DNotify::queue_tail = 0; // Only modified by signal handler | |
| +DNotify::EventHandler DNotify::ehandler; | |
| + | |
| +DNotify::DirWatch *DNotify::dir_hash[DIR_HASHSIZE]; | |
| +DNotify::FileWatch *DNotify::file_hash[FILE_HASHSIZE]; | |
| + | |
| +struct DNotify::FileWatch | |
| +{ | |
| +    DirWatch *dir_watch; | |
| +    dev_t file_dev; | |
| +    ino_t file_ino; | |
| +    FileWatch *next; // The DirWatch.watches list | |
| +    FileWatch *hash_link; | |
| +}; | |
| + | |
| +struct DNotify::DirWatch | |
| +{ | |
| +    int fd; | |
| +    dev_t dir_dev; | |
| +    ino_t dir_ino; | |
| +     | |
| +    DirWatch *hash_link; | |
| +    FileWatch *watches; | |
| +}; | |
| + | |
| +struct DNotify::ChangeEventData | |
| +{ | |
| +    dev_t file_dev; | |
| +    ino_t file_ino; | |
| +};     | |
| + | |
| +DNotify::DNotify(EventHandler h) | |
| +{ | |
| +    assert(ehandler == NULL); | |
| +    ehandler = h; | |
| +} | |
| + | |
| +DNotify::~DNotify() | |
| +{ | |
| +    if (pipe_read_fd >= 0) | |
| +    { | |
| +	//  Tell the scheduler. | |
| + | |
| +	(void) Scheduler::remove_read_handler(pipe_read_fd); | |
| + | |
| +	//  Close the pipe. | |
| + | |
| +	if (close(pipe_read_fd) < 0) | |
| +	    Log::perror("can't pipe read end"); | |
| +	else | |
| +	    Log::debug("closed pipe read end"); | |
| +	 | |
| +	if (close(pipe_write_fd) < 0) | |
| +	    Log::perror("can't pipe write end"); | |
| +	else | |
| +	    Log::debug("closed pipe write end"); | |
| +	pipe_read_fd = -1; | |
| +    } | |
| +    ehandler = NULL; | |
| +} | |
| + | |
| +void | |
| +DNotify::overflow_signal_handler(int sig, siginfo_t *si, void *data) | |
| +{ | |
| +  char c = 'x'; | |
| + | |
| +  { | |
| +    char *str = "*************** overflow sigqueue ***********************\n"; | |
| +    write (STDERR_FILENO, str, strlen(str)); | |
| +  } | |
| + | |
| +  if (!queue_overflowed) | |
| +  { | |
| +      queue_overflowed = 1; | |
| +      // Trigger the read handler | |
| +      write(pipe_write_fd, &c, 1); | |
| +  } | |
| +} | |
| + | |
| +void | |
| +DNotify::signal_handler(int sig, siginfo_t *si, void *data) | |
| +{ | |
| +  int left; | |
| +  char c = 'x'; | |
| + | |
| +  if (queue_head <= queue_tail) | |
| +    left = (QUEUESIZE + queue_head) - queue_tail; | |
| +  else  | |
| +    left = queue_head - queue_tail; | |
| +   | |
| +  // Must leave at least one item unused to see difference | |
| +  // Betweeen empty and full | |
| +  if (left <= 1) | |
| +  { | |
| +      queue_overflowed = 1; | |
| +      { | |
| +	char *str = "*************** overflow famqueue ****************\n"; | |
| +	write (STDERR_FILENO, str, strlen(str)); | |
| +      } | |
| +  } | |
| +  else | |
| +  { | |
| +      change_queue[queue_tail] = si->si_fd; | |
| +      queue_tail = (queue_tail + 1) % QUEUESIZE; | |
| +  } | |
| +   | |
| +  if (!queue_changed) | |
| +  { | |
| +      queue_changed = 1; | |
| +      // Trigger the read handler | |
| +      write(pipe_write_fd, &c, 1); | |
| +  } | |
| +} | |
| + | |
| +bool | |
| +DNotify::is_active() | |
| +{ | |
| +    if (pipe_read_fd == -2) | |
| +    { | |
| +        int filedes[2]; | |
| +	int res; | |
| +	 | |
| +	res = pipe (filedes); | |
| +	if (res >= 0) | |
| +	{   Log::debug("opened pipe"); | |
| +   	    pipe_read_fd = filedes[0]; | |
| +   	    pipe_write_fd = filedes[1]; | |
| + | |
| +	    // Setup signal handler: | |
| +	    struct sigaction act; | |
| +	     | |
| +	    act.sa_sigaction = signal_handler; | |
| +	    sigemptyset(&act.sa_mask); | |
| +	    act.sa_flags = SA_SIGINFO; | |
| +	    sigaction(SIGRTMIN, &act, NULL); | |
| + | |
| +	    // When the RT queue overflows we get a SIGIO | |
| +	    act.sa_sigaction = overflow_signal_handler; | |
| +	    sigemptyset(&act.sa_mask); | |
| +	    sigaction(SIGIO, &act, NULL); | |
| + | |
| +	    (void) Scheduler::install_read_handler(pipe_read_fd, read_handler, NULL); | |
| +	} | |
| +    } | |
| +    return pipe_read_fd >= 0; | |
| +} | |
| + | |
| +DNotify::DirWatch * | |
| +DNotify::lookup_dirwatch (int fd) | |
| +{ | |
| +  DirWatch **p; | |
| +  DirWatch *w; | |
| + | |
| +  p = dir_hashchain (fd); | |
| + | |
| +  while (*p) | |
| +    { | |
| +      w = *p; | |
| + | |
| +      if (w->fd == fd) | |
| +	return w; | |
| + | |
| +      p = &w->hash_link; | |
| +    } | |
| +   | |
| +  return *p; | |
| +} | |
| + | |
| +// This colud be made faster by using another hash table. | |
| +// But it's not that bad, since it is only used by express/revoke | |
| +DNotify::DirWatch * | |
| +DNotify::lookup_dirwatch (dev_t dir_dev, ino_t dir_ino) | |
| +{ | |
| +  DirWatch *p; | |
| +  int i; | |
| + | |
| +  for (i=0;i<DIR_HASHSIZE;i++) | |
| +    { | |
| +      p = dir_hash[i]; | |
| +       | |
| +      while (p) | |
| +	{ | |
| +	  if (p->dir_dev == dir_dev && p->dir_ino == dir_ino) | |
| +	    return p; | |
| +	   | |
| +	  p = p->hash_link; | |
| +	} | |
| +    } | |
| +   | |
| +  return NULL; | |
| +} | |
| + | |
| +DNotify::FileWatch * | |
| +DNotify::lookup_filewatch (dev_t dev, ino_t ino) | |
| +{ | |
| +  FileWatch **p; | |
| +  FileWatch *w; | |
| + | |
| +  p = file_hashchain (dev, ino); | |
| + | |
| +  while (*p) | |
| +    { | |
| +      w = *p; | |
| + | |
| +      if (w->file_dev == dev && w->file_ino == ino) | |
| +	return w; | |
| + | |
| +      p = &w->hash_link; | |
| +    } | |
| +   | |
| +  return *p; | |
| +} | |
| + | |
| +// Make sure w is not already in the hash table before calling | |
| +// this function. | |
| +void | |
| +DNotify::hash_dirwatch(DirWatch *w) | |
| +{ | |
| +  DirWatch **p; | |
| +  p = dir_hashchain (w->fd); | |
| +  w->hash_link = *p; | |
| +  *p = w; | |
| +} | |
| + | |
| +// Make sure w is not already in the hash table before calling | |
| +// this function. | |
| +void | |
| +DNotify::hash_filewatch(FileWatch *w) | |
| +{ | |
| +  FileWatch **p; | |
| +  p = file_hashchain (w->file_dev, w->file_ino); | |
| +  w->hash_link = *p; | |
| +  *p = w; | |
| +} | |
| + | |
| +void | |
| +DNotify::unhash_dirwatch(DirWatch *w) | |
| +{ | |
| +  DirWatch **p; | |
| +   | |
| +  p = dir_hashchain (w->fd); | |
| +   | |
| +  while (*p) | |
| +    { | |
| +      if (*p == w) | |
| +	{ | |
| +	  *p = w->hash_link; | |
| +	  break; | |
| +	} | |
| +      p = &(*p)->hash_link; | |
| +    } | |
| +  w->hash_link = NULL; | |
| +} | |
| + | |
| +void | |
| +DNotify::unhash_filewatch(FileWatch *w) | |
| +{ | |
| +  FileWatch **p; | |
| +   | |
| +  p = file_hashchain (w->file_dev, w->file_ino); | |
| +   | |
| +  while (*p) | |
| +    { | |
| +      if (*p == w) | |
| +	{ | |
| +	  *p = w->hash_link; | |
| +	  break; | |
| +	} | |
| +      p = &(*p)->hash_link; | |
| +    } | |
| +  w->hash_link = NULL; | |
| +} | |
| + | |
| +DNotify::Status | |
| +DNotify::watch_dir(const char *notify_dir, dev_t file_dev, ino_t file_ino) | |
| +{ | |
| +  struct stat stat; | |
| +  dev_t dir_dev; | |
| +  ino_t dir_ino; | |
| +  DirWatch *dwatch; | |
| +  FileWatch *fw; | |
| +     | |
| +  if (lstat (notify_dir, &stat) == -1) | |
| +      return BAD; | |
| +   | |
| +  dwatch = lookup_dirwatch(stat.st_dev, stat.st_ino); | |
| +  if (!dwatch) | |
| +    { | |
| +      Log::debug ("New DirWatch for %s (%x %x)\n", | |
| +		  notify_dir, (int)stat.st_dev, (int)stat.st_ino); | |
| +      dwatch = new DirWatch; | |
| +      dwatch->watches = NULL; | |
| +      dwatch->hash_link = NULL; | |
| +      dwatch->dir_dev = stat.st_dev; | |
| +      dwatch->dir_ino = stat.st_ino; | |
| +       | |
| +      dwatch->fd = open(notify_dir, O_RDONLY); | |
| +      fcntl (dwatch->fd, F_SETSIG, SIGRTMIN); | |
| +      if (fcntl (dwatch->fd, F_NOTIFY, | |
| +                 (DN_MODIFY|DN_CREATE|DN_DELETE|DN_RENAME|DN_ATTRIB)  | |
| +		  | DN_MULTISHOT) == -1) { | |
| +	      return BAD; | |
| +      } | |
| +      hash_dirwatch (dwatch); | |
| +    } | |
| + | |
| +  fw = lookup_filewatch (file_dev, file_ino); | |
| +  if (fw && fw->dir_watch == dwatch) | |
| +	return OK; | |
| +   | |
| +  // No old FileWatch, need to add one: | |
| +  Log::debug("New FileWatch for %x %x\n", (int)file_dev, (int)file_ino); | |
| +  fw = new FileWatch; | |
| +  fw->next = dwatch->watches; | |
| +  dwatch->watches = fw; | |
| +  fw->file_dev = file_dev; | |
| +  fw->file_ino = file_ino; | |
| +  fw->dir_watch = dwatch; | |
| +  hash_filewatch(fw); | |
| +  return OK; | |
| +} | |
| + | |
| +char * | |
| +dirname_dup (const char *name) | |
| +{ | |
| +  char *copy = strdup(name); | |
| +  char *res = dirname(copy); | |
| +  res = strdup(res); | |
| +  free (copy); | |
| +  return res; | |
| +} | |
| + | |
| +DNotify::Status | |
| +DNotify::express(const char *name, struct stat *status) | |
| +{ | |
| +    struct stat stat; | |
| +    char *notify_dir; | |
| +    int res; | |
| +    Status s; | |
| +    dev_t dev; | |
| +    ino_t ino; | |
| + | |
| +    Log::debug("express() name: %s\n", name); | |
| + | |
| +    if (!is_active()) | |
| +	return BAD; | |
| + | |
| +    if (::lstat (name, &stat) == -1) | |
| +      return BAD; | |
| + | |
| +    dev = stat.st_dev; | |
| +    ino = stat.st_ino; | |
| +     | |
| +    if ((stat.st_mode & S_IFMT) != S_IFDIR) | |
| +	notify_dir = dirname_dup (name); | |
| +    else | |
| +	notify_dir = (char *)name; | |
| + | |
| +    s = watch_dir (notify_dir, dev, ino); | |
| +    if (notify_dir != name) | |
| +        free (notify_dir); | |
| +    if (s) | |
| +      return s; | |
| + | |
| +    // Check for a race condition; if someone removed or changed the | |
| +    // file at the same time that we are expressing interest in it, | |
| +    // revoke the interest so we don't get notifications about changes | |
| +    // to a recycled inode that we don't otherwise care about. | |
| +    // | |
| +    struct stat st; | |
| +    if (status == NULL) { | |
| +	status = &st; | |
| +    } | |
| +    if (::lstat(name, status) == -1) { | |
| +	Log::perror("stat on \"%s\" failed", name); | |
| +	revoke(name, stat.st_dev, stat.st_ino); | |
| +	return BAD; | |
| +    } | |
| +    if (status->st_dev != stat.st_dev | |
| +	|| status->st_ino != stat.st_ino) { | |
| +	Log::error("File \"%s\" changed between express and stat", | |
| +		   name); | |
| +	revoke(name, stat.st_dev, stat.st_ino); | |
| +	return BAD; | |
| +    }	 | |
| + | |
| +    Log::debug("told dnotify to monitor \"%s\" = dev %d/%d, ino %d", name, | |
| +	       major(status->st_dev), minor(status->st_dev), | |
| +	       status->st_ino); | |
| +    return OK; | |
| +} | |
| + | |
| +DNotify::Status | |
| +DNotify::revoke(const char *name, dev_t dev, ino_t ino) | |
| +{ | |
| +    FileWatch *fwatch; | |
| +    DirWatch *dwatch; | |
| +     | |
| +    Log::debug("revoke() name: %s, dev: %x, ino: %x\n", name, dev, ino); | |
| + | |
| +    if (!is_active()) | |
| +	return BAD; | |
| + | |
| +    // Lookup FileWatch by dev:ino, and its DirWatch. | |
| +    fwatch = lookup_filewatch (dev, ino); | |
| +    if (fwatch == NULL) | |
| +	return BAD; | |
| +     | |
| +    dwatch = fwatch->dir_watch; | |
| +     | |
| +    // delete FileWatch, if last FileWatch: close fd, delete DirWatch | |
| +    Log::debug ("Destroying FileWatch for (%x %x)\n", | |
| +		(int)fwatch->file_dev, (int)fwatch->file_ino); | |
| +    FileWatch **p; | |
| +    for (p=&dwatch->watches; *p; p=&(*p)->next) | |
| +    { | |
| +      if (*p == fwatch) | |
| +	{ | |
| +	  *p = (*p)->next; | |
| +	  break; | |
| +	} | |
| +    } | |
| +    unhash_filewatch(fwatch); | |
| +    delete fwatch; | |
| +    if (dwatch->watches == NULL) | |
| +      { | |
| +	Log::debug ("Destroying DirWatch for (%x %x)\n", | |
| +		    (int)dwatch->dir_dev, (int)dwatch->dir_ino); | |
| +	close(dwatch->fd); | |
| +	unhash_dirwatch(dwatch); | |
| +	delete dwatch; | |
| +      } | |
| +   | |
| +    return OK; | |
| +} | |
| + | |
| + | |
| +void | |
| +DNotify::all_watches_changed(void) | |
| +{ | |
| +  int i; | |
| +  FileWatch *fw; | |
| + | |
| +  for (i=0; i<FILE_HASHSIZE; i++) | |
| +  { | |
| +      fw = file_hash[i]; | |
| +      while (fw) | |
| +      { | |
| +	  (*ehandler)(fw->file_dev, fw->file_ino, CHANGE); | |
| + | |
| +	  fw = fw->hash_link; | |
| +      } | |
| +  } | |
| +} | |
| + | |
| + | |
| +void | |
| +DNotify::read_handler(int fd, void *) | |
| +{ | |
| +    static char readbuf[5000]; | |
| +    DirWatch *dw; | |
| +    FileWatch *fw; | |
| +    int snap_queue_tail; | |
| +    int last_fd; | |
| + | |
| +    int rc = read(fd, readbuf, sizeof readbuf); | |
| +    queue_changed = 0; | |
| +    if (rc < 0) | |
| +        Log::perror("pipe read"); | |
| +    else if (queue_overflowed) | |
| +    { | |
| +	  // There is a *slight* race condition here. Between reading | |
| +	  // the queue_overflow flag and resetting it. But it doesn't | |
| +	  // matter, since I'm gonna handle the overflow after reseting | |
| +	  // anyway. | |
| +	  queue_overflowed = false; | |
| + | |
| +	  // We're soon gonna check all watches anyway, so | |
| +	  // get rid of the current queue | |
| +	  queue_head = queue_tail; | |
| +	   | |
| +	  all_watches_changed (); | |
| +    } | |
| +    else | |
| +    { | |
| +	// Don't read events that happen later than | |
| +	// the initial read. (Otherwise skipping fd's | |
| +	// might miss some changes). | |
| +	snap_queue_tail = queue_tail; | |
| +	last_fd = -1; | |
| +	while (queue_head != snap_queue_tail) | |
| +	{ | |
| +	    fd = change_queue[queue_head]; | |
| +	    queue_head = (queue_head + 1) % QUEUESIZE; | |
| + | |
| +	    // Skip multiple changes to the same fd | |
| +	    if (fd != last_fd) | |
| +	    { | |
| +		dw = lookup_dirwatch (fd); | |
| +		if (dw) | |
| +		{ | |
| +		    int n_watches, i; | |
| +		    ChangeEventData *data; | |
| +		     | |
| +		    Log::debug("dnotify said dev %d/%d, ino %ld changed", | |
| +			       major(dw->dir_dev), minor(dw->dir_dev), dw->dir_ino); | |
| +		     | |
| +		    n_watches = 0; | |
| +		    for (fw=dw->watches; fw; fw=fw->next) | |
| +			n_watches++; | |
| +		     | |
| +		    data = new ChangeEventData[n_watches]; | |
| + | |
| +		    i = 0; | |
| +		    for (fw=dw->watches; fw; fw=fw->next) | |
| +		    { | |
| +			data[i].file_dev = fw->file_dev; | |
| +			data[i].file_ino = fw->file_ino; | |
| +			i++; | |
| +		    } | |
| + | |
| +		    for (i = 0; i < n_watches; i++) | |
| +		    { | |
| +			(*ehandler)(data[i].file_dev, data[i].file_ino, CHANGE); | |
| +		    } | |
| +		     | |
| +		    delete[] data; | |
| +		} | |
| +	    } | |
| +	    last_fd = fd; | |
| +	} | |
| +    } | |
| +} | |
| + | |
| diff -Naur fam-2.7.0.orig/src/DNotify.h fam-2.7.0/src/DNotify.h | |
| --- fam-2.7.0.orig/src/DNotify.h	1970-01-01 00:00:00.000000000 +0000 | |
| +++ fam-2.7.0/src/DNotify.h	2004-03-15 21:31:34.546060811 +0000 | |
| @@ -0,0 +1,98 @@ | |
| +//  Copyright (C) 2001 Red Hat, Inc.  All Rights Reserved. | |
| +//  Copyright (C) 1999 Silicon Graphics, Inc.  All Rights Reserved. | |
| +// | |
| +//  This program is free software; you can redistribute it and/or modify it | |
| +//  under the terms of version 2 of the GNU General Public License as | |
| +//  published by the Free Software Foundation. | |
| +// | |
| +//  This program is distributed in the hope that it would be useful, but | |
| +//  WITHOUT ANY WARRANTY; without even the implied warranty of | |
| +//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  Further, any | |
| +//  license provided herein, whether implied or otherwise, is limited to | |
| +//  this program in accordance with the express provisions of the GNU | |
| +//  General Public License.  Patent licenses, if any, provided herein do not | |
| +//  apply to combinations of this program with other product or programs, or | |
| +//  any other product whatsoever.  This program is distributed without any | |
| +//  warranty that the program is delivered free of the rightful claim of any | |
| +//  third person by way of infringement or the like.  See the GNU General | |
| +//  Public License for more details. | |
| +// | |
| +//  You should have received a copy of the GNU General Public License along | |
| +//  with this program; if not, write the Free Software Foundation, Inc., 59 | |
| +//  Temple Place - Suite 330, Boston MA 02111-1307, USA. | |
| + | |
| +#ifndef DNotify_included | |
| +#define DNotify_included | |
| + | |
| +#include "config.h" | |
| +#include "Monitor.h" | |
| +#include <signal.h> | |
| + | |
| +//  DNotify is an object encapsulating the dnotify linux fcntl. | |
| +//  It "emulates" the IMon interface. | |
| +//  There can only be one instantiation of the DNotify object. | |
| +// | |
| +//  The user of this object uses express() and revoke() to | |
| +//  express/revoke interest in a file.  There is also | |
| +//  a callback, the EventHandler.  When an dnotify event comes in, | |
| +//  the EventHandler is called. | |
| +// | |
| +//  The user of the DNotify object is the Interest class. | |
| + | |
| +class DNotify : public Monitor { | |
| +public: | |
| +    DNotify(EventHandler h); | |
| +    ~DNotify(); | |
| + | |
| +    static bool is_active(); | |
| + | |
| +    virtual Status express(const char *name, struct stat *stat_return); | |
| +    virtual Status revoke(const char *name, dev_t dev, ino_t ino); | |
| + | |
| +private: | |
| +    struct FileWatch; | |
| +    struct DirWatch; | |
| +    struct ChangeEventData; | |
| +   | |
| +    //  Class Variables | |
| +    enum { QUEUESIZE = 1024 }; | |
| +    static int pipe_write_fd; | |
| +    static int pipe_read_fd; | |
| +    static int change_queue[QUEUESIZE]; | |
| +    static volatile sig_atomic_t DNotify::queue_overflowed; | |
| +    static volatile sig_atomic_t DNotify::queue_changed; | |
| +    static volatile int queue_head; // Only modified by read handler | |
| +    static volatile int queue_tail; // Only modified by signal handler | |
| +    static EventHandler ehandler; | |
| +    static void overflow_signal_handler(int sig, siginfo_t *si, void *data); | |
| +    static void signal_handler(int sig, siginfo_t *si, void *data); | |
| +    static void read_handler(int fd, void *closure); | |
| +  | |
| +    enum { DIR_HASHSIZE = 367 }; | |
| +    static DirWatch *dir_hash[DIR_HASHSIZE]; | |
| +    enum { FILE_HASHSIZE = 823 }; | |
| +    static FileWatch *file_hash[FILE_HASHSIZE]; | |
| + | |
| +    static DirWatch **dir_hashchain(int fd) | |
| +			  { return &dir_hash[(unsigned) (fd) % DIR_HASHSIZE]; } | |
| +    static FileWatch **file_hashchain(dev_t d, ino_t i) | |
| +			  { return &file_hash[(unsigned) (d+i) % FILE_HASHSIZE]; } | |
| + | |
| +    static DirWatch *lookup_dirwatch (int fd); | |
| +    static DirWatch *lookup_dirwatch (dev_t dir_dev, ino_t dir_ino); | |
| +    static FileWatch *lookup_filewatch (dev_t file_dev, ino_t file_ino); | |
| +    static void hash_dirwatch(DirWatch *w); | |
| +    static void hash_filewatch(FileWatch *w); | |
| +    static void unhash_dirwatch(DirWatch *w); | |
| +    static void unhash_filewatch(FileWatch *w); | |
| +    static Status watch_dir(const char *notify_dir, dev_t file_dev, ino_t file_ino); | |
| + | |
| +    static void all_watches_changed(void); | |
| +     | |
| +    DNotify(const DNotify&);			// Do not copy | |
| +    DNotify & operator = (const DNotify&);	//  or assign. | |
| +}; | |
| + | |
| +#endif /* !IMon_included */ | |
| + | |
| + | |
| diff -Naur fam-2.7.0.orig/src/IMon.h fam-2.7.0/src/IMon.h | |
| --- fam-2.7.0.orig/src/IMon.h	2003-01-18 14:18:12.000000000 +0000 | |
| +++ fam-2.7.0/src/IMon.h	2004-03-15 21:31:34.558058615 +0000 | |
| @@ -24,10 +24,7 @@ | |
|  #define IMon_included | |
|   | |
|  #include "config.h" | |
| -#include <sys/stat.h> | |
| -#include <sys/types.h> | |
| - | |
| -#include "Boolean.h" | |
| +#include "Monitor.h" | |
|   | |
|  struct stat; | |
|   | |
| @@ -41,25 +38,18 @@ | |
|  // | |
|  //  The user of the IMon object is the Interest class. | |
|   | |
| -class IMon { | |
| +class IMon : public Monitor { | |
|   | |
|  public: | |
| - | |
| -    enum Status { OK = 0, BAD = -1 }; | |
| -    enum Event { EXEC, EXIT, CHANGE }; | |
| - | |
| -    typedef void (*EventHandler)(dev_t, ino_t, int event); | |
| - | |
|      IMon(EventHandler h); | |
|      ~IMon(); | |
|   | |
|      static bool is_active(); | |
|   | |
| -    Status express(const char *name, struct stat *stat_return); | |
| -    Status revoke(const char *name, dev_t dev, ino_t ino); | |
| +    virtual Status express(const char *name, struct stat *stat_return); | |
| +    virtual Status revoke(const char *name, dev_t dev, ino_t ino); | |
|   | |
|  private: | |
| - | |
|      //  Class Variables | |
|   | |
|      static int imonfd; | |
| diff -Naur fam-2.7.0.orig/src/Interest.c++ fam-2.7.0/src/Interest.c++ | |
| --- fam-2.7.0.orig/src/Interest.c++	2003-01-18 14:18:12.000000000 +0000 | |
| +++ fam-2.7.0/src/Interest.c++	2004-03-15 21:31:34.550060079 +0000 | |
| @@ -42,12 +42,21 @@ | |
|  #include "Event.h" | |
|  #include "FileSystem.h" | |
|  #include "IMon.h" | |
| +#include "DNotify.h" | |
|  #include "Log.h" | |
|  #include "Pollster.h" | |
|  #include "timeval.h" | |
|   | |
|  Interest *Interest::hashtable[]; | |
| -IMon      Interest::imon(imon_handler); | |
| + | |
| +#ifdef USE_DNOTIFY  | |
| +static DNotify dnotify(Interest::monitor_handler); | |
| +Monitor * Interest::monitor = &dnotify; | |
| +#else | |
| +static IMon imon(Interest::monitor_handler); | |
| +Monitor * Interest::monitor = &imon; | |
| +#endif | |
| + | |
|  bool      Interest::xtab_verification = true; | |
|   | |
|  Interest::Interest(const char *name, FileSystem *fs, in_addr host, ExportVerification ev) | |
| @@ -60,10 +69,10 @@ | |
|        mypath_exported_to_host(ev == NO_VERIFY_EXPORTED) | |
|  { | |
|      memset(&old_stat, 0, sizeof(old_stat));  | |
| -    IMon::Status s = IMon::BAD; | |
|   | |
| -    s = imon.express(name, &old_stat); | |
| -    if (s != IMon::OK) | |
| +    Monitor::Status s = Monitor::BAD; | |
| +    s = monitor->express(name, &old_stat); | |
| +    if (s != Monitor::OK) | |
|      {   int rc = lstat(name, &old_stat); | |
|  	if (rc < 0) | |
|  	{   Log::info("can't lstat %s", name); | |
| @@ -100,7 +109,7 @@ | |
|      } | |
|  #endif | |
|   | |
| -    if (exported_to_host()) fs->ll_monitor(this, s == IMon::OK); | |
| +    if (exported_to_host()) fs->ll_monitor(this, s == Monitor::OK); | |
|  } | |
|   | |
|  Interest::~Interest() | |
| @@ -128,7 +137,7 @@ | |
|  		pp = &p->hashlink;	// move to next element | |
|  	    } | |
|  	if (!found_same) | |
| -	    (void) imon.revoke(name(), dev, ino); | |
| +	  (void) monitor->revoke(name(), dev, ino); | |
|      } | |
|  } | |
|   | |
| @@ -147,7 +156,7 @@ | |
|   | |
|          // Express interest. | |
|          IMon::Status s = IMon::BAD; | |
| -	s = imon.express(name(), NULL); | |
| +	s = monitor->express(name(), NULL); | |
|          if (s != IMon::OK) { | |
|              return true; | |
|          } | |
| @@ -248,23 +257,23 @@ | |
|  } | |
|   | |
|  void | |
| -Interest::imon_handler(dev_t device, ino_t inumber, int event) | |
| +Interest::monitor_handler(dev_t device, ino_t inumber, int event) | |
|  { | |
|      assert(device || inumber); | |
|   | |
|      for (Interest *p = *hashchain(device, inumber), *next = p; p; p = next) | |
|      {	next = p->hashlink; | |
|  	if (p->ino == inumber && p->dev == device) | |
| -	{   if (event == IMon::EXEC) | |
| +	  {   if (event == Monitor::EXEC) | |
|  	    {   p->cur_exec_state = EXECUTING; | |
|  		(void) p->report_exec_state(); | |
|  	    } | |
| -	    else if (event == IMon::EXIT) | |
| +	    else if (event == Monitor::EXIT) | |
|  	    {   p->cur_exec_state = NOT_EXECUTING; | |
|  		(void) p->report_exec_state(); | |
|  	    } | |
|  	    else | |
| -	    {   assert(event == IMon::CHANGE); | |
| +	    {   assert(event == Monitor::CHANGE); | |
|  		p->scan(); | |
|  	    } | |
|  	} | |
| diff -Naur fam-2.7.0.orig/src/Interest.h fam-2.7.0/src/Interest.h | |
| --- fam-2.7.0.orig/src/Interest.h	2003-01-18 14:18:12.000000000 +0000 | |
| +++ fam-2.7.0/src/Interest.h	2004-03-15 21:31:34.560058249 +0000 | |
| @@ -32,7 +32,7 @@ | |
|   | |
|  class Event; | |
|  class FileSystem; | |
| -class IMon; | |
| +class Monitor; | |
|  struct stat; | |
|   | |
|  //  Interest -- abstract base class for filesystem entities of interest. | |
| @@ -74,7 +74,7 @@ | |
|   | |
|      //  Public Class Method | |
|   | |
| -    static void imon_handler(dev_t, ino_t, int event); | |
| +    static void monitor_handler(dev_t, ino_t, int event); | |
|   | |
|      static void enable_xtab_verification(bool enable); | |
|   | |
| @@ -121,7 +121,7 @@ | |
|   | |
|      //  Class Variables | |
|   | |
| -    static IMon imon; | |
| +    static Monitor *monitor; | |
|      static Interest *hashtable[HASHSIZE]; | |
|      static bool xtab_verification; | |
|   | |
| diff -Naur fam-2.7.0.orig/src/Makefile.am fam-2.7.0/src/Makefile.am | |
| --- fam-2.7.0.orig/src/Makefile.am	2003-01-19 12:00:17.000000000 +0000 | |
| +++ fam-2.7.0/src/Makefile.am	2004-03-15 21:31:34.562057882 +0000 | |
| @@ -71,7 +71,11 @@ | |
|    main.c++ \ | |
|    timeval.c++ \ | |
|    timeval.h \ | |
| -  @[email protected]++ | |
| +  Monitor.h \ | |
| +  DNotify.h \ | |
| +  DNotify.c++ \ | |
| +  @[email protected]++  | |
|   | |
| -EXTRA_famd_SOURCES = IMonIrix.c++ IMonLinux.c++ IMonNone.c++ | |
| +EXTRA_famd_SOURCES = IMonIrix.c++ IMonLinux.c++ IMonNone.c++ DNotify.c++ \ | |
| +  DNotify.h Monitor.h | |
|   | |
| diff -Naur fam-2.7.0.orig/src/Monitor.h fam-2.7.0/src/Monitor.h | |
| --- fam-2.7.0.orig/src/Monitor.h	1970-01-01 00:00:00.000000000 +0000 | |
| +++ fam-2.7.0/src/Monitor.h	2004-03-15 21:31:34.564057516 +0000 | |
| @@ -0,0 +1,57 @@ | |
| +//  Copyright (C) 2001 Red Hat, Inc.  All Rights Reserved. | |
| +//  Copyright (C) 1999 Silicon Graphics, Inc.  All Rights Reserved. | |
| +//   | |
| +//  This program is free software; you can redistribute it and/or modify it | |
| +//  under the terms of version 2 of the GNU General Public License as | |
| +//  published by the Free Software Foundation. | |
| +// | |
| +//  This program is distributed in the hope that it would be useful, but | |
| +//  WITHOUT ANY WARRANTY; without even the implied warranty of | |
| +//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  Further, any | |
| +//  license provided herein, whether implied or otherwise, is limited to | |
| +//  this program in accordance with the express provisions of the GNU | |
| +//  General Public License.  Patent licenses, if any, provided herein do not | |
| +//  apply to combinations of this program with other product or programs, or | |
| +//  any other product whatsoever.  This program is distributed without any | |
| +//  warranty that the program is delivered free of the rightful claim of any | |
| +//  third person by way of infringement or the like.  See the GNU General | |
| +//  Public License for more details. | |
| +// | |
| +//  You should have received a copy of the GNU General Public License along | |
| +//  with this program; if not, write the Free Software Foundation, Inc., 59 | |
| +//  Temple Place - Suite 330, Boston MA 02111-1307, USA. | |
| + | |
| +#ifndef Monitor_included | |
| +#define Monitor_included | |
| + | |
| +#include "config.h" | |
| +#include <sys/stat.h> | |
| +#include <sys/types.h> | |
| + | |
| +struct stat; | |
| + | |
| +//  Monitor is an abstract baseclass for differend file monitoring | |
| +//  systems. The original system used was IMon, and the Montor API | |
| +//  is heavily influenced by that. | |
| +//  There can only be one instantiation of the Monitor object. | |
| +// | |
| +//  The user of this object uses express() and revoke() to | |
| +//  express/revoke interest in a file to imon.  There is also | |
| +//  a callback, the EventHandler.  When an event comes in, | |
| +//  the EventHandler is called. | |
| +// | |
| +//  The main implementers of the Monitor class is IMon and DNotify | |
| + | |
| +class Monitor { | |
| +public: | |
| + | |
| +    enum Status { OK = 0, BAD = -1 }; | |
| +    enum Event { EXEC, EXIT, CHANGE }; | |
| + | |
| +    typedef void (*EventHandler)(dev_t, ino_t, int event); | |
| + | |
| +    virtual Status express(const char *name, struct stat *stat_return) = 0; | |
| +    virtual Status revoke(const char *name, dev_t dev, ino_t ino) = 0; | |
| +}; | |
| + | |
| +#endif /* !Monitor_included */
 | |
| 
 |