Finds member initializations in the constructor body which can be placed into the initialization list instead. This does not only improves the readability of the code but also affects positively its performance. Class-member assignments inside a control statement or following the first control statement are ignored.
Details
Diff Detail
- Repository
- rG LLVM Github Monorepo
Event Timeline
affects positively its performance.
Interesting! Do you have some numbers you could share? thanks
Can you refresh my memory on whether a rule for "if init expr is constant, initialise in class body instead" exists for init list members? If so, this will be a funny "two pass needed to fix" kind of check.
clang-tools-extra/clang-tidy/readability/PreferInitializationListCheck.cpp | ||
---|---|---|
19 ↗ | (On Diff #232816) | I'm working on a checker which has the need for similarly knowing occurrences of "control flow breaking statements". How about goto and calling a [[noreturn]] function, such as (std::)longjmp? Or there is no point in matching such in your checker? |
71 ↗ | (On Diff #232816) | FIXME remained. Did you upload the right patch set? |
85 ↗ | (On Diff #232816) | can -> should? |
clang-tools-extra/clang-tidy/readability/PreferInitializationListCheck.h | ||
18 ↗ | (On Diff #232816) | FIXME remained here. |
clang-tools-extra/docs/clang-tidy/checks/readability-prefer-initialization-list.rst | ||
7 ↗ | (On Diff #232816) | improves -> improve |
8 ↗ | (On Diff #232816) | word order: also positively affects |
27 ↗ | (On Diff #232816) | typo: construcotr |
27–28 ↗ | (On Diff #232816) | [l]ist, unlike m, as m's initialization follow a control statement (if) |
clang-tools-extra/test/clang-tidy/checkers/readability-prefer-initialization-list.cpp | ||
---|---|---|
221 ↗ | (On Diff #232816) | Comment diverged from what's actually written in the code. |
Clang-tidy also has modernize-use-default-member-init. Will be good idea to mention this check in documentation and otherwise as well as draw distinction (C++ version, coding guidelines, etc) in use cases.
clang-tools-extra/clang-tidy/readability/PreferInitializationListCheck.cpp | ||
---|---|---|
66 ↗ | (On Diff #232816) | Will be good idea to check for C++. |
clang-tools-extra/docs/ReleaseNotes.rst | ||
85 | Please synchronize with first statement in documentation. |
This worries me as well -- we already have checks that prefer doing an in-class initialization to using a constructor member initialization list. How should this check interact with modernize-use-default-member-init?
clang-tools-extra/clang-tidy/readability/ReadabilityTidyModule.cpp | ||
---|---|---|
91 ↗ | (On Diff #233087) | I have a concern with the name using initialization-list -- that sounds a lot like the check is going to prefer {42} to = 42. Perhaps the name should be prefer-ctor-member-initialization or something along those lines? |
This check is to enforce C++ core guideline C.49 while the modernize check enforces guideline C.48. The two must be synchronized, and I think that this new check should do that: for initializations that should be done using in-class initializers according to the other checker this checker must suggest the same. For the rest we should suggest member initializers in the constructor.
This check is to enforce C++ core guideline C.49 while the modernize check enforces guideline C.48. The two must be synchronized, and I think that this new check should do that: for initializations that should be done using in-class initializers according to the other checker this checker must suggest the same. For the rest we should suggest member initializers in the constructor.
It should be in cppcoreguidelines module.
Now checker proposes default member initialization if applicable. Thus it is in sync with checker modernize-use-default-member-init. Also moved to ícppcoreguidelinesí group.
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
53 | Please elide braces. | |
clang-tools-extra/docs/ReleaseNotes.rst | ||
70 | Is it relevant? | |
73 | Is it relevant? | |
clang-tools-extra/docs/clang-tidy/checks/cppcoreguidelines-prefer-member-initializer.rst | ||
16 | Enum should be in double back-ticks as language construct. |
By the word, please rebase, because Release Notes were reset after version 10 branching.
clang-tools-extra/docs/clang-tidy/checks/cppcoreguidelines-prefer-member-initializer.rst | ||
---|---|---|
73 | Please make indentation 2 characters, same as in above code snippets. |
clang-tools-extra/clang-tidy/cppcoreguidelines/CppCoreGuidelinesTidyModule.cpp | ||
---|---|---|
48 | Please keep this list sorted alphabetically. | |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
44 | What about other kinds of literals like user-defined literals, or literal class types? Should this be using E->getType()->isLiteralType()? |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
44 | E->getType()->isLiteralType() does not work since it tells about the type, not the expression itself. Here we should return true for literals only, not for any expressions whose type is a literal type. Thus int is a literal type, 5 is an int literal, but n or return_any_int() not. |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
26 | How about throw expressions? | |
62–66 | if (const auto *DRE = dyn_cast<DeclRefExpr>(Value)) return isa<EnumConstantDecl>(DRE->getDecl()) return false; | |
127–129 | This can be hoisted into the matcher with hasBody(anything()) I believe. One interesting test case that's somewhat related are constructors that use a function-try-block instead of a compound statement. e.g., class C { int i; public: C() try { i = 12; } catch (...) { } }; | |
clang-tools-extra/docs/clang-tidy/checks/cppcoreguidelines-prefer-member-initializer.rst | ||
7 | placed to the -> converted into | |
8 | does not -> not |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
127–129 | I think we should skip these cases similarly to initialization inside a try block in a body that is compound statement. Maybe the initialization throws thus it is inappropriate to convert it to member initializer. I also added a test case for this. |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
107–108 | This should now be done by overriding bool isLanguageVersionSupported(const LangOptions &LangOpts) const override;, see bugprone/ThrowKeywordMissingCheck.h for an example. | |
clang-tools-extra/docs/clang-tidy/checks/cppcoreguidelines-prefer-member-initializer.rst | ||
4 | You should probably link to the rule from somewhere in these docs. | |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
29 ↗ | (On Diff #266277) | By my reading of the core guideline (https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#c48-prefer-in-class-initializers-to-member-initializers-in-constructors-for-constant-initializers), it looks like n should also be diagnosed because all of the constructors in the class initialize the member to the same constant value. Is there a reason to deviate from the rule (or have I missed something)? Also, I'd like to see a test case like: class C { int n; public: C() { n = 0; } explicit C(int) { n = 12; } }; |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) | This check only cares for initializations inside the body (rule C.49, but if the proper fix is to convert them to default member initializer according to rule C.48 then we follow that rule in the fix). For initializations implemented as constructor member initializers but according to C.48 they should have been implemented as default member initializers we already have check modernize-use-default-member-init. |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) | Thank you for the explanation. I have the feeling that these two rules are going to have some really weird interactions together. For instance, the example you added at my request shows behavior I can't easily explain as a user: class Complex19 { int n; // CHECK-FIXES: int n{0}; public: Complex19() { n = 0; // CHECK-MESSAGES: :[[@LINE-1]]:5: warning: 'n' should be initialized in an in-class default member initializer [cppcoreguidelines-prefer-member-initializer] // CHECK-FIXES: {{^\ *$}} } explicit Complex19(int) { // CHECK-FIXES: Complex19(int) : n(12) { n = 12; // CHECK-MESSAGES: :[[@LINE-1]]:5: warning: 'n' should be initialized in a member initializer of the constructor [cppcoreguidelines-prefer-member-initializer] // CHECK-FIXES: {{^\ *$}} } ~Complex19() = default; }; Despite both constructors having the assignment expression n = <literal>; one gets one diagnostic + fixit and the other gets a different diagnostic + fixit. Also, from reading C.49, I don't see anything about using in-class initialization. I see C.49 as being about avoiding the situation where the ctor runs a constructor for a data member only to then immediately in the ctor body make an assignment to that member. You don't need to initialize then reinitialize when you can use a member init list to construct the object properly initially. |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) | Thank you for your comment. While I can agree with you, I think that this check should be fully consistent with modernize-use-default-member-init. Thus I tried it on the following example: class C { int n; public: C() : n(1) {} C(int nn) : n(nn) {} int get_n() { return n; } }; I got the following message from Clang-Tidy: warning: use default member initializer for 'n' [modernize-use-default-member-init] int n; ^ {1} This is no surprise, however. If you take a look on the example in C.48: Prefer in-class initializers to member initializers in constructors for constant initializers you can see that our code is almost the same as the example considered bad there. So rule C.49 does not speak anything about in-class initialization, but C.48 does, our check enforcing C.49 must not suggest a fixit the the check enforcing C.48 fixes again. We should not force the user to run Clang-Tidy in multiple passes. Maybe we can mention the numbers of the rules in the error messages to make them clearer. |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) |
Agreed, but it's also a different example from the one I posted where the behavior is mysterious.
Agreed, but the checks are also independent of one another (e.g., I can run one but not the other) and that use case needs to be considered as well as the combined use case. I personally don't use the C++ Core Guidelines and so I'm not as well-versed in their nuances -- I'm happy to let someone more familiar with the standard sway my opinion, but this smells a bit like the two rules are less orthogonal than the rule authors may believe. Both rules say to prefer something regarding member initialization, but neither rule says what the priority ordering is for those preferences when either approach works. Is it implicitly assumed that lower-numbered rules take precedence and so we should prefer in-class initialization over member initialization over assignment (that would seem defensible, but isn't stated anywhere in the rules)? |
Just my 2 cents, but would it not be wise to introduce a test case that runs the 2 aforementioned checks together demonstrating how they interact with each other. This will also force compliance if either check is updated down the line.
I can do that, and I will do it. However, all checks work on the original files, thus no interaction is expected.
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) |
The example in C.48 is the result of the fixit of this check for C.49 if I do not take C.48 in account. Thus without considering C.48 the check would suggest fixits which is bad according to C.48. This means that if the user accepts these fixits and runs Clang-Tidy again then the other check marks these fixits as bad and suggests another fixit for them. The big question is that which one is the majority? Users who wish to conform to all the core guidelines or users who accept C.49 but not C.48. My wild guess is the first one. So in my opinion the biggest problem is that if we suggest a fixit by one check considered as bad by another one than the problem that we suggest different fixits for initializations in the default constructors than in other constructors. Of course, the ideal solution would be to define the execution order of the checks and every check should work on a code fixed by the previous check. I do not think this would ever happen. It could also help if I could examine in this check whether the other one is enabled. However, I do not think this is possible either. |
clang-tools-extra/test/clang-tidy/checkers/cppcoreguidelines-prefer-member-initializer-assignment.cpp | ||
---|---|---|
29 ↗ | (On Diff #266277) |
Yup, and that's the situation I'm worried about.
My experience has been that users frequently disable individual warning flags and checks, but as a reaction to a poor-quality diagnostic in the project, and so they usually leave the entire group of checks enabled at first. So I'd say both are likely situations.
This is possible, but it's something we should really stop and think about before we decide to open the floodgates because I can imagine the testing matrix for these sort of interactions getting out of control. e.g., when this check is enabled, but not if it's spelled using that alias, then this other check behaves differently, but it may depend on options set for the first check, etc. But you can use ClangTidyContext::isCheckEnabled() to test this. |
Since no better idea cam to my mind, in this version I check whether modernize-use-default-member-init is enabled. If it is, then we issue a warning and suggest a fix that uses default member initializer. We also take into account the option for that checker whether we should use brackets or assignment. This checker has no options now. If the other checker is not enabled, we always warn and suggest fix to use constructor member initializer.
While I don't feel awesome about this approach, I don't have a better idea either. Unless @alexfh, @gribozavr2, or one of the other reviewers has concerns, this LGTM. Thank you for your patience while we thought our way through this!
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
104 | OptionsView::get(...) is specialized for bools, so you can just pass false as the second argument and negate the need to compare against zero |
So i've just reverted this in rGebf496d805521b53022a351f35854de977fee844.
@aaron.ballman @baloghadamsoftware how's QC going on nowadays here?
Was this evaluated on anything other than it's tests?
It appears to be either unacceptably slow, or subtly broken, because it takes at least 100x time more than all of the other clang-tidy checks enabled, and e.g.
clang-tidy-12 --checks="-*,cppcoreguidelines-prefer-member-initializer" -p . /repositories/llvm-project/llvm/lib/Transforms/InstCombine/InstructionCombining.cpp
never finishes (minutes and counting).
Surely. After I commit a patch, lots of buildbots verify it. They passed so far.
It appears to be either unacceptably slow, or subtly broken, because it takes at least 100x time more than all of the other clang-tidy checks enabled, and e.g.
clang-tidy-12 --checks="-*,cppcoreguidelines-prefer-member-initializer" -p . /repositories/llvm-project/llvm/lib/Transforms/InstCombine/InstructionCombining.cpp
never finishes (minutes and counting).
I see nothing there that could be slow, thus this is probably some hang. I will investigate it.
@baloghadamsoftware, i think you understand that wasn't the question.
It appears to be either unacceptably slow, or subtly broken, because it takes at least 100x time more than all of the other clang-tidy checks enabled, and e.g.
clang-tidy-12 --checks="-*,cppcoreguidelines-prefer-member-initializer" -p . /repositories/llvm-project/llvm/lib/Transforms/InstCombine/InstructionCombining.cpp
never finishes (minutes and counting).I see nothing there that could be slow, thus this is probably some hang. I will investigate it.
This feedback is a bit terse and not very constructive. FWIW, we don't typically ask patch authors to run their patch over a large corpus of code unless a reviewer expects there to be a performance concern and asks explicitly. Given that this checks constructor bodies, there was no obvious reason to ask for that here. Also, I can't recall a time when we expected a patch reviewer to do that work. I appreciate that you noticed an issue and reverted so we could investigate, but when reporting an issue like this, please try to keep in mind that we're all in the same community trying to make a great product.
I found the problem: hasBody() always returns true if the function has somewhere a body. This means that also the hasBody matcher matches forward declarations. However, I only need the definition. This far I could not find any method to achieve that: isDefinition() also returns true for forward declarations, as well as comparing with the CanonicalDecl. I am looking further...
OK, I reversed the matcher expression now, it seems to work, but I will check it on the LLVM/Clang codebase first.
Minor inline comments.
Could you please add a test case where the constructor is defined out of line? So far in every test, the constructor body was inside the class. Something like:
struct S { int i, j; S(); }; S::S() { i = 1; j = 2; }
I'm also particularly interested in the case when the constructor is implemented in its "own" translation unit. Where will the initialiser be moved in that case? Into the header, where the record is defined, still?
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
21–23 | It was you who figured out in D85728 that isa has changed recently and is now variadic. Then this function can be simplified with the variadic version. | |
39–41 | Ditto. | |
126 | ||
clang-tools-extra/docs/clang-tidy/checks/cppcoreguidelines-prefer-member-initializer.rst | ||
6 |
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
44 | @baloghadamsoftware The original comment by Aaron is left unmarked as "done" here. |
Prerequeisite patch is committed, the check is tested now on the LLVM Project. @lebedev.ri, @aaron.ballman can I recommit it?
clang-tools-extra/clang-tidy/cppcoreguidelines/PreferMemberInitializerCheck.cpp | ||
---|---|---|
185 | The following example generates invalid fixes, if modernize-use-default-member-init check isn't enabled, because Ctor->getNumCtorInitializers() returns 1. class Example { public: Example() { a = 0; }; int a; std::string string; }; |
Please keep this list sorted alphabetically.