34 #ifdef FOAM_USE_INOTIFY 36 #include <sys/inotify.h> 37 #include <sys/ioctl.h> 39 #define EVENT_SIZE ( sizeof (struct inotify_event) ) 40 #define EVENT_LEN (EVENT_SIZE + 16) 41 #define EVENT_BUF_LEN ( 1024 * EVENT_LEN ) 71 unsigned int operator()(
const unsigned int x,
const unsigned int y)
76 unsigned int mask = 3u;
77 unsigned int shift = 0;
78 unsigned int result = 0;
83 unsigned int xState = (x & mask) >> shift;
84 unsigned int yState = (y & mask) >> shift;
88 unsigned int state =
min(xState, yState);
89 result |= (state << shift);
136 useInotify_(useInotify),
141 #ifdef FOAM_USE_INOTIFY 142 inotifyFd_ = inotify_init();
148 static bool hasWarned =
false;
153 <<
"Failed allocating an inotify descriptor : " 155 <<
" Please increase the number of allowable " 156 <<
"inotify instances" <<
endl 157 <<
" (/proc/sys/fs/inotify/max_user_instances" 158 <<
" on Linux)" <<
endl 159 <<
" , switch off runTimeModifiable." <<
endl 160 <<
" or compile this file without " 161 <<
"FOAM_USE_INOTIFY" 162 <<
" to use time stamps instead of inotify." <<
endl 163 <<
" Continuing without additional file" 170 <<
"You selected inotify but this file was compiled" 171 <<
" without FOAM_USE_INOTIFY" 172 <<
" Please select another fileModification test method" 185 #ifdef FOAM_USE_INOTIFY 186 if (useInotify_ && inotifyFd_ >= 0)
190 if (dirWatches_[i] >= 0)
192 if (inotify_rm_watch(inotifyFd_,
int(dirWatches_[i])))
195 <<
"Failed deleting directory watch " 196 << dirWatches_[i] <<
endl;
213 #ifdef FOAM_USE_INOTIFY 220 label dirWatchID = -1;
223 dirWatchID = inotify_add_watch
233 <<
"Failed adding watch " << watchFd
234 <<
" to directory " << fName <<
" due to " 235 <<
string(strerror(errno))
240 if (watchFd < dirWatches_.
size() && dirWatches_[watchFd] != -1)
244 <<
"Problem adding watch " << watchFd
245 <<
" to file " << fName
249 dirWatches_(watchFd) = dirWatchID;
250 dirFiles_(watchFd) = fName.
name();
255 if (watchFd < lastMod_.
size() && lastMod_[watchFd] != 0)
259 <<
"Problem adding watch " << watchFd
260 <<
" to file " << fName
279 dirWatches_[watchFd] = -1;
283 lastMod_[watchFd] = 0;
294 void Foam::fileMonitor::checkFiles()
const 298 #ifdef FOAM_USE_INOTIFY 300 char buffer[EVENT_BUF_LEN];
304 struct timeval zeroTimeout = {0, 0};
310 FD_SET(watcher_->inotifyFd_, &fdSet);
314 watcher_->inotifyFd_+1,
324 <<
"Problem in issuing select." 327 else if (FD_ISSET(watcher_->inotifyFd_, &fdSet))
332 watcher_->inotifyFd_,
340 <<
"read of " << watcher_->inotifyFd_
341 <<
" failed with " <<
label(nBytes)
349 const struct inotify_event* inotifyEvent =
350 reinterpret_cast<const struct inotify_event*
> 363 (inotifyEvent->mask & IN_CLOSE_WRITE)
368 forAll(watcher_->dirWatches_, i)
370 label id = watcher_->dirWatches_[i];
373 id == inotifyEvent->wd
374 && inotifyEvent->name == watcher_->dirFiles_[i]
378 localState_[i] = MODIFIED;
383 i += EVENT_SIZE + inotifyEvent->len;
396 forAll(watcher_->lastMod_, watchFd)
398 double oldTime = watcher_->lastMod_[watchFd];
402 const fileName& fName = watchFile_[watchFd];
407 localState_[watchFd] = DELETED;
413 localState_[watchFd] = MODIFIED;
417 localState_[watchFd] = UNMODIFIED;
431 useInotify_(useInotify),
454 label sz = freeWatchFds_.size();
458 watchFd = freeWatchFds_[sz-1];
459 freeWatchFds_.setSize(sz-1);
463 watchFd = state_.
size();
466 watcher_->addWatch(watchFd, fName);
470 Pout<<
"fileMonitor : added watch " << watchFd <<
" on file " 477 <<
"could not add watch for file " << fName <<
endl;
483 watchFile_(watchFd) = fName;
493 Pout<<
"fileMonitor : removing watch " << watchFd <<
" on file " 494 << watchFile_[watchFd] <<
endl;
497 freeWatchFds_.append(watchFd);
498 return watcher_->removeWatch(watchFd);
504 return watchFile_[watchFd];
511 return state_[watchFd];
517 const bool masterOnly,
535 stats[watchFd] =
static_cast<unsigned int> 547 if (stats.storage().size() == 1)
559 if (stats.storage().size() == 1)
579 unsigned int stat = stats[watchFd];
585 if (state_[watchFd] != localState_[watchFd])
589 Pout<<
"fileMonitor : Delaying reading " 590 << watchFile_[watchFd]
591 <<
" due to inconsistent " 592 "file time-stamps between processors" 597 <<
"Delaying reading " << watchFile_[watchFd]
598 <<
" due to inconsistent " 599 "file time-stamps between processors" <<
endl;
606 state_ = localState_;
fileMonitor(const bool useInotify)
Construct null.
DynamicList< double > lastMod_
From watch descriptor to modified time.
static const NamedEnum< fileState, 3 > fileStateNames_
static void listCombineScatter(const List< commsStruct > &comms, List< T > &Value, const int tag, const label comm)
Scatter data. Reverse of combineGather.
#define forAll(list, i)
Loop across all elements in list.
FvWallInfoData< WallInfo, label > label
A label is an int32_t or int64_t as specified by the pre-processor macro WM_LABEL_SIZE.
A class for handling file names.
Inter-processor communication reduction functions.
errorManipArg< error, int > exit(error &err, const int errNo=1)
Internal tracking via stat(3p) or inotify(7)
#define FatalErrorInFunction
Report an error message using Foam::FatalError.
fileState getState(const label watchFd) const
Check state using handle.
fileState
Enumeration defining the file state.
void setUnmodified(const label watchFd)
Reset state (e.g. after having read it) using handle.
void size(const label)
Override size to be inconsistent with allocated storage.
Ostream & endl(Ostream &os)
Add newline and flush stream.
static bool master(const label communicator=0)
Am I the master process.
DynamicList< label > dirWatches_
Current watchIDs and corresponding directory id.
Initialise the NamedEnum HashTable from the static list of names.
static void listCombineGather(const List< commsStruct > &comms, List< T > &Value, const CombineOp &cop, const int tag, const label comm)
Reduction operator for PackedList of fileState.
Useful combination of include files which define Sin, Sout and Serr and the use of IO streams general...
fileMonitorWatcher(const bool useInotify, const label sz=20)
Initialise inotify.
void operator()(unsigned int &x, const unsigned int y) const
Functions used by OpenFOAM that are specific to POSIX compliant operating systems and need to be repl...
bool read(const char *, int32_t &)
bool removeWatch(const label watchFd)
Remove file to watch. Return true if successful.
A 1D vector of objects of type <T> that resizes itself as necessary to accept the new objects...
bool isDir(const fileName &, const bool followLink=true)
Does the name exist as a directory in the file system?
void updateStates(const bool masterOnly, const bool syncPar) const
Check state of all files. Updates state_.
const fileName & getFile(const label watchFd) const
Get name of file being watched.
unsigned int operator()(const unsigned int x, const unsigned int y) const
void setCapacity(const label)
Alter the size of the underlying storage.
word name() const
Return file name (part beyond last /)
layerAndWeight min(const layerAndWeight &a, const layerAndWeight &b)
errorManip< error > abort(error &err)
static void scatter(const List< commsStruct > &comms, T &Value, const int tag, const label comm)
Scatter data. Distribute without modification. Reverse of gather.
int inotifyFd_
File descriptor for the inotify instance.
defineTypeNameAndDebug(combustionModel, 0)
~fileMonitor()
Destructor.
void reduce(const List< UPstream::commsStruct > &comms, T &Value, const BinaryOp &bop, const int tag, const label comm)
double highResLastModified(const fileName &, const bool checkVariants=true, const bool followLink=true)
Return time of last file modification.
#define WarningInFunction
Report a warning using Foam::Warning.
label addWatch(const fileName &)
Add file to watch. Return watch descriptor.
prefixOSstream Pout(cout, "Pout")
DynamicList< fileName > dirFiles_
static float fileModificationSkew
fileName path() const
Return directory path name (part before last /)
bool removeWatch(const label watchFd)
~fileMonitorWatcher()
Remove all watches.
Combine operator for PackedList of fileState.
bool addWatch(const label watchFd, const fileName &fName)
A class for handling character strings derived from std::string.