[PATCH] D83513: [AST][ObjC] Fix crash when printing invalid objc categories
David Goldman via Phabricator via cfe-commits
cfe-commits at lists.llvm.org
Thu Jul 9 14:37:27 PDT 2020
dgoldman created this revision.
dgoldman added reviewers: sammccall, gribozavr.
Herald added a project: clang.
Herald added a subscriber: cfe-commits.
If no valid interface definition was found previously we would crash.
With this change instead we just print `<<error-type>>` in place
of the NULL interface. In the future this could be improved by
saving the invalid interface's name and using that.
Repository:
rG LLVM Github Monorepo
https://reviews.llvm.org/D83513
Files:
clang/lib/AST/DeclPrinter.cpp
clang/unittests/AST/DeclPrinterTest.cpp
Index: clang/unittests/AST/DeclPrinterTest.cpp
===================================================================
--- clang/unittests/AST/DeclPrinterTest.cpp
+++ clang/unittests/AST/DeclPrinterTest.cpp
@@ -76,14 +76,16 @@
PrintedDeclMatches(StringRef Code, const std::vector<std::string> &Args,
const DeclarationMatcher &NodeMatch,
StringRef ExpectedPrinted, StringRef FileName,
- PrintingPolicyModifier PolicyModifier = nullptr) {
+ PrintingPolicyModifier PolicyModifier = nullptr,
+ bool AllowError = false) {
PrintMatch Printer(PolicyModifier);
MatchFinder Finder;
Finder.addMatcher(NodeMatch, &Printer);
std::unique_ptr<FrontendActionFactory> Factory(
newFrontendActionFactory(&Finder));
- if (!runToolOnCodeWithArgs(Factory->create(), Code, Args, FileName))
+ if (!runToolOnCodeWithArgs(Factory->create(), Code, Args, FileName) &&
+ !AllowError)
return testing::AssertionFailure()
<< "Parsing error in \"" << Code.str() << "\"";
@@ -170,16 +172,12 @@
"input.cc");
}
-::testing::AssertionResult PrintedDeclObjCMatches(
- StringRef Code,
- const DeclarationMatcher &NodeMatch,
- StringRef ExpectedPrinted) {
+::testing::AssertionResult
+PrintedDeclObjCMatches(StringRef Code, const DeclarationMatcher &NodeMatch,
+ StringRef ExpectedPrinted, bool AllowError = false) {
std::vector<std::string> Args(1, "");
- return PrintedDeclMatches(Code,
- Args,
- NodeMatch,
- ExpectedPrinted,
- "input.m");
+ return PrintedDeclMatches(Code, Args, NodeMatch, ExpectedPrinted, "input.m",
+ /*PolicyModifier*/ nullptr, AllowError);
}
} // unnamed namespace
@@ -1321,3 +1319,17 @@
namedDecl(hasName("P1")).bind("id"),
"@protocol P1<P2>\n at end"));
}
+
+TEST(DeclPrinter, TestObjCCategoryInvalidInterface) {
+ ASSERT_TRUE(PrintedDeclObjCMatches(
+ "@interface I (Extension) @end",
+ namedDecl(hasName("Extension")).bind("id"),
+ "@interface <<error-type>>(Extension)\n at end", /*AllowError*/ true));
+}
+
+TEST(DeclPrinter, TestObjCCategoryImplInvalidInterface) {
+ ASSERT_TRUE(PrintedDeclObjCMatches(
+ "@implementation I (Extension) @end",
+ namedDecl(hasName("Extension")).bind("id"),
+ "@implementation <<error-type>>(Extension)\n at end", /*AllowError*/ true));
+}
Index: clang/lib/AST/DeclPrinter.cpp
===================================================================
--- clang/lib/AST/DeclPrinter.cpp
+++ clang/lib/AST/DeclPrinter.cpp
@@ -1374,7 +1374,12 @@
}
void DeclPrinter::VisitObjCCategoryImplDecl(ObjCCategoryImplDecl *PID) {
- Out << "@implementation " << *PID->getClassInterface() << '(' << *PID <<")\n";
+ Out << "@implementation ";
+ if (const auto *CID = PID->getClassInterface())
+ Out << *CID;
+ else
+ Out << "<<error-type>>";
+ Out << '(' << *PID << ")\n";
VisitDeclContext(PID, false);
Out << "@end";
@@ -1382,7 +1387,11 @@
}
void DeclPrinter::VisitObjCCategoryDecl(ObjCCategoryDecl *PID) {
- Out << "@interface " << *PID->getClassInterface();
+ Out << "@interface ";
+ if (const auto *CID = PID->getClassInterface())
+ Out << *CID;
+ else
+ Out << "<<error-type>>";
if (auto TypeParams = PID->getTypeParamList()) {
PrintObjCTypeParams(TypeParams);
}
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D83513.276841.patch
Type: text/x-patch
Size: 3581 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/cfe-commits/attachments/20200709/144fe69b/attachment-0001.bin>
More information about the cfe-commits
mailing list