33#include "kcoreaddons_debug.h"
34#include "kdirwatch_p.h"
35#include "kfilesystemtype.h"
36#include "knetworkmounts.h"
38#include <io/config-kdirwatch.h>
40#include <QCoreApplication>
43#include <QLoggingCategory>
44#include <QSocketNotifier>
46#include <QThreadStorage>
52#include <qplatformdefs.h>
59#include <sys/inotify.h>
63#define IN_DONT_FOLLOW 0x02000000
67#define IN_ONLYDIR 0x01000000
73#include <sys/utsname.h>
77Q_DECLARE_LOGGING_CATEGORY(KDIRWATCH)
79Q_LOGGING_CATEGORY(KDIRWATCH,
"kf.coreaddons.kdirwatch", QtWarningMsg)
82static bool s_verboseDebug =
false;
85static KDirWatchPrivate *createPrivate()
87 if (!dwp_self.hasLocalData()) {
88 dwp_self.setLocalData(
new KDirWatchPrivate);
92static void destroyPrivate()
94 dwp_self.localData()->deleteLater();
95 dwp_self.setLocalData(
nullptr);
99static KDirWatch::Method methodFromString(
const QByteArray &method)
101 if (method ==
"Stat") {
102 return KDirWatch::Stat;
103 }
else if (method ==
"QFSWatch") {
104 return KDirWatch::QFSWatch;
106#if HAVE_SYS_INOTIFY_H
108 return KDirWatch::INotify;
110 return KDirWatch::QFSWatch;
115static const char *methodToString(KDirWatch::Method method)
118 case KDirWatch::INotify:
120 case KDirWatch::Stat:
122 case KDirWatch::QFSWatch:
129static const char s_envNfsPoll[] =
"KDIRWATCH_NFSPOLLINTERVAL";
130static const char s_envPoll[] =
"KDIRWATCH_POLLINTERVAL";
131static const char s_envMethod[] =
"KDIRWATCH_METHOD";
132static const char s_envNfsMethod[] =
"KDIRWATCH_NFSMETHOD";
162KDirWatchPrivate::KDirWatchPrivate()
163 : m_statRescanTimer()
171#if HAVE_SYS_INOTIFY_H
179 s_verboseDebug =
true;
181 m_statRescanTimer.setObjectName(QStringLiteral(
"KDirWatchPrivate::timer"));
184 m_nfsPollInterval = qEnvironmentVariableIsSet(s_envNfsPoll) ? qEnvironmentVariableIntValue(s_envNfsPoll) : 5000;
185 m_PollInterval = qEnvironmentVariableIsSet(s_envPoll) ? qEnvironmentVariableIntValue(s_envPoll) : 500;
187 m_preferredMethod = methodFromString(qEnvironmentVariableIsSet(s_envMethod) ? qgetenv(s_envMethod) :
"inotify");
189 m_nfsPreferredMethod = methodFromString(qEnvironmentVariableIsSet(s_envNfsMethod) ? qgetenv(s_envNfsMethod) :
"Stat");
193 availableMethods <<
"Stat";
196 rescan_timer.setObjectName(QStringLiteral(
"KDirWatchPrivate::rescan_timer"));
197 rescan_timer.setSingleShot(
true);
200#if HAVE_SYS_INOTIFY_H
201#if HAVE_INOTIFY_DIRECT_READV
202 m_inotify_fd = inotify_init1(IN_DIRECT);
204 m_inotify_fd = inotify_init();
206 supports_inotify = m_inotify_fd > 0;
208 if (!supports_inotify) {
209 qCDebug(KDIRWATCH) <<
"Can't use Inotify, kernel doesn't support it:" << strerror(errno);
211 availableMethods <<
"INotify";
212 (void)fcntl(m_inotify_fd, F_SETFD, FD_CLOEXEC);
218#if HAVE_QFILESYSTEMWATCHER
219 availableMethods <<
"QFileSystemWatcher";
223 qCDebug(KDIRWATCH) <<
"Available methods: " << availableMethods <<
"preferred=" << methodToString(m_preferredMethod);
227KDirWatchPrivate::~KDirWatchPrivate()
229 m_statRescanTimer.stop();
233 for (
auto it = m_mapEntries.begin(); it != m_mapEntries.end(); it++) {
234 auto &entry = it.value();
235 for (
auto &client : entry.m_clients) {
236 client.instance->d =
nullptr;
240 for (
auto &referenceObject : m_referencesObjects) {
241 referenceObject->d =
nullptr;
244#if HAVE_SYS_INOTIFY_H
245 if (supports_inotify) {
246#if HAVE_INOTIFY_DIRECT_READV
251 ret = libinotify_direct_close(m_inotify_fd);
252 }
while (ret == -1 && errno == EINTR);
254 QT_CLOSE(m_inotify_fd);
258#if HAVE_QFILESYSTEMWATCHER
263void KDirWatchPrivate::inotifyEventReceived()
265#if HAVE_SYS_INOTIFY_H
266 if (!supports_inotify) {
270 assert(m_inotify_fd > -1);
272 auto processEvent = [
this](
const struct inotify_event *
const event)
276 int len =
event->len;
277 while (len > 1 && !
event->name[len - 1]) {
283 if (!
path.
isEmpty() && isNoisyFile(cpath.data())) {
288 const bool isDir = (
event->mask & (IN_ISDIR));
290 Entry *e = m_inotify_wd_to_entry.value(
event->wd);
294 const bool wasDirty = e->dirty;
299 qCDebug(KDIRWATCH).nospace() <<
"got event " << inotifyEventName(event) <<
" for entry " << e->path
300 << (
event->mask & IN_ISDIR ?
" [directory] " :
" [file] ") << path;
302 if (
event->mask & IN_DELETE_SELF) {
303 e->m_status = NonExistent;
304 m_inotify_wd_to_entry.remove(e->wd);
306 e->m_ctime = invalid_ctime;
307 emitEvent(e, Deleted);
309 Entry *parentEntry = entry(e->parentDirectory());
311 parentEntry->dirty =
true;
314 addEntry(
nullptr, e->parentDirectory(), e,
true );
316 if (
event->mask & IN_IGNORED) {
320 if (
event->mask & (IN_CREATE | IN_MOVED_TO)) {
321 Entry *sub_entry = e->findSubEntry(tpath);
323 qCDebug(KDIRWATCH) <<
"-->got CREATE signal for" << (tpath) <<
"sub_entry=" << sub_entry;
327 sub_entry->dirty =
true;
328 rescan_timer.start(0);
329 }
else if (e->isDir && !e->m_clients.empty()) {
334 for (
const Client *client : clients) {
339 emitEvent(e, Created, tpath);
340 qCDebug(KDIRWATCH).nospace() << clients.
count() <<
" instance(s) monitoring the new " << (isDir ?
"dir " :
"file ") << tpath;
342 e->m_pendingFileChanges.
append(e->path);
343 if (!rescan_timer.isActive()) {
344 rescan_timer.start(m_PollInterval);
348 if (
event->mask & (IN_DELETE | IN_MOVED_FROM)) {
349 if ((e->isDir) && (!e->m_clients.empty())) {
355 int counter = std::count_if(e->m_clients.cbegin(), e->m_clients.cend(), [flag](
const Client &client) {
356 return client.m_watchModes & flag;
360 emitEvent(e, Deleted, tpath);
364 if (
event->mask & (IN_MODIFY | IN_ATTRIB)) {
365 if ((e->isDir) && (!e->m_clients.empty())) {
381 e->m_pendingFileChanges.append(tpath);
387 if (!rescan_timer.isActive()) {
388 rescan_timer.start(m_PollInterval);
392#if HAVE_INOTIFY_DIRECT_READV
393 struct iovec *received[10];
394 int num_events = libinotify_direct_readv(m_inotify_fd, received, (
sizeof (received) /
sizeof ((received)[0])), 0);
395 for (
int i = 0; i < num_events; i++) {
396 struct iovec *cur_event = received[i];
397 while (cur_event->iov_base) {
398 const struct inotify_event *
const event = (
struct inotify_event *) cur_event->iov_base;
404 libinotify_free_iovec(received[i]);
408 int offsetStartRead = 0;
411 ioctl(m_inotify_fd, FIONREAD, &pending);
413 while (pending > 0) {
414 const int bytesToRead = qMin<int>(pending,
sizeof(buf) - offsetStartRead);
416 int bytesAvailable =
read(m_inotify_fd, &buf[offsetStartRead], bytesToRead);
417 pending -= bytesAvailable;
418 bytesAvailable += offsetStartRead;
421 int offsetCurrent = 0;
422 while (bytesAvailable >=
int(
sizeof(
struct inotify_event))) {
423 const struct inotify_event *
const event =
reinterpret_cast<inotify_event *
>(&buf[offsetCurrent]);
425 if (
event->mask & IN_Q_OVERFLOW) {
426 qCWarning(KDIRWATCH) <<
"Inotify Event queue overflowed, check max_queued_events value";
430 const int eventSize =
sizeof(
struct inotify_event) +
event->len;
431 if (bytesAvailable < eventSize) {
435 bytesAvailable -= eventSize;
436 offsetCurrent += eventSize;
440 if (bytesAvailable > 0) {
442 memmove(buf, &buf[offsetCurrent], bytesAvailable);
443 offsetStartRead = bytesAvailable;
450KDirWatchPrivate::Entry::~Entry()
458void KDirWatchPrivate::Entry::propagate_dirty()
460 for (Entry *sub_entry : std::as_const(m_entries)) {
461 if (!sub_entry->dirty) {
462 sub_entry->dirty =
true;
463 sub_entry->propagate_dirty();
473 if (instance ==
nullptr) {
477 auto it = findInstance(instance);
478 if (it != m_clients.end()) {
479 Client &client = *it;
481 client.m_watchModes = watchModes;
485 m_clients.emplace_back(instance, watchModes);
488void KDirWatchPrivate::Entry::removeClient(
KDirWatch *instance)
490 auto it = findInstance(instance);
491 if (it != m_clients.end()) {
492 Client &client = *it;
494 if (client.count == 0) {
501int KDirWatchPrivate::Entry::clientCount()
const
504 for (
const Client &client : m_clients) {
505 clients += client.count;
511QString KDirWatchPrivate::Entry::parentDirectory()
const
523 for (
const Client &client : m_clients) {
524 if (client.m_watchModes & flag) {
543 for (
const Client &client : m_clients) {
544 if (client.m_watchModes & flag) {
553 if (!dwp_self.hasLocalData()) {
554 debug <<
"KDirWatch not used";
557 debug << dwp_self.localData();
561QDebug operator<<(
QDebug debug,
const KDirWatchPrivate &dwp)
563 debug <<
"Entries watched:";
564 if (dwp.m_mapEntries.count() == 0) {
567 auto it = dwp.m_mapEntries.cbegin();
568 for (; it != dwp.m_mapEntries.cend(); ++it) {
569 const KDirWatchPrivate::Entry &e = it.value();
572 for (
const KDirWatchPrivate::Client &c : e.m_clients) {
574 if (c.watchingStopped) {
575 if (c.pending & KDirWatchPrivate::Deleted) {
576 pending +=
"deleted ";
578 if (c.pending & KDirWatchPrivate::Created) {
579 pending +=
"created ";
581 if (c.pending & KDirWatchPrivate::Changed) {
582 pending +=
"changed ";
585 pending =
" (pending: " + pending +
')';
587 pending =
", stopped" + pending;
589 debug <<
" by " << c.instance->objectName() <<
" (" << c.count <<
" times)" << pending;
591 if (!e.m_entries.isEmpty()) {
592 debug <<
" dependent entries:";
593 for (KDirWatchPrivate::Entry *d : e.m_entries) {
594 debug <<
" " << d << d->path << (d->m_status == KDirWatchPrivate::NonExistent ?
"NonExistent" :
"EXISTS this is an ERROR!");
595 if (s_verboseDebug) {
596 Q_ASSERT(d->m_status == KDirWatchPrivate::NonExistent);
605QDebug operator<<(
QDebug debug,
const KDirWatchPrivate::Entry &entry)
607 debug.
nospace() <<
"[ Entry for " << entry.path <<
", " << (entry.isDir ?
"dir" :
"file");
608 if (entry.m_status == KDirWatchPrivate::NonExistent) {
609 debug <<
", non-existent";
612 << ((entry.m_mode == KDirWatchPrivate::INotifyMode) ?
"INotify"
613 : (entry.m_mode == KDirWatchPrivate::QFSWatchMode) ?
"QFSWatch"
614 : (entry.m_mode == KDirWatchPrivate::StatMode) ?
"Stat"
616#if HAVE_SYS_INOTIFY_H
617 if (entry.m_mode == KDirWatchPrivate::INotifyMode) {
618 debug <<
" inotify_wd=" << entry.wd;
621 debug <<
", has " << entry.m_clients.size() <<
" clients";
623 if (!entry.m_entries.isEmpty()) {
624 debug <<
", nonexistent subentries:";
625 for (KDirWatchPrivate::Entry *subEntry : std::as_const(entry.m_entries)) {
626 debug << subEntry << subEntry->path;
633KDirWatchPrivate::Entry *KDirWatchPrivate::entry(
const QString &_path)
645 auto it = m_mapEntries.find(path);
646 return it != m_mapEntries.end() ? &it.value() :
nullptr;
650void KDirWatchPrivate::useFreq(Entry *e,
int newFreq)
655 if (e->freq < freq) {
657 if (m_statRescanTimer.isActive()) {
658 m_statRescanTimer.start(freq);
660 qCDebug(KDIRWATCH) <<
"Global Poll Freq is now" << freq <<
"msec";
664#if HAVE_SYS_INOTIFY_H
666bool KDirWatchPrivate::useINotify(Entry *e)
671 if (!supports_inotify) {
675 e->m_mode = INotifyMode;
677 if (e->m_status == NonExistent) {
678 addEntry(
nullptr, e->parentDirectory(), e,
true);
683 int mask = IN_DELETE | IN_DELETE_SELF | IN_CREATE | IN_MOVE | IN_MOVE_SELF | IN_DONT_FOLLOW | IN_MOVED_FROM | IN_MODIFY | IN_ATTRIB;
685 if ((e->wd = inotify_add_watch(m_inotify_fd,
QFile::encodeName(e->path).data(), mask)) != -1) {
686 m_inotify_wd_to_entry.insert(e->wd, e);
687 if (s_verboseDebug) {
688 qCDebug(KDIRWATCH) <<
"inotify successfully used for monitoring" << e->path <<
"wd=" << e->wd;
693 if (errno == ENOSPC) {
696 qCWarning(KDIRWATCH) <<
"inotify failed for monitoring" << e->path <<
"\n"
697 <<
"Because it reached its max_user_watches,\n"
698 <<
"you can increase the maximum number of file watches per user,\n"
699 <<
"by setting an appropriate fs.inotify.max_user_watches parameter in your /etc/sysctl.conf";
701 qCDebug(KDIRWATCH) <<
"inotify failed for monitoring" << e->path <<
":" << strerror(errno) <<
" (errno:" << errno <<
")";
706#if HAVE_QFILESYSTEMWATCHER
707bool KDirWatchPrivate::useQFSWatch(Entry *e)
709 e->m_mode = QFSWatchMode;
712 if (e->m_status == NonExistent) {
713 addEntry(
nullptr, e->parentDirectory(), e,
true );
723 fsWatcher->addPath(e->path);
728bool KDirWatchPrivate::useStat(Entry *e)
731 useFreq(e, m_nfsPollInterval);
733 useFreq(e, m_PollInterval);
736 if (e->m_mode != StatMode) {
737 e->m_mode = StatMode;
740 if (statEntries == 1) {
742 m_statRescanTimer.start(freq);
743 qCDebug(KDIRWATCH) <<
" Started Polling Timer, freq " << freq;
747 qCDebug(KDIRWATCH) <<
" Setup Stat (freq " << e->freq <<
") for " << e->path;
761 qCWarning(KDIRWATCH) <<
"Cannot watch QRC-like path" <<
path;
777 auto it = m_mapEntries.find(path);
778 if (it != m_mapEntries.end()) {
779 Entry &entry = it.value();
781 entry.m_entries.append(sub_entry);
782 if (s_verboseDebug) {
783 qCDebug(KDIRWATCH) <<
"Added already watched Entry" <<
path <<
"(for" << sub_entry->path <<
")";
786 entry.addClient(instance, watchModes);
787 if (s_verboseDebug) {
788 qCDebug(KDIRWATCH) <<
"Added already watched Entry" <<
path <<
"(now" << entry.clientCount() <<
"clients)"
800 auto newIt = m_mapEntries.insert(path, Entry());
802 Entry *e = &(*newIt);
805 e->isDir = (stat_buf.st_mode & QT_STAT_MASK) == QT_STAT_DIR;
808 if (e->isDir && !isDir) {
810 if ((stat_buf.st_mode & QT_STAT_MASK) == QT_STAT_LNK) {
818 if (e->isDir && !isDir) {
819 qCWarning(KCOREADDONS_DEBUG) <<
"KDirWatch:" <<
path <<
"is a directory. Use addDir!";
820 }
else if (!e->isDir && isDir) {
821 qCWarning(KCOREADDONS_DEBUG) <<
"KDirWatch:" <<
path <<
"is a file. Use addFile!";
825 qCWarning(KCOREADDONS_DEBUG) <<
"KDirWatch:" <<
path
826 <<
"is a file. You can't use recursive or "
827 "watchFiles options";
833 e->m_ctime = stat_buf.st_mtime;
835 e->m_ctime = stat_buf.st_ctime;
838 e->m_nlink = stat_buf.st_nlink;
839 e->m_ino = stat_buf.st_ino;
842 e->m_ctime = invalid_ctime;
843 e->m_status = NonExistent;
850 e->m_entries.append(sub_entry);
852 e->addClient(instance, watchModes);
855 if (s_verboseDebug) {
856 qCDebug(KDIRWATCH).nospace() <<
"Added " << (e->isDir ?
"Dir " :
"File ") << path << (e->m_status == NonExistent ?
" NotExisting" :
"") <<
" for "
880#if HAVE_SYS_INOTIFY_H
881 if (m_preferredMethod == KDirWatch::INotify) {
886 filters &=
~QDir::Files;
890 QDir basedir(e->path);
891 const QFileInfoList contents = basedir.entryInfoList(filters);
892 for (
const QFileInfo &fileInfo : contents) {
894 bool isDir = fileInfo.isDir() && !fileInfo.isSymLink();
903void KDirWatchPrivate::addWatch(Entry *e)
914 KDirWatch::Method preferredMethod = m_preferredMethod;
915 if (m_nfsPreferredMethod != m_preferredMethod) {
917 preferredMethod = m_nfsPreferredMethod;
922 bool inotifyFailed =
false;
923 bool entryAdded =
false;
924 switch (preferredMethod) {
925#if HAVE_SYS_INOTIFY_H
926 case KDirWatch::INotify:
927 entryAdded = useINotify(e);
929 inotifyFailed =
true;
933 case KDirWatch::INotify:
937#if HAVE_QFILESYSTEMWATCHER
938 case KDirWatch::QFSWatch:
939 entryAdded = useQFSWatch(e);
942 case KDirWatch::QFSWatch:
946 case KDirWatch::Stat:
947 entryAdded = useStat(e);
953#if HAVE_SYS_INOTIFY_H
954 if (preferredMethod != KDirWatch::INotify && useINotify(e)) {
958#if HAVE_QFILESYSTEMWATCHER
961 if (preferredMethod != KDirWatch::QFSWatch && !inotifyFailed && useQFSWatch(e)) {
965 if (preferredMethod != KDirWatch::Stat) {
971void KDirWatchPrivate::removeWatch(Entry *e)
973#if HAVE_SYS_INOTIFY_H
974 if (e->m_mode == INotifyMode) {
975 m_inotify_wd_to_entry.remove(e->wd);
976 (void)inotify_rm_watch(m_inotify_fd, e->wd);
977 if (s_verboseDebug) {
978 qCDebug(KDIRWATCH).nospace() <<
"Cancelled INotify (fd " << m_inotify_fd <<
", " << e->wd <<
") for " << e->path;
982#if HAVE_QFILESYSTEMWATCHER
983 if (e->m_mode == QFSWatchMode && fsWatcher) {
984 if (s_verboseDebug) {
985 qCDebug(KDIRWATCH) <<
"fsWatcher->removePath" << e->path;
987 fsWatcher->removePath(e->path);
992void KDirWatchPrivate::removeEntry(
KDirWatch *instance,
const QString &_path, Entry *sub_entry)
994 qCDebug(KDIRWATCH) <<
"path=" << _path <<
"sub_entry:" << sub_entry;
996 Entry *e = entry(_path);
998 removeEntry(instance, e, sub_entry);
1002void KDirWatchPrivate::removeEntry(
KDirWatch *instance, Entry *e, Entry *sub_entry)
1004 removeList.remove(e);
1007 e->m_entries.removeAll(sub_entry);
1009 e->removeClient(instance);
1012 if (!e->m_clients.empty() || !e->m_entries.empty()) {
1017 removeList.insert(e);
1022 if (e->m_status == Normal) {
1027 removeEntry(
nullptr, e->parentDirectory(), e);
1033 if (e->m_mode == StatMode) {
1035 if (statEntries == 0) {
1036 m_statRescanTimer.stop();
1037 qCDebug(KDIRWATCH) <<
" Stopped Polling Timer";
1041 if (s_verboseDebug) {
1042 qCDebug(KDIRWATCH).nospace() <<
"Removed " << (e->isDir ?
"Dir " :
"File ") << e->path <<
" for " << (sub_entry ? sub_entry->path :
QString()) <<
" ["
1046#if HAVE_SYS_INOTIFY_H
1047 m_inotify_wd_to_entry.
remove(e->wd);
1055void KDirWatchPrivate::removeEntries(
KDirWatch *instance)
1057 int minfreq = 3600000;
1061 for (
auto it = m_mapEntries.begin(); it != m_mapEntries.end(); ++it) {
1062 Entry &entry = it.value();
1063 auto clientIt = entry.findInstance(instance);
1064 if (clientIt != entry.m_clients.end()) {
1065 clientIt->count = 1;
1066 pathList.
append(entry.path);
1067 }
else if (entry.m_mode == StatMode && entry.freq < minfreq) {
1068 minfreq = entry.freq;
1072 for (
const QString &path : std::as_const(pathList)) {
1073 removeEntry(instance, path,
nullptr);
1076 if (minfreq > freq) {
1079 if (m_statRescanTimer.isActive()) {
1080 m_statRescanTimer.start(freq);
1082 qCDebug(KDIRWATCH) <<
"Poll Freq now" << freq <<
"msec";
1087bool KDirWatchPrivate::stopEntryScan(
KDirWatch *instance, Entry *e)
1089 int stillWatching = 0;
1090 for (Client &client : e->m_clients) {
1091 if (!instance || instance == client.instance) {
1092 client.watchingStopped =
true;
1093 }
else if (!client.watchingStopped) {
1094 stillWatching += client.count;
1098 qCDebug(KDIRWATCH) << (instance ? instance->
objectName() : QStringLiteral(
"all")) <<
"stopped scanning" << e->path <<
"(now" << stillWatching
1101 if (stillWatching == 0) {
1104 e->m_ctime = invalid_ctime;
1114bool KDirWatchPrivate::restartEntryScan(
KDirWatch *instance, Entry *e,
bool notify)
1116 int wasWatching = 0;
1117 int newWatching = 0;
1118 for (Client &client : e->m_clients) {
1119 if (!client.watchingStopped) {
1120 wasWatching += client.count;
1121 }
else if (!instance || instance == client.instance) {
1122 client.watchingStopped =
false;
1123 newWatching += client.count;
1126 if (newWatching == 0) {
1130 qCDebug(KDIRWATCH) << (instance ? instance->
objectName() : QStringLiteral(
"all")) <<
"restarted scanning" << e->path <<
"(now" << wasWatching + newWatching
1136 if (wasWatching == 0) {
1138 QT_STATBUF stat_buf;
1139 bool exists = (QT_STAT(
QFile::encodeName(e->path).constData(), &stat_buf) == 0);
1143 e->m_ctime = qMax(stat_buf.st_ctime, stat_buf.st_mtime);
1145 if (s_verboseDebug) {
1146 qCDebug(KDIRWATCH) <<
"Setting status to Normal for" << e << e->path;
1148 e->m_nlink = stat_buf.st_nlink;
1149 e->m_ino = stat_buf.st_ino;
1152 removeEntry(
nullptr, e->parentDirectory(), e);
1154 e->m_ctime = invalid_ctime;
1155 e->m_status = NonExistent;
1157 if (s_verboseDebug) {
1158 qCDebug(KDIRWATCH) <<
"Setting status to NonExistent for" << e << e->path;
1171void KDirWatchPrivate::stopScan(
KDirWatch *instance)
1173 for (
auto it = m_mapEntries.begin(); it != m_mapEntries.end(); ++it) {
1174 stopEntryScan(instance, &it.value());
1178void KDirWatchPrivate::startScan(
KDirWatch *instance,
bool notify,
bool skippedToo)
1181 resetList(instance, skippedToo);
1184 for (
auto it = m_mapEntries.begin(); it != m_mapEntries.end(); ++it) {
1185 restartEntryScan(instance, &it.value(), notify);
1192void KDirWatchPrivate::resetList(
KDirWatch *instance,
bool skippedToo)
1196 for (
auto it = m_mapEntries.begin(); it != m_mapEntries.end(); ++it) {
1197 for (Client &client : it.value().m_clients) {
1198 if (!client.watchingStopped || skippedToo) {
1199 client.pending = NoChange;
1207int KDirWatchPrivate::scanEntry(Entry *e)
1210 if (e->m_mode == UnknownMode) {
1214 if (e->m_mode == INotifyMode) {
1222 if (e->m_mode == StatMode) {
1227 e->msecLeft -= freq;
1228 if (e->msecLeft > 0) {
1231 e->msecLeft += e->freq;
1234 QT_STATBUF stat_buf;
1235 const bool exists = (QT_STAT(
QFile::encodeName(e->path).constData(), &stat_buf) == 0);
1237 if (e->m_status == NonExistent) {
1240 e->m_ctime = qMax(stat_buf.st_ctime, stat_buf.st_mtime);
1242 e->m_ino = stat_buf.st_ino;
1243 if (s_verboseDebug) {
1244 qCDebug(KDIRWATCH) <<
"Setting status to Normal for just created" << e << e->path;
1247 removeEntry(
nullptr, e->parentDirectory(), e);
1253 if (s_verboseDebug) {
1254 struct tm *tmp = localtime(&e->m_ctime);
1256 strftime(outstr,
sizeof(outstr),
"%H:%M:%S", tmp);
1257 qCDebug(KDIRWATCH) << e->path <<
"e->m_ctime=" << e->m_ctime << outstr <<
"stat_buf.st_ctime=" << stat_buf.st_ctime
1258 <<
"stat_buf.st_mtime=" << stat_buf.st_mtime <<
"e->m_nlink=" << e->m_nlink <<
"stat_buf.st_nlink=" << stat_buf.st_nlink
1259 <<
"e->m_ino=" << e->m_ino <<
"stat_buf.st_ino=" << stat_buf.st_ino;
1263 if ((e->m_ctime != invalid_ctime)
1264 && (qMax(stat_buf.st_ctime, stat_buf.st_mtime) != e->m_ctime || stat_buf.st_ino != e->m_ino
1265 ||
int(stat_buf.st_nlink) !=
int(e->m_nlink)
1270 || e->m_mode == QFSWatchMode
1273 e->m_ctime = qMax(stat_buf.st_ctime, stat_buf.st_mtime);
1274 e->m_nlink = stat_buf.st_nlink;
1275 if (e->m_ino != stat_buf.st_ino) {
1279 e->m_ino = stat_buf.st_ino;
1280 return (Deleted | Created);
1293 e->m_status = NonExistent;
1295 if (e->m_ctime == invalid_ctime) {
1299 e->m_ctime = invalid_ctime;
1307void KDirWatchPrivate::emitEvent(Entry *e,
int event,
const QString &fileName)
1316#elif defined(Q_OS_WIN)
1323 if (s_verboseDebug) {
1324 qCDebug(KDIRWATCH) <<
event <<
path << e->m_clients.
size() <<
"clients";
1327 for (Client &c : e->m_clients) {
1328 if (c.instance ==
nullptr || c.count == 0) {
1332 if (c.watchingStopped) {
1337 if (event == NoChange || event == Changed) {
1340 c.pending = NoChange;
1341 if (event == NoChange) {
1347 if (event & Deleted) {
1351 c.instance->setDeleted(path);
1356 if (event & Created) {
1360 c.instance->setCreated(path);
1366 if (event & Changed) {
1370 c.instance->setDirty(path);
1378void KDirWatchPrivate::slotRemoveDelayed()
1380 delayRemove =
false;
1384 while (!removeList.isEmpty()) {
1385 Entry *entry = *removeList.begin();
1386 removeEntry(
nullptr, entry,
nullptr);
1393void KDirWatchPrivate::slotRescan()
1395 if (s_verboseDebug) {
1399 EntryMap::Iterator it;
1405 bool timerRunning = m_statRescanTimer.isActive();
1407 m_statRescanTimer.stop();
1417 it = m_mapEntries.begin();
1418 for (; it != m_mapEntries.end(); ++it) {
1424 it = m_mapEntries.begin();
1425 for (; it != m_mapEntries.end(); ++it) {
1426 if (((*it).m_mode == INotifyMode || (*it).m_mode == QFSWatchMode) && (*it).dirty) {
1427 (*it).propagate_dirty();
1432#if HAVE_SYS_INOTIFY_H
1436 it = m_mapEntries.
begin();
1437 for (; it != m_mapEntries.end(); ++it) {
1439 Entry *entry = &(*it);
1440 if (!entry->isValid()) {
1444 const int ev = scanEntry(entry);
1445 if (s_verboseDebug) {
1446 qCDebug(KDIRWATCH) <<
"scanEntry for" << entry->path <<
"says" << ev;
1449 switch (entry->m_mode) {
1450#if HAVE_SYS_INOTIFY_H
1452 if (ev == Deleted) {
1453 if (s_verboseDebug) {
1454 qCDebug(KDIRWATCH) <<
"scanEntry says" << entry->path <<
"was deleted";
1456 addEntry(
nullptr, entry->parentDirectory(), entry,
true);
1457 }
else if (ev == Created) {
1458 if (s_verboseDebug) {
1459 qCDebug(KDIRWATCH) <<
"scanEntry says" << entry->path <<
"was created. wd=" << entry->wd;
1461 if (entry->wd < 0) {
1469 if (ev == Created) {
1478#if HAVE_SYS_INOTIFY_H
1484 QStringList pendingFileChanges = entry->m_pendingFileChanges;
1486 for (
const QString &changedFilename : std::as_const(pendingFileChanges)) {
1487 if (s_verboseDebug) {
1488 qCDebug(KDIRWATCH) <<
"processing pending file change for" << changedFilename;
1490 emitEvent(entry, Changed, changedFilename);
1492 entry->m_pendingFileChanges.clear();
1496 if (ev != NoChange) {
1497 emitEvent(entry, ev);
1502 m_statRescanTimer.start(freq);
1505#if HAVE_SYS_INOTIFY_H
1507 for (Entry *e : std::as_const(cList)) {
1508 removeEntry(
nullptr, e->parentDirectory(), e);
1515bool KDirWatchPrivate::isNoisyFile(
const char *filename)
1518 if (*filename ==
'.') {
1519 if (strncmp(filename,
".X.err", 6) == 0) {
1522 if (strncmp(filename,
".xsession-errors", 16) == 0) {
1528 if (strncmp(filename,
".fonts.cache", 12) == 0) {
1536void KDirWatchPrivate::ref(
KDirWatch *watch)
1538 m_referencesObjects.push_back(watch);
1541void KDirWatchPrivate::unref(
KDirWatch *watch)
1543 m_referencesObjects.removeOne(watch);
1544 if (m_referencesObjects.isEmpty()) {
1549#if HAVE_SYS_INOTIFY_H
1550QString KDirWatchPrivate::inotifyEventName(
const inotify_event *event)
const
1552 if (
event->mask & IN_OPEN)
1553 return QStringLiteral(
"OPEN");
1554 else if (
event->mask & IN_CLOSE_NOWRITE)
1555 return QStringLiteral(
"CLOSE_NOWRITE");
1556 else if (
event->mask & IN_CLOSE_WRITE)
1557 return QStringLiteral(
"CLOSE_WRITE");
1558 else if (
event->mask & IN_MOVED_TO)
1559 return QStringLiteral(
"MOVED_TO");
1560 else if (
event->mask & IN_MOVED_FROM)
1561 return QStringLiteral(
"MOVED_FROM");
1562 else if (
event->mask & IN_MOVE)
1563 return QStringLiteral(
"MOVE");
1564 else if (
event->mask & IN_CREATE)
1565 return QStringLiteral(
"CREATE");
1566 else if (
event->mask & IN_DELETE)
1567 return QStringLiteral(
"DELETE");
1568 else if (
event->mask & IN_DELETE_SELF)
1569 return QStringLiteral(
"DELETE_SELF");
1570 else if (
event->mask & IN_MOVE_SELF)
1571 return QStringLiteral(
"MOVE_SELF");
1572 else if (
event->mask & IN_ATTRIB)
1573 return QStringLiteral(
"ATTRIB");
1574 else if (
event->mask & IN_MODIFY)
1575 return QStringLiteral(
"MODIFY");
1576 if (
event->mask & IN_ACCESS)
1577 return QStringLiteral(
"ACCESS");
1578 if (
event->mask & IN_IGNORED)
1579 return QStringLiteral(
"IGNORED");
1580 if (
event->mask & IN_UNMOUNT)
1581 return QStringLiteral(
"IN_UNMOUNT");
1583 return QStringLiteral(
"UNKWOWN");
1587#if HAVE_QFILESYSTEMWATCHER
1589void KDirWatchPrivate::fswEventReceived(
const QString &path)
1591 if (s_verboseDebug) {
1592 qCDebug(KDIRWATCH) <<
path;
1595 auto it = m_mapEntries.find(path);
1596 if (it != m_mapEntries.end()) {
1597 Entry *entry = &it.value();
1598 entry->dirty =
true;
1599 const int ev = scanEntry(entry);
1600 if (s_verboseDebug) {
1601 qCDebug(KDIRWATCH) <<
"scanEntry for" << entry->path <<
"says" << ev;
1603 if (ev != NoChange) {
1604 emitEvent(entry, ev);
1606 if (ev == Deleted) {
1608 addEntry(
nullptr, entry->parentDirectory(), entry,
true);
1612 }
else if (ev == Created) {
1615 }
else if (entry->isDir) {
1617 for (Entry *sub_entry : std::as_const(entry->m_entries)) {
1618 fswEventReceived(sub_entry->path);
1627 fsWatcher->addPath(entry->path);
1632void KDirWatchPrivate::fswEventReceived(
const QString &path)
1635 qCWarning(KCOREADDONS_DEBUG) <<
"QFileSystemWatcher event received but QFileSystemWatcher is not supported";
1643Q_GLOBAL_STATIC(
KDirWatch, s_pKDirWatchSelf)
1646 return s_pKDirWatchSelf();
1653 return s_pKDirWatchSelf.exists() && dwp_self.hasLocalData();
1658 , d(createPrivate())
1661 static QBasicAtomicInt nameCounter = Q_BASIC_ATOMIC_INITIALIZER(1);
1662 const int counter = nameCounter.fetchAndAddRelaxed(1);
1669 d->removeEntries(
this);
1681 d->addEntry(
this, _path,
nullptr,
true, watchModes);
1695 d->addEntry(
this, _path,
nullptr,
false);
1700 KDirWatchPrivate::Entry *e = d->entry(_path);
1712 d->removeEntry(
this, _path,
nullptr);
1719 d->removeEntry(
this, _path,
nullptr);
1726 KDirWatchPrivate::Entry *e = d->entry(_path);
1727 if (e && e->isDir) {
1728 return d->stopEntryScan(
this, e);
1737 KDirWatchPrivate::Entry *e = d->entry(_path);
1741 return d->restartEntryScan(
this, e,
false);
1751 d->_isStopped =
true;
1757 return d->_isStopped;
1763 d->_isStopped =
false;
1764 d->startScan(
this, notify, skippedToo);
1770 KDirWatchPrivate::Entry *e = d->entry(_path);
1775 for (
const KDirWatchPrivate::Client &client : e->m_clients) {
1776 if (client.instance ==
this) {
1786 qCDebug(KDIRWATCH) <<
objectName() <<
"emitting created" << _file;
1797 qCDebug(KDIRWATCH) <<
objectName() <<
"emitting deleted" << _file;
1804 switch (d->m_preferredMethod) {
1805 case KDirWatch::INotify:
1806#if HAVE_SYS_INOTIFY_H
1807 if (d->supports_inotify) {
1808 return KDirWatch::INotify;
1812 case KDirWatch::QFSWatch:
1813#if HAVE_QFILESYSTEMWATCHER
1814 return KDirWatch::QFSWatch;
1818 case KDirWatch::Stat:
1819 return KDirWatch::Stat;
1822#if HAVE_SYS_INOTIFY_H
1823 if (d->supports_inotify) {
1824 return KDirWatch::INotify;
1827#if HAVE_QFILESYSTEMWATCHER
1828 return KDirWatch::QFSWatch;
1830 return KDirWatch::Stat;
1840 qCCritical(KDIRWATCH) <<
"KDirwatch is moving its thread. This is not supported at this time; your watch will not watch anything anymore!"
1841 <<
"Create and use watches on the correct thread"
1842 <<
"Watch:" <<
this;
1846 d->removeEntries(
this);
1854 d = createPrivate();
1864#include "moc_kdirwatch.cpp"
1865#include "moc_kdirwatch_p.cpp"
Class for watching directory and file changes.
Method internalMethod() const
Returns the preferred internal method to watch for changes.
void setDeleted(const QString &path)
Emits deleted().
void addFile(const QString &file)
Adds a file to be watched.
bool event(QEvent *event) override
Trivial override.
void setCreated(const QString &path)
Emits created().
void stopScan()
Stops scanning of all directories in internal list.
QFlags< WatchMode > WatchModes
Stores a combination of WatchMode values.
void removeFile(const QString &file)
Removes a file from the list of watched files.
bool contains(const QString &path) const
Check if a directory is being watched by this KDirWatch instance.
bool stopDirScan(const QString &path)
Stops scanning the specified path.
static KDirWatch * self()
The KDirWatch instance usually globally used in an application.
void removeDir(const QString &path)
Removes a directory from the list of scanned directories.
void deleted(const QString &path)
Emitted when a file or directory is deleted.
void startScan(bool notify=false, bool skippedToo=false)
Starts scanning of all dirs in list.
static bool exists()
Returns true if there is an instance of KDirWatch.
void addDir(const QString &path, WatchModes watchModes=WatchDirOnly)
Adds a directory to be watched.
KDirWatch(QObject *parent=nullptr)
Constructor.
@ WatchSubDirs
Watch also all the subdirs contained by the directory.
@ WatchDirOnly
Watch just the specified directory.
@ WatchFiles
Watch also all files contained by the directory.
bool isStopped()
Is scanning stopped?
void dirty(const QString &path)
Emitted when a watched object is changed.
void setDirty(const QString &path)
Emits dirty().
QDateTime ctime(const QString &path) const
Returns the time the directory/file was last changed.
void created(const QString &path)
Emitted when a file or directory (being watched explicitly) is created.
~KDirWatch() override
Destructor.
bool restartDirScan(const QString &path)
Restarts scanning for specified path.
static KNetworkMounts * self()
Returns (and creates if necessary) the singleton instance.
@ KDirWatchDontAddWatches
Disables dir watching completely for slow paths, avoids stat() calls on added dirs and subdirs.
AKONADI_CALENDAR_EXPORT KCalendarCore::Event::Ptr event(const Akonadi::Item &item)
AKONADI_MIME_EXPORT const char Deleted[]
@ Nfs
NFS or other full-featured networked filesystems (autofs, subfs, cachefs, sshfs)
KCOREADDONS_EXPORT Type fileSystemType(const QString &path)
For a given path, returns the filesystem type, one of KFileSystemType::Type values.
QVariant read(const QByteArray &data, int versionOverride=0)
QString path(const QString &relativePath)
Absolute libexec path resolved in relative relation to the current shared object.
bool isEmpty() const const
QDateTime fromSecsSinceEpoch(qint64 secs)
QString cleanPath(const QString &path)
bool isRelativePath(const QString &path)
QString decodeName(const QByteArray &localFileName)
QByteArray encodeName(const QString &fileName)
QString absolutePath() const const
void directoryChanged(const QString &path)
void fileChanged(const QString &path)
void append(QList< T > &&value)
qsizetype count() const const
bool isEmpty() const const
virtual bool event(QEvent *e)
QObject * parent() const const
void setObjectName(QAnyStringView name)
QThread * thread() const const
void activated(QSocketDescriptor socket, QSocketNotifier::Type type)
QString arg(Args &&... args) const const
bool endsWith(QChar c, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
qsizetype length() const const
QString & remove(QChar ch, Qt::CaseSensitivity cs)
qsizetype size() const const
bool startsWith(QChar c, Qt::CaseSensitivity cs) const const
qsizetype removeDuplicates()
QStringView left(qsizetype length) const const
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)