On Mon, Jul 24, 2017 at 11:44:56AM -0700, Darrick J. Wong wrote: > On Mon, Jul 24, 2017 at 09:07:52AM -0400, Brian Foster wrote: > > On Mon, Jul 24, 2017 at 03:26:38PM +1000, Dave Chinner wrote: > > > On Sat, Jul 22, 2017 at 08:29:44AM -0400, Brian Foster wrote: > > > > On Sat, Jul 22, 2017 at 09:00:58AM +1000, Dave Chinner wrote: > > > > > On Fri, Jul 21, 2017 at 09:54:05AM -0400, Brian Foster wrote: > > > > > > On Fri, Jul 21, 2017 at 08:49:34AM +1000, Dave Chinner wrote: > > > > > > but I'm also not against crafting a new, verifier > > > > > > specific one to accomplish that. Technically, it doesn't have to be > > > > > > shouty :), but IMO, the diagnostic/usability benefit outweighs the > > > > > > aesthetic cost. > > > > > > > > > > I disagree - there are so many verifier checks (and we only grow > > > > > more as time goes on) so whatever we do needs them to be > > > > > easily maintainable and not compromise the readabilty of the verifer > > > > > code. > > > > > > > > > > > > > I agree. ;) But I disagree that using a macro somehow automatically > > > > negatively affects the readability of verifers. Anybody who can make > > > > sense of this: > > > > > > > > if (!xfs_sb_version_hascrc(&mp->m_sb)) > > > > return __line__; > > > > if (dsl->sl_magic != cpu_to_be32(XFS_SYMLINK_MAGIC)) > > > > return __line__; > > > > if (!uuid_equal(&dsl->sl_uuid, &mp->m_sb.sb_meta_uuid)) > > > > return __line__; > > > > > > > > ... could just as easily make sense of something like this: > > > > > > > > xfs_verify_assert(xfs_sb_version_hascrc(&mp->m_sb)); > > > > xfs_verify_assert(dsl->sl_magic == cpu_to_be32(XFS_SYMLINK_MAGIC)); > > > > xfs_verify_assert(uuid_equal(&dsl->sl_uuid, &mp->m_sb.sb_meta_uuid)); > > > > > > Still needs branches to return on error so the verifier code > > > functions correctly. So it simply isn't as clean as your example > > > suggests. And, no hiding goto/return/errno variables inside a macro > > > is not an option - that makes for horrible code and we should not > > > repeat those past mistakes we inherited from the old Irix > > > codebase. > > > > > > > Clearly the example implies (and the supporting text explains) that the > > macro would lead to function exit. > > It does? Every time I see an *assert() I assume that execution either > continues past it or stops dead in its tracks. The _GOTO macros make > that clear by pairing the word goto with a label, though I concede that > it feels odd to pass a label as an argument. > I wasn't referring to the macro name, I just picked that out of a hat. I mentioned that the macro would return in the sentence immediately following the example. _GOTO/_RETURN or something like that certainly makes sense for a legitimate name in this case. > > I don't really see the problem with doing that. Again, we already do > > that all over the place. If we don't want to alter function execution > > via the macro, then have it return an expression that resolves to > > false (or an error code or some such) and leave the function execution > > logic in function context. It doesn't really matter to me either way. > > I get that hiding branches inside macros makes it too easy for later > readers to be mislead, but if there's one aspect of them that I /do/ > like, its that (provided the naming makes it clear that we're branching > somewhere) we /can/ use them to ensure that the return values are > consistent and avoid screwing up repetitive if tests. > > Anyway, I was picturing something like this: > > #define XFS_VERIFY_FAIL_UNLESS(expr) do { if (expr) return _THIS_IP_; } while(0) > > static unsigned long > xfs_blah_verify(...) > { > XFS_VERIFY_FAIL_UNLESS(blah1 == GOOD1); > XFS_VERIFY_FAIL_UNLESS(blah2 == GOOD2); > XFS_VERIFY_FAIL_UNLESS(blah3 == GOOD3); > ... > return 0; > } > > Granted, with code review to spot inconsistencies maybe the macro really > is pointless, and we can just open-code everything. I dunno. To be > honest I actually do prefer the macro in this particular case. > I think you could argue it actually cleans up the code by hiding the fact that the code does something unusual by returning a line number or instruction pointer). That's not something you typically see in !DEBUG code in general. > static unsigned long > xfs_blah_verify(...) > { > if (!(blah1 == GOOD1)) > return _THIS_IP_; > if (!(blah2 == GOOD2)) > return _THIS_IP_; > if (!(blah3 == GOOD3)) > return _THIS_IP_; > ... > return 0; > } > ... > > I don't think there's much difference in the generated code between > opencoded verifier tests and a macro that more or less does the same > thing but also helps us to avoid copy-pasta mistakes. TBH I'm more > worried about us making a mistake in the verifiers that screw up what we > accept coming from / allow to be written to disk than I am about the > exact size of each verifier function. > > That said, I've warmed to Dave's idea of passing back _THIS_IP_ as a > compromise between __line__ (not quite enough info) and throwing > (#fs_ok, __func__, __line__) back to callers. > I like the idea of using _THIS_IP_ over __LINE__ because the former at least returns a precise reference straight from the kernel (as opposed to one that is partly subject to interpretation). It can be easily converted to a file/line combination with existing tools as well. ... > > Admittedly, bloating up the scrub module with stringified stuff has made > it way quicker for me to figure out what got flagged as a failure and > then decide if the check I wrote is incorrect or if we really did find > something broken. One more upside to the macro is that we could easily > make each verifier test spit out more information if CONFIG_XFS_DEBUG is > set. This won't help us with customers encountering production > problems, obviously, but I think there's still value. > I agree that there is value here... > > Of course, once we get into debugging an actual problem, we've kind of > > moved outside the scope of this discussion... > > > > > Really, verifiers are performance critical fast paths, so the code > > > that runs inside them needs to be implemented with this in mind. I'm > > > not against making messages pretty, but let's not do it based on the > > > on the premise that pretty error messages are more important than > > > runtime performance when there are no errors... > > > > > > > I agree. As noted above, I would not expect any more overhead than the > > original suggestion to return a line number in the normal runtime case. > > The difference is primarily what we do in the code between when a > > verifier check fails and we report the failure. > > > > FWIW, an approach that might be a reasonable compromise is to use the > > original suggestion to pass along the line number, but also wrap each > > verifier check in a thinner macro that simply asserts[2] on the verifier > > expression. The new macro could be ifdef'd out completely in production > > builds to eliminate code size concerns. Then if we have reports with > > significant enough corruption issues to warrant it, we collect a > > metadump and use any local debug kernel to access the fs and decode the > > errors for us (or ask the user to run a debug kernel). We can expand the > > verifier macro in the future with production build support if warranted. > > Thoughts? > > <shrug> _THIS_IP_ ? > I'm not following how _THIS_IP_ is relevant to this point..? The compromise I suggested above is very similar to what you've suggested earlier with regard to CONFIG_XFS_DEBUG. We'd basically automatically print assert failures for failed verifier checks (it would require at least XFS_WARN and perhaps skip the backtrace, but that's getting into details) via use of a macro. The value of the macro is that we can at least conditionally stringify the checks to support the ability to use a debug kernel to process metadumps or targeted environments that might have too many corruption errors to resolve to specific LOC manually. It kind of sounds like we fundamentally agree on that point. :) Brian > (Sorry for starting to sound like a broken record...) > > > Brian > > > > [1] A simple test that we can do at the moment is to compare a > > production xfs.ko with one with asserts enabled. My local for-next > > branch builds a module of 40104000 bytes without XFS_WARN and 40540208 > > bytes with XFS_WARN enabled. That makes for a difference of ~425k and > > roughly a 1% code size increase (accounting for ~1750 asserts and > > supporting code). > > > > [2] We may not want to use ASSERT() here, but perhaps define a new > > variant to omit the stack trace and otherwise customize to a verifier > > report. > > Speaking of ASSERTs, I was going to propose removing the ASSERT calls > from the XFS_WANT_CORRUPTED_* macros since they're used in some of the > directory verifiers and we shouldn't really BUG_ON corrupted disk > contents. An XFS_ERROR_REPORT follows soon after the ASSERT, so imho I > think we could get by with only one round of complaining. > > --D > > > > > > Cheers, > > > > > > Dave. > > > > > > -- > > > Dave Chinner > > > david@xxxxxxxxxxxxx > > > -- > > > To unsubscribe from this list: send the line "unsubscribe linux-xfs" in > > > the body of a message to majordomo@xxxxxxxxxxxxxxx > > > More majordomo info at http://vger.kernel.org/majordomo-info.html > > -- > > To unsubscribe from this list: send the line "unsubscribe linux-xfs" in > > the body of a message to majordomo@xxxxxxxxxxxxxxx > > More majordomo info at http://vger.kernel.org/majordomo-info.html > -- > To unsubscribe from this list: send the line "unsubscribe linux-xfs" in > the body of a message to majordomo@xxxxxxxxxxxxxxx > More majordomo info at http://vger.kernel.org/majordomo-info.html -- To unsubscribe from this list: send the line "unsubscribe linux-xfs" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html