[clang] [InstallAPI] Simplify & improve symbol printing for diagnostics (PR #85894)
Cyndy Ishida via cfe-commits
cfe-commits at lists.llvm.org
Tue Mar 19 21:47:16 PDT 2024
https://github.com/cyndyishida created https://github.com/llvm/llvm-project/pull/85894
* Defer mangling of symbols until an error is ready to report
* Pass around fewer parameters when reporting
>From f172f1c95c7011af4216b9e412fbb305d4f37e95 Mon Sep 17 00:00:00 2001
From: Cyndy Ishida <cyndy_ishida at apple.com>
Date: Tue, 19 Mar 2024 21:32:29 -0700
Subject: [PATCH] [InstallAPI] Simplify & improve symbol printing for
diagnostics
* Defer mangling of symbols until error is ready to report.
* Pass around less parameters when reporting
---
.../include/clang/InstallAPI/DylibVerifier.h | 4 +
clang/include/clang/InstallAPI/MachO.h | 2 +
clang/lib/InstallAPI/DylibVerifier.cpp | 134 +++++++++---------
clang/test/InstallAPI/diagnostics-cpp.test | 4 +-
4 files changed, 73 insertions(+), 71 deletions(-)
diff --git a/clang/include/clang/InstallAPI/DylibVerifier.h b/clang/include/clang/InstallAPI/DylibVerifier.h
index 8269715c7f2345..bbfa8711313e47 100644
--- a/clang/include/clang/InstallAPI/DylibVerifier.h
+++ b/clang/include/clang/InstallAPI/DylibVerifier.h
@@ -128,6 +128,10 @@ class DylibVerifier {
/// Find matching dylib slice for target triple that is being parsed.
void assignSlice(const Target &T);
+ /// Gather annotations for symbol for error reporting.
+ std::string getAnnotatedName(const Record *R, SymbolContext &SymCtx,
+ bool ValidSourceLoc = true);
+
// Symbols in dylib.
llvm::MachO::Records Dylib;
diff --git a/clang/include/clang/InstallAPI/MachO.h b/clang/include/clang/InstallAPI/MachO.h
index a77766511fa3e5..f0dea8bbd24ccd 100644
--- a/clang/include/clang/InstallAPI/MachO.h
+++ b/clang/include/clang/InstallAPI/MachO.h
@@ -26,11 +26,13 @@
using SymbolFlags = llvm::MachO::SymbolFlags;
using RecordLinkage = llvm::MachO::RecordLinkage;
using Record = llvm::MachO::Record;
+using EncodeKind = llvm::MachO::EncodeKind;
using GlobalRecord = llvm::MachO::GlobalRecord;
using ObjCContainerRecord = llvm::MachO::ObjCContainerRecord;
using ObjCInterfaceRecord = llvm::MachO::ObjCInterfaceRecord;
using ObjCCategoryRecord = llvm::MachO::ObjCCategoryRecord;
using ObjCIVarRecord = llvm::MachO::ObjCIVarRecord;
+using ObjCIFSymbolKind = llvm::MachO::ObjCIFSymbolKind;
using Records = llvm::MachO::Records;
using RecordsSlice = llvm::MachO::RecordsSlice;
using BinaryAttrs = llvm::MachO::RecordsSlice::BinaryAttrs;
diff --git a/clang/lib/InstallAPI/DylibVerifier.cpp b/clang/lib/InstallAPI/DylibVerifier.cpp
index 700763b3fee0db..24e0d0addf2f46 100644
--- a/clang/lib/InstallAPI/DylibVerifier.cpp
+++ b/clang/lib/InstallAPI/DylibVerifier.cpp
@@ -10,9 +10,6 @@ namespace installapi {
/// Metadata stored about a mapping of a declaration to a symbol.
struct DylibVerifier::SymbolContext {
- // Name to use for printing in diagnostics.
- std::string PrettyPrintName{""};
-
// Name to use for all querying and verification
// purposes.
std::string SymbolName{""};
@@ -30,11 +27,35 @@ struct DylibVerifier::SymbolContext {
bool Inlined = false;
};
-static std::string
-getAnnotatedName(const Record *R, EncodeKind Kind, StringRef Name,
- bool ValidSourceLoc = true,
- ObjCIFSymbolKind ObjCIF = ObjCIFSymbolKind::None) {
- assert(!Name.empty() && "Need symbol name for printing");
+static bool isCppMangled(StringRef Name) {
+ // InstallAPI currently only supports itanium manglings.
+ return (Name.starts_with("_Z") || Name.starts_with("__Z") ||
+ Name.starts_with("___Z"));
+}
+
+static std::string demangle(StringRef Name) {
+ // InstallAPI currently only supports itanium manglings.
+ if (!isCppMangled(Name))
+ return Name.str();
+ char *Result = llvm::itaniumDemangle(Name);
+ if (!Result)
+ return Name.str();
+
+ std::string Demangled(Result);
+ free(Result);
+ return Demangled;
+}
+
+std::string DylibVerifier::getAnnotatedName(const Record *R,
+ SymbolContext &SymCtx,
+ bool ValidSourceLoc) {
+ assert(!SymCtx.SymbolName.empty() && "Expected symbol name");
+
+ const StringRef SymbolName = SymCtx.SymbolName;
+ std::string PrettyName =
+ (Demangle && (SymCtx.Kind == EncodeKind::GlobalSymbol))
+ ? demangle(SymbolName)
+ : SymbolName.str();
std::string Annotation;
if (R->isWeakDefined())
@@ -45,15 +66,16 @@ getAnnotatedName(const Record *R, EncodeKind Kind, StringRef Name,
Annotation += "(tlv) ";
// Check if symbol represents only part of a @interface declaration.
- const bool IsAnnotatedObjCClass = ((ObjCIF != ObjCIFSymbolKind::None) &&
- (ObjCIF <= ObjCIFSymbolKind::EHType));
+ const bool IsAnnotatedObjCClass =
+ ((SymCtx.ObjCIFKind != ObjCIFSymbolKind::None) &&
+ (SymCtx.ObjCIFKind <= ObjCIFSymbolKind::EHType));
if (IsAnnotatedObjCClass) {
- if (ObjCIF == ObjCIFSymbolKind::EHType)
+ if (SymCtx.ObjCIFKind == ObjCIFSymbolKind::EHType)
Annotation += "Exception Type of ";
- if (ObjCIF == ObjCIFSymbolKind::MetaClass)
+ if (SymCtx.ObjCIFKind == ObjCIFSymbolKind::MetaClass)
Annotation += "Metaclass of ";
- if (ObjCIF == ObjCIFSymbolKind::Class)
+ if (SymCtx.ObjCIFKind == ObjCIFSymbolKind::Class)
Annotation += "Class of ";
}
@@ -61,42 +83,30 @@ getAnnotatedName(const Record *R, EncodeKind Kind, StringRef Name,
// tied to it. This can only ever happen when the location has to come from
// debug info.
if (ValidSourceLoc) {
- if ((Kind == EncodeKind::GlobalSymbol) && Name.starts_with("_"))
- return Annotation + Name.drop_front(1).str();
- return Annotation + Name.str();
+ StringRef PrettyNameRef(PrettyName);
+ if ((SymCtx.Kind == EncodeKind::GlobalSymbol) &&
+ !isCppMangled(SymbolName) && PrettyNameRef.starts_with("_"))
+ return Annotation + PrettyNameRef.drop_front(1).str();
+ return Annotation + PrettyName;
}
if (IsAnnotatedObjCClass)
- return Annotation + Name.str();
+ return Annotation + PrettyName;
- switch (Kind) {
+ switch (SymCtx.Kind) {
case EncodeKind::GlobalSymbol:
- return Annotation + Name.str();
+ return Annotation + PrettyName;
case EncodeKind::ObjectiveCInstanceVariable:
- return Annotation + "(ObjC IVar) " + Name.str();
+ return Annotation + "(ObjC IVar) " + PrettyName;
case EncodeKind::ObjectiveCClass:
- return Annotation + "(ObjC Class) " + Name.str();
+ return Annotation + "(ObjC Class) " + PrettyName;
case EncodeKind::ObjectiveCClassEHType:
- return Annotation + "(ObjC Class EH) " + Name.str();
+ return Annotation + "(ObjC Class EH) " + PrettyName;
}
llvm_unreachable("unexpected case for EncodeKind");
}
-static std::string demangle(StringRef Name) {
- // InstallAPI currently only supports itanium manglings.
- if (!(Name.starts_with("_Z") || Name.starts_with("__Z") ||
- Name.starts_with("___Z")))
- return Name.str();
- char *Result = llvm::itaniumDemangle(Name);
- if (!Result)
- return Name.str();
-
- std::string Demangled(Result);
- free(Result);
- return Demangled;
-}
-
static DylibVerifier::Result updateResult(const DylibVerifier::Result Prev,
const DylibVerifier::Result Curr) {
if (Prev == Curr)
@@ -193,19 +203,18 @@ bool DylibVerifier::compareObjCInterfaceSymbols(const Record *R,
// The decl represents a complete ObjCInterface, but the symbols in the
// dylib do not. Determine which symbol is missing. To keep older projects
// building, treat this as a warning.
- if (!DR->isExportedSymbol(ObjCIFSymbolKind::Class))
+ if (!DR->isExportedSymbol(ObjCIFSymbolKind::Class)) {
+ SymCtx.ObjCIFKind = ObjCIFSymbolKind::Class;
PrintDiagnostic(DR->getLinkageForSymbol(ObjCIFSymbolKind::Class), R,
- getAnnotatedName(R, SymCtx.Kind, SymCtx.PrettyPrintName,
- /*ValidSourceLoc=*/true,
- ObjCIFSymbolKind::Class),
+ getAnnotatedName(R, SymCtx),
/*PrintAsWarning=*/true);
-
- if (!DR->isExportedSymbol(ObjCIFSymbolKind::MetaClass))
+ }
+ if (!DR->isExportedSymbol(ObjCIFSymbolKind::MetaClass)) {
+ SymCtx.ObjCIFKind = ObjCIFSymbolKind::MetaClass;
PrintDiagnostic(DR->getLinkageForSymbol(ObjCIFSymbolKind::MetaClass), R,
- getAnnotatedName(R, SymCtx.Kind, SymCtx.PrettyPrintName,
- /*ValidSourceLoc=*/true,
- ObjCIFSymbolKind::MetaClass),
+ getAnnotatedName(R, SymCtx),
/*PrintAsWarning=*/true);
+ }
return true;
}
@@ -221,7 +230,7 @@ bool DylibVerifier::compareObjCInterfaceSymbols(const Record *R,
// At this point that means there was not a matching class symbol
// to represent the one discovered as a declaration.
PrintDiagnostic(DR->getLinkageForSymbol(SymCtx.ObjCIFKind), R,
- SymCtx.PrettyPrintName);
+ SymCtx.SymbolName);
return false;
}
@@ -234,7 +243,7 @@ DylibVerifier::Result DylibVerifier::compareVisibility(const Record *R,
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_library_missing_symbol)
- << SymCtx.PrettyPrintName;
+ << getAnnotatedName(R, SymCtx);
});
return Result::Invalid;
}
@@ -242,7 +251,7 @@ DylibVerifier::Result DylibVerifier::compareVisibility(const Record *R,
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_library_hidden_symbol)
- << SymCtx.PrettyPrintName;
+ << getAnnotatedName(R, SymCtx);
});
return Result::Invalid;
}
@@ -269,7 +278,7 @@ DylibVerifier::Result DylibVerifier::compareVisibility(const Record *R,
}
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(), ID)
- << SymCtx.PrettyPrintName;
+ << getAnnotatedName(R, SymCtx);
});
return Outcome;
}
@@ -293,14 +302,14 @@ DylibVerifier::Result DylibVerifier::compareAvailability(const Record *R,
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::warn_header_availability_mismatch)
- << SymCtx.PrettyPrintName << IsDeclAvailable << IsDeclAvailable;
+ << getAnnotatedName(R, SymCtx) << IsDeclAvailable << IsDeclAvailable;
});
return Result::Ignore;
case VerificationMode::Pedantic:
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_header_availability_mismatch)
- << SymCtx.PrettyPrintName << IsDeclAvailable << IsDeclAvailable;
+ << getAnnotatedName(R, SymCtx) << IsDeclAvailable << IsDeclAvailable;
});
return Result::Invalid;
case VerificationMode::ErrorsOnly:
@@ -313,15 +322,11 @@ DylibVerifier::Result DylibVerifier::compareAvailability(const Record *R,
bool DylibVerifier::compareSymbolFlags(const Record *R, SymbolContext &SymCtx,
const Record *DR) {
- std::string DisplayName =
- Demangle ? demangle(DR->getName()) : DR->getName().str();
-
if (DR->isThreadLocalValue() && !R->isThreadLocalValue()) {
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_dylib_symbol_flags_mismatch)
- << getAnnotatedName(DR, SymCtx.Kind, DisplayName)
- << DR->isThreadLocalValue();
+ << getAnnotatedName(DR, SymCtx) << DR->isThreadLocalValue();
});
return false;
}
@@ -329,7 +334,7 @@ bool DylibVerifier::compareSymbolFlags(const Record *R, SymbolContext &SymCtx,
Ctx.emitDiag([&]() {
SymCtx.FA->D->getLocation(),
Ctx.Diag->Report(diag::err_header_symbol_flags_mismatch)
- << SymCtx.PrettyPrintName << R->isThreadLocalValue();
+ << getAnnotatedName(DR, SymCtx) << R->isThreadLocalValue();
});
return false;
}
@@ -338,8 +343,7 @@ bool DylibVerifier::compareSymbolFlags(const Record *R, SymbolContext &SymCtx,
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_dylib_symbol_flags_mismatch)
- << getAnnotatedName(DR, SymCtx.Kind, DisplayName)
- << R->isWeakDefined();
+ << getAnnotatedName(DR, SymCtx) << R->isWeakDefined();
});
return false;
}
@@ -347,7 +351,7 @@ bool DylibVerifier::compareSymbolFlags(const Record *R, SymbolContext &SymCtx,
Ctx.emitDiag([&]() {
Ctx.Diag->Report(SymCtx.FA->D->getLocation(),
diag::err_header_symbol_flags_mismatch)
- << SymCtx.PrettyPrintName << R->isWeakDefined();
+ << getAnnotatedName(R, SymCtx) << R->isWeakDefined();
});
return false;
}
@@ -457,10 +461,7 @@ DylibVerifier::Result DylibVerifier::verify(ObjCIVarRecord *R,
std::string FullName =
ObjCIVarRecord::createScopedName(SuperClass, R->getName());
- SymbolContext SymCtx{
- getAnnotatedName(R, EncodeKind::ObjectiveCInstanceVariable,
- Demangle ? demangle(FullName) : FullName),
- FullName, EncodeKind::ObjectiveCInstanceVariable, FA};
+ SymbolContext SymCtx{FullName, EncodeKind::ObjectiveCInstanceVariable, FA};
return verifyImpl(R, SymCtx);
}
@@ -485,11 +486,8 @@ DylibVerifier::Result DylibVerifier::verify(ObjCInterfaceRecord *R,
SymCtx.SymbolName = R->getName();
SymCtx.ObjCIFKind = assignObjCIFSymbolKind(R);
- std::string DisplayName =
- Demangle ? demangle(SymCtx.SymbolName) : SymCtx.SymbolName;
SymCtx.Kind = R->hasExceptionAttribute() ? EncodeKind::ObjectiveCClassEHType
: EncodeKind::ObjectiveCClass;
- SymCtx.PrettyPrintName = getAnnotatedName(R, SymCtx.Kind, DisplayName);
SymCtx.FA = FA;
return verifyImpl(R, SymCtx);
@@ -504,8 +502,6 @@ DylibVerifier::Result DylibVerifier::verify(GlobalRecord *R,
SimpleSymbol Sym = parseSymbol(R->getName());
SymbolContext SymCtx;
SymCtx.SymbolName = Sym.Name;
- SymCtx.PrettyPrintName =
- getAnnotatedName(R, Sym.Kind, Demangle ? demangle(Sym.Name) : Sym.Name);
SymCtx.Kind = Sym.Kind;
SymCtx.FA = FA;
SymCtx.Inlined = R->isInlined();
diff --git a/clang/test/InstallAPI/diagnostics-cpp.test b/clang/test/InstallAPI/diagnostics-cpp.test
index 9319a7a61d4839..65888653750722 100644
--- a/clang/test/InstallAPI/diagnostics-cpp.test
+++ b/clang/test/InstallAPI/diagnostics-cpp.test
@@ -12,8 +12,8 @@
// RUN: --verify-mode=Pedantic -o %t/output.tbd --demangle 2> %t/errors.log
// RUN: FileCheck -input-file %t/errors.log %s
-CHECK: warning: violations found for arm64-apple-macos13
-CHECK: CPP.h:5:7: error: declaration has external linkage, but symbol has internal linkage in dynamic library 'vtable for Bar'
+CHECK: warning: violations found for arm64-apple-macos13
+CHECK: CPP.h:5:7: error: declaration has external linkage, but symbol has internal linkage in dynamic library 'vtable for Bar'
CHECK-NEXT: class Bar : Foo {
CHECK-NEXT: ^
CHECK-NEXT: CPP.h:5:7: error: declaration has external linkage, but symbol has internal linkage in dynamic library 'typeinfo for Bar'
More information about the cfe-commits
mailing list