pike.git / lib / modules / System.pmod / Inotify.pmod

version» Context lines:

pike.git/lib/modules/System.pmod/Inotify.pmod:60:      class FileWatch {    inherit Watch;       void handle_event(int wd, int mask, int cookie,    int|string name) {    callback(mask, cookie, this::name, @extra);    }   }    - // The reason for this extra indirection is to remove - // cyclic references from the Instance class to make - // sure inotify instances are destructed when they go out of - // external references. This would not be happening otherwise - // since File objects still have references from the backend. - function parse_fun(mapping watches) { -  void parse(mixed id, string data) { -  while (sizeof(data)) { -  array a = parse_event(data); -  object watch; -  -  if (watch = watches[a[0]]) { -  if (a[1] == IN_IGNORED) { -  m_delete(watches, a[0]); -  } -  watch->handle_event(a[0], a[1], a[2], a[3]); -  } -  -  data = data[a[4]..]; -  } -  }; -  return parse; - } -  +    //! More convenient interface to inotify(7). Automatically reads events   //! from the inotify file descriptor and parses them.   //!   //! @note   //! Objects of this class will be destructed when they go out of external   //! references. As such they behave differently from other classes which use   //! callbacks, e.g. @[Stdio.File].   //! @note   //! The number of inotify instances is limited by ulimits.   class Instance { -  protected object instance; -  protected Stdio.File file; +  inherit _Instance;    protected mapping(int:object) watches = ([]);    -  +  void event_callback(int wd, int event, int cookie, string path) +  { +  Watch watch = watches[wd]; +  if (watch) { +  if (event == IN_IGNORED) { +  m_delete(watches, wd); +  werror("Watch %O (wd: %d) deleted.\n", watch, wd); +  } +  watch->handle_event(wd, event, cookie, path); +  } +  } +     void create() { -  instance = _Instance(); -  file = Stdio.File(); -  file->assign(instance->fd()); -  file->set_nonblocking(); -  file->set_read_callback(parse_fun(watches)); +  set_event_callback(event_callback); +  set_nonblocking();    }       //! Add a watch for a certain file and a set of events specified by    //! @expr{mask@}. The function @expr{callback@} will be called when    //! such an event occurs. The arguments to the callback will be the    //! events mask, the cookie, the @expr{filename@} and @expr{extra@}.    //! @returns    //! Returns a watch descriptor which can be used to remove the    //! watch.    //! @note
pike.git/lib/modules/System.pmod/Inotify.pmod:128:    //! The mask of an event may be a subset of the mask given when    //! adding the watch.    //! @note    //! In case the watch is added for a regular file, the filename    //! will be passed to the callback. In case the watch is added    //! for a directory, the name of the file to which the event    //! happened inside the directory will be concatenated.    //!    int add_watch(string filename, int mask,    function(int, int, string, mixed ...:void) callback, -  mixed ... extra) { -  int wd = instance->add_watch(filename, mask); +  mixed ... extra) +  { +  int wd = ::add_watch(filename, mask);      //! @ignore   # if constant(@module@.IN_MASK_ADD)    /*    * just in case we want to use the mask in the watches later,    * they better be correct    */    if ((mask & IN_MASK_ADD) && has_index(watches, wd)) {    mask |= watches[wd]->mask;    }   # endif   //! @endignore    if (Stdio.is_dir(filename)) {    watches[wd] = DirWatch(filename, mask, callback, extra);    } else {    watches[wd] = FileWatch(filename, mask, callback, extra);    }       return wd;    } -  -  //! Remove the watch associated with the watch descriptor @expr{wd@}. -  void rm_watch(int wd) { -  if (!has_index(watches, wd)) { -  error("Watch %d does not exist.\n", wd); +    } -  -  m_delete(watches, wd); -  instance->rm_watch(wd); -  } -  -  void destroy() { -  file->set_read_callback(0); -  /* -  * closing the last copy of an inotify fd currently takes a long time (~100ms). -  * make sure to close the file first, to allow releasing the interpreter lock -  * during the last close call in the instance EXIT handler. -  */ -  file->close(); -  } - } +