<div dir="ltr"><br><div class="gmail_extra"><br><div class="gmail_quote">On Thu, Apr 9, 2015 at 10:51 AM, Samuel Benzaquen <span dir="ltr"><<a href="mailto:sbenza@google.com" target="_blank">sbenza@google.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Author: sbenza<br>
Date: Thu Apr 9 12:51:01 2015<br>
New Revision: 234512<br>
<br>
URL: <a href="http://llvm.org/viewvc/llvm-project?rev=234512&view=rev" target="_blank">http://llvm.org/viewvc/llvm-project?rev=234512&view=rev</a><br>
Log:<br>
[clang-tidy] Ignore expressions with incompatible deleters.<br>
<br>
Summary:<br>
Do not warn on .reset(.release()) expressions if the deleters are not<br>
compatible.<br>
Using plain assignment will probably not work.<br></blockquote><div><br>Should we have a separate check for that, then? It seems error prone... <br> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br>
Reviewers: klimek<br>
<br>
Subscribers: curdeius, cfe-commits<br>
<br>
Differential Revision: <a href="http://reviews.llvm.org/D8422" target="_blank">http://reviews.llvm.org/D8422</a><br>
<br>
Modified:<br>
clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp<br>
clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp<br>
<br>
Modified: clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp?rev=234512&r1=234511&r2=234512&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp?rev=234512&r1=234511&r2=234512&view=diff</a><br>
==============================================================================<br>
--- clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp (original)<br>
+++ clang-tools-extra/trunk/clang-tidy/misc/UniqueptrResetReleaseCheck.cpp Thu Apr 9 12:51:01 2015<br>
@@ -22,17 +22,75 @@ void UniqueptrResetReleaseCheck::registe<br>
memberCallExpr(<br>
on(expr().bind("left")), callee(memberExpr().bind("reset_member")),<br>
callee(methodDecl(hasName("reset"),<br>
- ofClass(hasName("::std::unique_ptr")))),<br>
+ ofClass(recordDecl(hasName("::std::unique_ptr"),<br>
+ decl().bind("left_class"))))),<br>
has(memberCallExpr(<br>
on(expr().bind("right")),<br>
callee(memberExpr().bind("release_member")),<br>
- callee(methodDecl(hasName("release"),<br>
- ofClass(hasName("::std::unique_ptr")))))))<br>
+ callee(methodDecl(<br>
+ hasName("release"),<br>
+ ofClass(recordDecl(hasName("::std::unique_ptr"),<br>
+ decl().bind("right_class"))))))))<br>
.bind("reset_call"),<br>
this);<br>
}<br>
<br>
+namespace {<br>
+const Type *getDeleterForUniquePtr(const MatchFinder::MatchResult &Result,<br>
+ StringRef ID) {<br>
+ const auto *Class =<br>
+ Result.Nodes.getNodeAs<ClassTemplateSpecializationDecl>(ID);<br>
+ if (!Class)<br>
+ return nullptr;<br>
+ auto DeleterArgument = Class->getTemplateArgs()[1];<br>
+ if (DeleterArgument.getKind() != TemplateArgument::Type)<br>
+ return nullptr;<br>
+ return DeleterArgument.getAsType().getTypePtr();<br>
+}<br>
+<br>
+bool areDeletersCompatible(const MatchFinder::MatchResult &Result) {<br>
+ const Type *LeftDeleterType = getDeleterForUniquePtr(Result, "left_class");<br>
+ const Type *RightDeleterType = getDeleterForUniquePtr(Result, "right_class");<br>
+<br>
+ if (LeftDeleterType->getUnqualifiedDesugaredType() ==<br>
+ RightDeleterType->getUnqualifiedDesugaredType()) {<br>
+ // Same type. We assume they are compatible.<br>
+ // This check handles the case where the deleters are function pointers.<br>
+ return true;<br>
+ }<br>
+<br>
+ const CXXRecordDecl *LeftDeleter = LeftDeleterType->getAsCXXRecordDecl();<br>
+ const CXXRecordDecl *RightDeleter = RightDeleterType->getAsCXXRecordDecl();<br>
+ if (!LeftDeleter || !RightDeleter)<br>
+ return false;<br>
+<br>
+ if (LeftDeleter->getCanonicalDecl() == RightDeleter->getCanonicalDecl()) {<br>
+ // Same class. We assume they are compatible.<br>
+ return true;<br>
+ }<br>
+<br>
+ const auto *LeftAsTemplate =<br>
+ dyn_cast<ClassTemplateSpecializationDecl>(LeftDeleter);<br>
+ const auto *RightAsTemplate =<br>
+ dyn_cast<ClassTemplateSpecializationDecl>(RightDeleter);<br>
+ if (LeftAsTemplate && RightAsTemplate &&<br>
+ LeftAsTemplate->getSpecializedTemplate() ==<br>
+ RightAsTemplate->getSpecializedTemplate()) {<br>
+ // They are different instantiations of the same template. We assume they<br>
+ // are compatible.<br>
+ // This handles things like std::default_delete<Base> vs.<br>
+ // std::default_delete<Derived>.<br>
+ return true;<br>
+ }<br>
+ return false;<br>
+}<br>
+<br>
+} // namespace<br>
+<br>
void UniqueptrResetReleaseCheck::check(const MatchFinder::MatchResult &Result) {<br>
+ if (!areDeletersCompatible(Result))<br>
+ return;<br>
+<br>
const auto *ResetMember = Result.Nodes.getNodeAs<MemberExpr>("reset_member");<br>
const auto *ReleaseMember =<br>
Result.Nodes.getNodeAs<MemberExpr>("release_member");<br>
<br>
Modified: clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp?rev=234512&r1=234511&r2=234512&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp?rev=234512&r1=234511&r2=234512&view=diff</a><br>
==============================================================================<br>
--- clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp (original)<br>
+++ clang-tools-extra/trunk/test/clang-tidy/misc-uniqueptr-reset-release.cpp Thu Apr 9 12:51:01 2015<br>
@@ -2,12 +2,16 @@<br>
// REQUIRES: shell<br>
<br>
namespace std {<br>
+<br>
template <typename T><br>
+struct default_delete {};<br>
+<br>
+template <typename T, class Deleter = std::default_delete<T>><br>
struct unique_ptr {<br>
- unique_ptr<T>();<br>
- explicit unique_ptr<T>(T *);<br>
- template <typename U><br>
- unique_ptr<T>(unique_ptr<U> &&);<br>
+ unique_ptr();<br>
+ explicit unique_ptr(T *);<br>
+ template <typename U, typename E><br>
+ unique_ptr(unique_ptr<U, E> &&);<br>
void reset(T *);<br>
T *release();<br>
};<br>
@@ -20,6 +24,9 @@ std::unique_ptr<Foo> Create();<br>
std::unique_ptr<Foo> &Look();<br>
std::unique_ptr<Foo> *Get();<br>
<br>
+using FooFunc = void (*)(Foo *);<br>
+using BarFunc = void (*)(Bar *);<br>
+<br>
void f() {<br>
std::unique_ptr<Foo> a, b;<br>
std::unique_ptr<Bar> c;<br>
@@ -44,5 +51,20 @@ void f() {<br>
Get()->reset(Get()->release());<br>
// CHECK-MESSAGES: :[[@LINE-1]]:10: warning: prefer ptr1 = std::move(ptr2)<br>
// CHECK-FIXES: *Get() = std::move(*Get());<br>
+<br>
+ std::unique_ptr<Bar, FooFunc> func_a, func_b;<br>
+ func_a.reset(func_b.release());<br>
+ // CHECK-MESSAGES: :[[@LINE-1]]:10: warning: prefer ptr1 = std::move(ptr2)<br>
+ // CHECK-FIXES: func_a = std::move(func_b);<br>
}<br>
<br>
+void negatives() {<br>
+ std::unique_ptr<Foo> src;<br>
+ struct OtherDeleter {};<br>
+ std::unique_ptr<Foo, OtherDeleter> dest;<br>
+ dest.reset(src.release());<br>
+<br>
+ std::unique_ptr<Bar, FooFunc> func_a;<br>
+ std::unique_ptr<Bar, BarFunc> func_b;<br>
+ func_a.reset(func_b.release());<br>
+}<br>
<br>
<br>
_______________________________________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@cs.uiuc.edu">cfe-commits@cs.uiuc.edu</a><br>
<a href="http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits</a><br>
</blockquote></div><br></div></div>