This checker is to warn about the performance overhead caused by concatenating strings using the operator+ instead of basic_string's append or operator+=. It is configurable. In strict mode it matches every instance of a supposed inefficient concatenation, in non-strict mode it matches only those which are inside a loop.
Details
Diff Detail
Event Timeline
Please mention this check in docs/ReleaseNotes.rst (in alphabetical order).
docs/clang-tidy/checks/performance-inefficient-string-addition.rst | ||
---|---|---|
8 ↗ | (On Diff #56998) | Please highlight operator+ and other class/methods with ``. Same for other places in documentation. |
21 ↗ | (On Diff #56998) | Please use constructor initialization instead of assignment. |
58 ↗ | (On Diff #56998) | Please add new line. |
test/clang-tidy/performance-inefficient-string-addition.cpp | ||
1 ↗ | (On Diff #56998) | Does it really need to be C++11? |
45 ↗ | (On Diff #56998) | Please add new line. |
drive-by, some nits.
clang-tidy/performance/InefficientStringAdditionCheck.cpp | ||
---|---|---|
31 ↗ | (On Diff #56998) | nits: I kind of prefer : const auto BasicStringType here and below. |
74 ↗ | (On Diff #56998) | nits: remove this line |
docs/clang-tidy/checks/performance-inefficient-string-addition.rst | ||
4 ↗ | (On Diff #56998) | line with "===" should be the same length than title. |
11 ↗ | (On Diff #56998) | nits: delete one blank line |
19 ↗ | (On Diff #56998) | remove extra blank line (only one) |
45 ↗ | (On Diff #56998) | nits: spaces between operator "+" |
docs/ReleaseNotes.rst | ||
---|---|---|
191 | Please highlight operator+ and operator+= with ``. |
clang-tidy/performance/InefficientStringAdditionCheck.cpp | ||
---|---|---|
35 ↗ | (On Diff #57294) | you should swap the two parameters hasOverloadedOperatorName("+"), hasAnyArgument(...) matching 'hasOverloadedOperatorName' is less expensive |
41 ↗ | (On Diff #57294) | this is not indented correctly. % clang-format -style=file <your-file> -i |
47 ↗ | (On Diff #57294) | space after "," |
54 ↗ | (On Diff #57294) | You forgot a comment here? |
Please create a diff with the full context: http://llvm.org/docs/Phabricator.html#requesting-a-review-via-the-web-interface
clang-tidy/performance/InefficientStringAdditionCheck.cpp | ||
---|---|---|
84 ↗ | (On Diff #57346) | nit: Warning messages are not full sentences, so they should not start with a capital letter and should not end with a period. Please also change the comma to a semicolon ;. |
85 ↗ | (On Diff #57346) | std::basic_string::append is correct, but is an unnecessary detail. I'd suggest changing this to string::append or to just append(). |
clang-tidy/performance/InefficientStringAdditionCheck.cpp | ||
---|---|---|
28 ↗ | (On Diff #57894) | s/IsStrictMode/StrictMode/ |
31 ↗ | (On Diff #57894) | This check is only useful for C++. Please add if (!getLangOpts().CPlusPlus) return; |
84 ↗ | (On Diff #57894) | The root of inefficiency is in unnecessary allocations of temporary strings, so maybe we should note this in the warning message, e.g. "string concatenation results in allocation of unnecessary temporary strings; consider using 'operator+=' or 'string::append()' instead". |
clang-tidy/performance/InefficientStringAdditionCheck.h | ||
24 ↗ | (On Diff #57894) | Please remove the empty line. |
25 ↗ | (On Diff #57894) | s/Addition/Concatenation/? |
33 ↗ | (On Diff #57894) | const bool StrictMode; |
docs/clang-tidy/checks/performance-inefficient-string-addition.rst | ||
15 ↗ | (On Diff #57894) | Please enclose std::string, std::basic_string and other inline code snippets in double backquotes. |
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
32 | clang-format -style=file, please. | |
67 | Each additional matcher has a certain overhead, so it's usually beneficial to merge matchers, if they share common parts. In this case, matchers can be rearranged like this: Finder->addMatcher(exprWithCleanups( hasAncestor(anyOf(cxxForRangeStmt(), whileStmt(), forStmt())), anyOf(hasDescendant(AssingOperator), hasDescendant(PlusOperatorMatcher)))); However, the really serious problem with these matchers is the liberal usage of hasDescendant matcher, which traverses the whole subtree. Nested hasDescendant matchers are even worse. Note that the subtree of a statement can be rather large, especially if you consider lambdas. Apart from performance issues, hasDescendant and hasAncestor can yield unexpected results, in particular, when lambdas or local classes are in play. It's always better to use has, if you only need to match direct children and specifically match intermediate nodes, if the descendant you're interested in is always on a fixed depths. Please try running your check (and maybe a few others for comparison, you can use clang-tidy -enable-check-profile to get the run time) on a few large translation units to estimate its performance and measure the improvement after changing the code. | |
docs/clang-tidy/checks/performance-inefficient-string-concatenation.rst | ||
7 | The problem subheader is the only one here. It doesn't seem like it helps making the document more structured. | |
9 | s/is to warn/warns/ | |
21 | Please make the code snippets use LLVM formatting for consistency. In particular, add a space after for and remove the line break before the opening brace. | |
40 | Add a space before the opening brace. |
Removed the unnecessary hasDescendant calls and simplified the checker as suggested. Tested on LLVM codebase, with minor improvements in speed (~1%).
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
41 | s/Matcher// | |
48 | s/Assing/Assign/ | |
57 | Please avoid unnecessary negation by putting the positive branch first, this will make the logic slightly simpler. | |
67 |
|
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
67 |
|
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
51 |
| |
69 | For 1: if hasAncestor(anyOf(cxxForRangeStmt(), ...)) doesn't work, you can try hasAncestor(stmt(anyOf(cxxForRangeStmt(), ...)). In any case, repeated hasAncestor matchers don't make the check faster ;) For 2: exprWithCleanups and other implicit nodes can be skipped using the ignoringImplicit() matchers. Leave hasAncestor for the cases, where there can actually be arbitrary nodes in the path. Here you're matching some arbitrary intermediate node (exprWithCleanups) and then you're constraining its parents and its children. This doesn't make much sense and it's rather inefficient. Since you're interested in the operators in the first place, try rearranging the matcher like this: cxxOperatorCallExpr( anyOf(AssignOperator, PlusOperator), hasAncestor(stmt(anyOf(cxxForRangeStmt(), whileStmt(), forStmt())))) |
A few more nits.
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
51 | The allOf(declRefExpr(x), declRefExpr(y)) construct can be replaced with declRefExpr(x, y). | |
55 | Indentation is confusing - as though hasDeclaration is an argument of stmt. Is it a result of clang-format? | |
64 | hasAncestor is potentially more expensive, so it should go last. |
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
56 | Yes this is the result of clang-format. |
A couple of nits.
clang-tidy/performance/InefficientStringConcatenationCheck.cpp | ||
---|---|---|
52 | As noted earlier, there's no need to repeat declRefExpr. This should be: declRefExpr(BasicStringType, hasDeclaration(decl().bind("lhsStrT"))) | |
clang-tidy/performance/InefficientStringConcatenationCheck.h | ||
26 | Please format the file with clang-format -style=llvm. |
Sorry for the delay. I almost missed that there was an update to the patch. Please mark addressed comments as "Done". This way it's easier for me to track changes and it's easier for you not to miss review comments.
One more thing that this check lacks is fix-it hints, however, this can be added later.
clang-tidy/performance/InefficientStringConcatenationCheck.h | ||
---|---|---|
27 | The formatting is still off (specifically, indentation of public: and private: and the lack of an empty line before private:). |
Looks good. Thank you for working on this!
Do you need me to commit the patch for you?
btw obtaining commit access and commiting is very simple, so if you are planning to send us some more cool patches then I think you should get the commit access :)
I'm planning to submit more patches in the future, as I have time for them. So it wouldn't be in vain :)
Please format the file with clang-format -style=llvm.