diff --git a/include/crucible/fd.h b/include/crucible/fd.h index 8492846..9818819 100644 --- a/include/crucible/fd.h +++ b/include/crucible/fd.h @@ -13,6 +13,10 @@ #include #include +// ioctl +#include +#include + // socket #include @@ -141,6 +145,8 @@ namespace crucible { Stat &lstat(const string &filename); }; + int ioctl_iflags_get(int fd); + string st_mode_ntoa(mode_t mode); // Because it's not trivial to do correctly diff --git a/lib/fd.cc b/lib/fd.cc index e0735a9..931167f 100644 --- a/lib/fd.cc +++ b/lib/fd.cc @@ -488,6 +488,14 @@ namespace crucible { lstat(filename); } + int + ioctl_iflags_get(int fd) + { + int attr = 0; + DIE_IF_MINUS_ONE(ioctl(fd, FS_IOC_GETFLAGS, &attr)); + return attr; + } + string readlink_or_die(const string &path) { diff --git a/src/bees-roots.cc b/src/bees-roots.cc index 921132c..e99dc66 100644 --- a/src/bees-roots.cc +++ b/src/bees-roots.cc @@ -576,6 +576,14 @@ BeesRoots::open_root_ino_nocache(uint64_t root, uint64_t ino) break; } + int attr = ioctl_iflags_get(rv); + if (attr & FS_NOCOW_FL) { + BEESLOG("Opening " << name_fd(root_fd) << "/" << file_path << " found incompatible flags " << attr << " (FS_NOCOW_FL)"); + rv = Fd(); + BEESCOUNT(open_wrong_flags); + break; + } + // Correct root? auto file_root = btrfs_get_root_id(rv); if (file_root != root) {