On Thu, Mar 21, 2019 at 08:27:40AM -0400, Brian Foster wrote: > On Thu, Mar 21, 2019 at 08:10:19AM +1100, Dave Chinner wrote: > > On Wed, Mar 20, 2019 at 08:02:05AM -0400, Brian Foster wrote: > > > On Wed, Mar 20, 2019 at 07:25:37AM +1100, Dave Chinner wrote: > > > > > > The broader point is that by controlling writeback ordering, we can > > > > > > explicitly demonstrate stale data exposure. If fixed properly, we should > > > > > > never expect stale data exposure even in the event of out of order > > > > > > writeback completion, regardless of the cause. > > > > > > > > > > Agree. Even if s_f_r is a giant trash fire, if we're exposing stale > > > > > disk contents it's game over anyway and we ought to fix it. > > > > > > > > Well, that's exactly the problem with SFR, isn't it? The dirty data > > > > is outside the range the user asks to sync and the implementation > > > > doesn't go anywhere near the filesystem so we can't actually do the > > > > right thing here. So we are left to either hack fixes around a > > > > shitty, broken interface and everyone pays the penalty, or we ignore > > > > it. We've simply ignored it for the past 10+ years because it really > > > > can't be used safely for it's intended purposes (as everyone who > > > > tries to use it finds out eventually)... > > > > > > > > > > Just to make this abundantly clear, there is nothing unique, magic or > > > special required of sync_file_range() to reproduce this stale data > > > exposure. See the following variation of a command sequence from the > > > fstest I posted the other day: > > > > > > ... > > > # xfs_io -fc "pwrite 0 68k" -c fsync -c "pwrite 96k 4k" \ > > > -c "fpunch 96k 4k" <file> > > > ... > > > # hexdump <file> > > > 0000000 cdcd cdcd cdcd cdcd cdcd cdcd cdcd cdcd > > > * > > > 0011000 abab abab abab abab abab abab abab abab > > > * > > > 0018000 0000 0000 0000 0000 0000 0000 0000 0000 > > > * > > > 0019000 > > > > > > Same problem, and it can be reproduced just as easily with a reflink or > > > even an overlapping direct I/O. > > > > How does overlapping direct IO cause stale data exposure given that > > it uses unwritten extent allocation? That, of course, is ignoring > > the fact that the result of overlapping concurent direct IO is, by > > definition, undefined and an application bug if it occurs.... > > > > It's overlapping a buffered write and so starts with a flush instead of Ok, but that's not the normal usage of the phrase "overlapping direct IO". Overlapping direct IO means two concurrent direct IOs to the same LBA range in the device underlying the filesystem. i.e. they overlap in both the temporal and physical (device LBA) domains. Subtle difference, yes, but a very important one. > an unwritten allocation (so even a dio read is sufficient). Also, the > target of the dio isn't where stale data is exposed, so even if there > was an existing underlying unwritten block before the initial buffered > write it wouldn't change anything. > > Yes, this is technically usage with undefined behavior, but that still > misses the forest from the trees. No, I'm not missing the forest for the trees. > I'm not claiming this is good or > supported practice just as I'm not not claiming sync_file_range() should > be used for anything in practice. I'm just pointing out it's one of > several possible, obvious data exposure vectors. I suspect there are > others; these were just the easy variants to reproduce. I'm aware that we have these problems - we've known about them for 20+ years and the many ways they can be exposed. Indeed, We've been talking about using unwritten extents for delalloc for as long as I can remember, but never done it because the actual occurrence of these problems in production systems is /extremely/ rare and the cost of using unwritten extents everywhere will affect /everyone/. i.e. It's always been easy to create test conditions to expose stale data, but reported cases of stale data exposure after a production system crash are pretty much non-existent. This just isn't a common problem that users are experiencing, but solving it with "unwritten extents for everyone" comes at a substantial cost for /everyone/, /all the time/. That's why a) it hasn't been done yet; and b) any discussion about "use unwritten extents everywhere" always ends up in trying to find a solution that isn't just applying the "big hammer" and ignoring the problems that causes. > If you prefer to > reason about reproducer patterns that don't use operations with quirks > or undefined behaviors, that's fine, just use the reflink or hole punch > example posted above that produce the same behavior. Well, yes. But it does go both ways - if you don't want people to say "that's undefined" or "that makes no sense", then don't use those quirky examples if you can use better, unambiguous examples. For example, that's exactly how I phrased this extension flushing mechanism trigger: > > Eseentially, what this says to me is we need to track when we > > extended the file via a write beyond EOF via an inode flag. Then if we need to > > flush a data range from the file for integrity reasons (such as a > > fpnuch) and this flag is set, we ignore the "start" parameter of the See? Entirely generic trigger description, uses hole punch as an example, but also covers all the quirky cases such as direct IO range flushing without having to mention them explicitly. > > range and flush from the start of the file instead. i.e. we > > guarantee that any dirty zeroed pages between the start of the file > > and the end of the range we are operating on gets written back. This > > will capture all these "didn't flush regions zeroed on file > > extension" problems in one go, without needing to change the way we > > do allocation or EOF zeroing. > > > > I don't think that would prevent the problem outside of the common write > extension case. I feel like we're going around in circles now, Brian. :/ I didn't propose this as the "only solution we need". I've proposed it in the context that we use "only use unwritten extents for allocations within EOF" to solve the common "write within EOF" case. This, however, doesn't solve this common write extension case where we don't want to use unwritten extents because of the performance cost. And now you're saying that "well, this extending write thing doesn't solve the inside EOF problem"...... > It may be possible to selectively (i.e. as an optimization) use an > XFS_ITRUNCATED like algorithm as you propose here to always force in > order writeback in the cases where writeback extends the on-disk isize > (and thus isize protects from stale exposure), but I think we'd still > need something tied to extent allocation one way or another in addition > (like Darrick's original unwritten extent patch or your proposed > intent/done idea in the previous reply) to address the fundamental > problem. The question is what is the ideal solution.. The solution will be a combination of things that work together, not one big hammer. The more we can avoid exposure vectors by operational ordering rather than transactional state changes, the better the solution will perform. Cheers, Dave. -- Dave Chinner david@xxxxxxxxxxxxx