[PATCH] D95771: [clang-tidy] fix modernize-loop-convert to retain needed array-like operator[]
Conrad Poelman via Phabricator via cfe-commits
cfe-commits at lists.llvm.org
Sun Jan 31 22:24:14 PST 2021
poelmanc created this revision.
poelmanc added reviewers: sammccall, hokein.
poelmanc added a project: clang-tools-extra.
Herald added a subscriber: xazax.hun.
poelmanc requested review of this revision.
Herald added a project: clang.
Herald added a subscriber: cfe-commits.
`modernize-loop-convert` handles //array-like// objects like vectors fairly well, but strips slightly too much information from the iteration expression by converting:
Vector<Vector<int>> X;
for (int J = 0; J < X[5].size(); ++J)
copyArg(X[5][J]);
to
Vector<Vector<int>> X;
for (int J : X) // should be for (int J : X[5])
copyArg(J);
The `[5]` is a call to `operator[]` and gets stripped by `LoopConvertCheck::getContainerString`. This patch fixes that and adds several test cases.
Repository:
rG LLVM Github Monorepo
https://reviews.llvm.org/D95771
Files:
clang-tools-extra/clang-tidy/modernize/LoopConvertCheck.cpp
clang-tools-extra/test/clang-tidy/checkers/modernize-loop-convert-multidimensional.cpp
Index: clang-tools-extra/test/clang-tidy/checkers/modernize-loop-convert-multidimensional.cpp
===================================================================
--- /dev/null
+++ clang-tools-extra/test/clang-tidy/checkers/modernize-loop-convert-multidimensional.cpp
@@ -0,0 +1,70 @@
+// RUN: %check_clang_tidy %s modernize-loop-convert %t
+
+template <class T>
+struct Vector {
+ using iterator = T*;
+ unsigned size() const;
+ const T &operator[](int) const;
+ T &operator[](int);
+ T *begin();
+ T *end();
+ const T *begin() const;
+ const T *end() const;
+};
+
+template <typename T>
+void copyArg(T);
+
+class TestArrayOfVector {
+ Vector<int> W[2];
+
+ void foo() const {
+ for (int I = 0; I < W[0].size(); ++I) {
+ if (W[0][I])
+ copyArg(W[0][I]);
+ }
+ // CHECK-MESSAGES: :[[@LINE-4]]:5: warning: use range-based for loop
+ // CHECK-FIXES: for (int I : W[0]) {
+ // CHECK-FIXES-NEXT: if (I)
+ // CHECK-FIXES-NEXT: copyArg(I);
+ }
+};
+
+class TestVectorOfVector {
+ Vector<Vector<int>> X;
+
+ void foo() const {
+ for (int J = 0; J < X[0].size(); ++J) {
+ if (X[0][J])
+ copyArg(X[0][J]);
+ }
+ // CHECK-MESSAGES: :[[@LINE-4]]:5: warning: use range-based for loop
+ // CHECK-FIXES: for (int J : X[0]) {
+ // CHECK-FIXES-NEXT: if (J)
+ // CHECK-FIXES-NEXT: copyArg(J);
+ }
+};
+
+void testVectorOfVectorOfVector() {
+ Vector<Vector<Vector<int>>> Y;
+ for (int J = 0; J < Y[3].size(); ++J) {
+ if (Y[3][J][7])
+ copyArg(Y[3][J][8]);
+ }
+ // CHECK-MESSAGES: :[[@LINE-4]]:3: warning: use range-based for loop
+ // CHECK-FIXES: for (auto & J : Y[3]) {
+ // CHECK-FIXES-NEXT: if (J[7])
+ // CHECK-FIXES-NEXT: copyArg(J[8]);
+};
+
+void testVectorOfVectorIterator() {
+ Vector<Vector<int>> Z;
+ for (Vector<int>::iterator it = Z[4].begin(); it != Z[4].end(); ++it) {
+ if (*it)
+ copyArg(*it);
+ }
+ // CHECK-MESSAGES: :[[@LINE-4]]:3: warning: use range-based for loop
+ // CHECK-FIXES: for (int & it : Z[4]) {
+ // CHECK-FIXES-NEXT: if (it)
+ // CHECK-FIXES-NEXT: copyArg(it);
+};
Index: clang-tools-extra/clang-tidy/modernize/LoopConvertCheck.cpp
===================================================================
--- clang-tools-extra/clang-tidy/modernize/LoopConvertCheck.cpp
+++ clang-tools-extra/clang-tidy/modernize/LoopConvertCheck.cpp
@@ -722,10 +722,11 @@
if (isa<CXXThisExpr>(ContainerExpr)) {
ContainerString = "this";
} else {
- // For CXXOperatorCallExpr (e.g. vector_ptr->size()), its first argument is
- // the class object (vector_ptr) we are targeting.
+ // For CXXOperatorCallExpr such as vector_ptr->size() we want the class
+ // object vector_ptr, but for vector[2] we need the whole expression.
if (const auto* E = dyn_cast<CXXOperatorCallExpr>(ContainerExpr))
- ContainerExpr = E->getArg(0);
+ if ( E->getOperator() != OO_Subscript )
+ ContainerExpr = E->getArg(0);
ContainerString =
getStringFromRange(Context->getSourceManager(), Context->getLangOpts(),
ContainerExpr->getSourceRange());
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D95771.320394.patch
Type: text/x-patch
Size: 3108 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/cfe-commits/attachments/20210201/088a4fd0/attachment.bin>
More information about the cfe-commits
mailing list