On Oct 29, 2013, at 11:37 AM, Jan Kara <jack@xxxxxxx> wrote: > On Tue 29-10-13 16:27:02, Pavel Raiskup wrote: >> >> Well, I now recalled somehow relevant Red Hat bug, sorry I have not >> mentioned it before: >> https://bugzilla.redhat.com/show_bug.cgi?id=757557 >> >> CC'ing fs-devel: The question is whether that ^^^^ is not a bug in >> filesystem — whether filesystem should not _always_ return to fstat() >> block count at least 1 if there are at least some data (even if these data >> are inlined in inode)? Just for catching the context, this thread starts >> here: http://lists.gnu.org/archive/html/bug-tar/2013-10/msg00030.html > > So 'st_blocks' should be "the number of blocks allocated to the file, > 512-byte units". If we are able to store the whole file within inode, we > have no blocks allocated and thus setting st_blocks to 0 looks as a decent > thing to do. Looking into filesystems where this is possible (ext4, btrfs, > reiserfs) they will all set st_blocks to 0 if the file body is inlined and > the size is smaller than 512 bytes. One could consider that any inode with inline data still needs to have the inode allocated to hold the data. > ext4 is yet a different matter. It does really report the number of > allocated blocks in st_blocks so it will report 0 while data can fit into > the inode (whose size is configurable during fs creation, default is 256). > In practice that will result in reporting non-zero st_blocks for 512-byte > and larger files anyways so there won't be an observable difference between > what ext4 and btrfs / reiserfs do. But we might still want to fix up ext4 > to be consistent with btrfs and reiserfs so that things are more > future-proof. Given that tar and rsync DO exist that silently drop user data for files with st_blocks == 0 (we have seen this in with Lustre tests) it makes sense to fix the ext4_getattr() to set st_blocks = 1 for files with inline data. It is already doing something similar for files with delalloc blocks to work around the same problem. >> If that is not a bug in fs, is there possible to detect that particular >> file is completely sparse? > As Joerg wrote, SEEK_DATA / SEEK_HOLE is a proper interface for this at > least for systems that support it. Once you have called stat(2) on the > file, inode will be in cache anyways so the additional cost of open(2), > lseek(2), close(2) won't be that big. For systems that don't support > SEEK_DATA / SEEK_HOLE, you can use some heuristic like: > if (st.st_size < st.st_blksize || st.st_blocks > 0) > /* Bite the bullet and scan the data for non-zero bytes */ > else > /* Assume the file is sparse */ I don’t see how this is better than the existing heuristic: if (st.st_blocks > 0) /* file has data */ else /* file is sparse */ that tools/applications are already using today. It isn’t possible to retroactively fix tar, rsync, etc. to use SEEK_DATA/SEEK_HOLE, but in the ext4 case the inline data feature is very new and it makes sense to fix this in the kernel. Of course it ALSO makes sense to fix this in userspace to handle the (st.st_size < st.st_blksize) case (this is not a lot of overhead) so that the chance of losing data in a variety of kernel/userspace combinations is minimized. Cheers, Andreas -- To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html