This check removes unneeded scaling of arguments when calling Abseil Time factory functions.
Details
Diff Detail
Event Timeline
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
37 | This will cause memory leaks, so may be unique_ptr should be used to hold pointer? May be LLVM ADT has better container? | |
60 | else after return. Same in other places. | |
165 | Should be llvm_unreachable(). | |
289 | Please remove trailing return and comment above. See readability-redundant-control-flow. | |
docs/ReleaseNotes.rst | ||
88 | Second statement belongs to documentation. | |
docs/clang-tidy/checks/abseil-duration-factory-scale.rst | ||
8 | Please fix double space. |
Address reviewer comments
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
37 | This is a tradeoff between leaking a small amount of known memory for the duration of the program, and constructing this map every time this function is invoked. Since I expect the latter to occur frequently, that's a tradeoff I think is acceptable. (Ideally, this would be a compile-time constant, but sadly we don't yet have a constexpr dictionary type.) Of course, if there is a more typical way of doing that here, I'm happy to use it. | |
docs/ReleaseNotes.rst | ||
88 | It already is in the user-facing documentation, are saying it should be removed from here? |
docs/ReleaseNotes.rst | ||
---|---|---|
88 | Release Notes are for brief descriptions. Documentation is for details. |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
37 | Why did you not use a static value type? |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
23 | nit: wrap it in anonymous namespace? | |
37 | I think main reason is lazy initialization. | |
75 | we are comparing with floats, I think we should use something AlmostEquals(Multiplier, 60.0). | |
203 | isn't Call->getExprLoc().isMacroID() enough? | |
267 | is this for debugging? |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
37 | Static locals are lazily initialized, so I'm still not certain why this shouldn't use static const std::unorderd_map<std::string, DurationScale> ScaleMap{...}; instead. | |
47 | Elide braces. | |
58 | Elide braces. | |
61 | Please add a message to the assertion. e.g., assert(foo && "bar"); | |
180–182 | ::absl instead -- otherwise this can trigger in unintended places. |
Addressed reviewer feedback.
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
75 | I can't find AlmostEquals elsewhere; could you point me to it so I can investigate it's use? But I am also curious why you think we should use this. We are checking for exact values. 60.0 is representable exactly as a double, and even values which aren't (e.g., 1e-3) will have the same representation in this function as they do in the code being transformed, so equality is still appropriate. | |
203 | Thanks! I've also added tests to verify this. |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
39 | GetScaleForFactory -> getScaleForFactory, per naming conventions. Same for the other functions. | |
48 | Drop top-level const (this returns an iterator, not a pointer/reference). | |
55 | it's -> its | |
57–58 | I really don't like this interface where you pass two arguments, only one of which is ever valid. That is pretty confusing. Given that the result of this function is only ever passed to GetNewMultScale(), and that function only does integral checks, I'd prefer logic more like:
With that logic, I don't see a need for GetValue() at all, but if a helper function is useful, I'd probably guess this is a better signature: int64_t getIntegralValue(const Expr *Literal, bool &ResultIsExact); | |
59 | Elide these braces. | |
63 | I believe the approximate results here can lead to bugs where the floating-point literal is subnormal -- it may return 0.0 for literals that are not zero. | |
72 | Hours with a multiplier of 1.0/60.0 would scale to minutes, wouldn't it? | |
81–84 | What about scaling with a multiplier of 3600 to go from seconds to hours, and other plausible conversions? | |
112 | Similar comments here as above. In fact, I feel like these functions should be combined into getNewScale() and have normalized the scaling value in the caller so that it covers both operations rather than manually spelling out the conversions in either direction. e.g., convert everything into multiplier form in the caller by treating divisors as a multiplication by 1/divisor. | |
150 | This should return a StringRef. | |
194 | and -> to | |
198 | I suspect you want to ignore paren expressions as well, so IgnoreParenImpCasts(). | |
213 | Do not use auto here as the type is not explicitly spelled out in the initialization. | |
clang-tidy/abseil/DurationFactoryScaleCheck.h | ||
19 | funciton -> function |
Addressed small concerns.
Larger issues pending feedback.
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
57–58 |
That's actually not true: GetNewMultScale() does checks against values like 1e-3 which aren't integers. Does this change your suggestion? | |
63 | Do you have an example which I could put in a test? | |
81–84 | That's a good point, and part of a broader design discussion: should we support all multipliers? (e.g., what about multiplying microseconds by 1.0/86400000000.0?) If we do think it's worth handling all of these cases, we probably want a different construct than the equivalent of a lookup table to do this computation. |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
57–58 | Hmm, yeah, I suppose it has to! :-D | |
63 | 0x0.000001p-126f should get you a new, exciting way to spell 0. | |
81–84 |
That's kind of what I'm leaning towards. It's certainly more explainable to users that all the various scaling operations just work, rather than some particular set. However, maybe you know more about the user base than I do and there's a sound reason to not handle all cases?
There's a part of me that wonders if we can use std::ratio to describe the scaling operations, but I freely admit I've not thought about implementation strategies all that hard. |
Combined multiplication and division logic, and also now handles scaling of multiple steps (e.g., Seconds * 3600).
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
57–58 | I've reworked the bulk of this logic. It still uses doubles, but that doesn't trouble our test cases. Please let me know if there's more to do here. | |
63 | I've added this as a test, and it resolves as normal. Was your comment intended to indicate that it should, or that doing so would be a bug? | |
81–84 | std::ratio didn't look like it made much sense here (though I don't have much first-hand experience using it), but we do now handle multiple scaling steps. |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
63 | Hmm, in hindsight, I think this code is okay as-is. The situation I was worried about is where compile-time evaluation gives you one value (say, 0) while runtime evaluation gives you another (nonzero) because of the floating-point format for the target architecture. However, 1) I think APFloat does a good job of avoiding that, so we may be fine by default, and 2) I am questioning whether abseil users would be writing those constants anyway. | |
74 | This doesn't seem quite right. 1.0/6000.0 is less than 1.0/60.0 but isn't a Minutes scaling. |
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
74 | After spending two days chasing this down, it finally occurred to me: we don't actually handle this case. Our (somewhat conservative) matchers only look for literal values, not division expressions like 1.0/60.0. And for this iteration of the tool, I think I'm fine with that, since I expect those kinds of things to be exceedingly rare, and probably tricky to handle, anyway. |
LGTM!
clang-tidy/abseil/DurationFactoryScaleCheck.cpp | ||
---|---|---|
74 | Oh jeez -- good point on not covering constant expressions from the matcher. If you think this is going to be exceedingly rare, I'm okay with punting until we need it. Thank you for looking into it! |
@aaron.ballman I don't actually have the commit bit, can you commit this, or are we waiting for further review?
No further review required. I'm happy to commit for you. I'll do it later today or tomorrow, unless someone else gets to it first.
funciton -> function