From 295172ef51c6b9a73bc0fdcfd25f8c41ead9034a Mon Sep 17 00:00:00 2001 From: Arthur Eubanks Date: Thu, 10 Feb 2022 11:56:14 -0800 Subject: [PATCH] [OpaquePtr][LLParser] Automatically detect opaque pointers in .ll files This allows us to not have to specify -opaque-pointers when updating IR tests from typed pointers to opaque pointers. We detect opaque pointers in .ll files by looking for relevant tokens, either "ptr" or "*". Reviewed By: #opaque-pointers, nikic Differential Revision: https://reviews.llvm.org/D119482 --- llvm/include/llvm/AsmParser/LLLexer.h | 2 +- llvm/include/llvm/AsmParser/LLParser.h | 7 +++++-- llvm/include/llvm/IR/LLVMContext.h | 3 +++ llvm/lib/AsmParser/LLLexer.cpp | 5 ++++- llvm/lib/AsmParser/LLParser.cpp | 20 +++++++++++++++++++ llvm/lib/IR/LLVMContext.cpp | 4 ++++ llvm/lib/IR/LLVMContextImpl.cpp | 12 +++++++++-- llvm/lib/IR/LLVMContextImpl.h | 1 + .../ptr-outside-opaque-pointers-mode.ll | 2 +- .../opaque-ptr.ll} | 2 +- llvm/test/Other/mixed-opaque-ptrs-2.ll | 6 ++++++ llvm/test/Other/mixed-opaque-ptrs.ll | 6 ++++++ 12 files changed, 62 insertions(+), 8 deletions(-) rename llvm/test/{Other/force-opaque-ptrs-typed-dis.ll => Bitcode/opaque-ptr.ll} (63%) create mode 100644 llvm/test/Other/mixed-opaque-ptrs-2.ll create mode 100644 llvm/test/Other/mixed-opaque-ptrs.ll diff --git a/llvm/include/llvm/AsmParser/LLLexer.h b/llvm/include/llvm/AsmParser/LLLexer.h index c30165e4a97b..7bcb33f18768 100644 --- a/llvm/include/llvm/AsmParser/LLLexer.h +++ b/llvm/include/llvm/AsmParser/LLLexer.h @@ -37,7 +37,7 @@ namespace llvm { lltok::Kind CurKind; std::string StrVal; unsigned UIntVal; - Type *TyVal; + Type *TyVal = nullptr; APFloat APFloatVal; APSInt APSIntVal; diff --git a/llvm/include/llvm/AsmParser/LLParser.h b/llvm/include/llvm/AsmParser/LLParser.h index 7e72493232c0..8fa110c1d8c6 100644 --- a/llvm/include/llvm/AsmParser/LLParser.h +++ b/llvm/include/llvm/AsmParser/LLParser.h @@ -95,6 +95,8 @@ namespace llvm { typedef LLLexer::LocTy LocTy; private: LLVMContext &Context; + // Lexer to determine whether to use opaque pointers or not. + LLLexer OPLex; LLLexer Lex; // Module being parsed, null if we are only parsing summary index. Module *M; @@ -157,8 +159,9 @@ namespace llvm { LLParser(StringRef F, SourceMgr &SM, SMDiagnostic &Err, Module *M, ModuleSummaryIndex *Index, LLVMContext &Context, SlotMapping *Slots = nullptr) - : Context(Context), Lex(F, SM, Err, Context), M(M), Index(Index), - Slots(Slots), BlockAddressPFS(nullptr) {} + : Context(Context), OPLex(F, SM, Err, Context), + Lex(F, SM, Err, Context), M(M), Index(Index), Slots(Slots), + BlockAddressPFS(nullptr) {} bool Run( bool UpgradeDebugInfo, DataLayoutCallbackTy DataLayoutCallback = [](StringRef) { return None; }); diff --git a/llvm/include/llvm/IR/LLVMContext.h b/llvm/include/llvm/IR/LLVMContext.h index e804ce90ff9f..d80d54693258 100644 --- a/llvm/include/llvm/IR/LLVMContext.h +++ b/llvm/include/llvm/IR/LLVMContext.h @@ -305,6 +305,9 @@ public: /// LLVMContext is used by compilation. void setOptPassGate(OptPassGate&); + /// Whether we've decided on using opaque pointers or typed pointers yet. + bool hasSetOpaquePointersValue() const; + /// Enable opaque pointers. Can only be called before creating the first /// pointer type. void enableOpaquePointers() const; diff --git a/llvm/lib/AsmParser/LLLexer.cpp b/llvm/lib/AsmParser/LLLexer.cpp index 6048066a44fb..8aac5a66c623 100644 --- a/llvm/lib/AsmParser/LLLexer.cpp +++ b/llvm/lib/AsmParser/LLLexer.cpp @@ -860,7 +860,10 @@ lltok::Kind LLLexer::LexIdentifier() { TYPEKEYWORD("token", Type::getTokenTy(Context)); if (Keyword == "ptr") { - if (Context.supportsTypedPointers()) { + // enableOpaquePointers() must be called before creating any pointer types. + if (!Context.hasSetOpaquePointersValue()) { + Context.enableOpaquePointers(); + } else if (Context.supportsTypedPointers()) { Warning("ptr type is only supported in -opaque-pointers mode"); return lltok::Error; } diff --git a/llvm/lib/AsmParser/LLParser.cpp b/llvm/lib/AsmParser/LLParser.cpp index f4a838bbdec1..04e651484872 100644 --- a/llvm/lib/AsmParser/LLParser.cpp +++ b/llvm/lib/AsmParser/LLParser.cpp @@ -59,9 +59,29 @@ static std::string getTypeString(Type *T) { return Tmp.str(); } +static void setContextOpaquePointers(LLLexer &L, LLVMContext &C) { + while (true) { + lltok::Kind K = L.Lex(); + // LLLexer will set the opaque pointers option in LLVMContext if it sees an + // explicit "ptr". + if (K == lltok::star || K == lltok::Error || K == lltok::Eof || + isa_and_nonnull(L.getTyVal())) { + return; + } + } +} + /// Run: module ::= toplevelentity* bool LLParser::Run(bool UpgradeDebugInfo, DataLayoutCallbackTy DataLayoutCallback) { + // If we haven't decided on whether or not we're using opaque pointers, do a + // quick lex over the tokens to see if we explicitly construct any typed or + // opaque pointer types. + // Don't bail out on an error so we do the same work in the parsing below + // regardless of if --opaque-pointers is set. + if (!Context.hasSetOpaquePointersValue()) + setContextOpaquePointers(OPLex, Context); + // Prime the lexer. Lex.Lex(); diff --git a/llvm/lib/IR/LLVMContext.cpp b/llvm/lib/IR/LLVMContext.cpp index 4b39599555a6..f4e917cc8992 100644 --- a/llvm/lib/IR/LLVMContext.cpp +++ b/llvm/lib/IR/LLVMContext.cpp @@ -351,6 +351,10 @@ std::unique_ptr LLVMContext::getDiagnosticHandler() { return std::move(pImpl->DiagHandler); } +bool LLVMContext::hasSetOpaquePointersValue() const { + return pImpl->hasOpaquePointersValue(); +} + void LLVMContext::enableOpaquePointers() const { assert(pImpl->PointerTypes.empty() && pImpl->ASPointerTypes.empty() && "Must be called before creating any pointer types"); diff --git a/llvm/lib/IR/LLVMContextImpl.cpp b/llvm/lib/IR/LLVMContextImpl.cpp index 8f9530290459..401af7a4dfca 100644 --- a/llvm/lib/IR/LLVMContextImpl.cpp +++ b/llvm/lib/IR/LLVMContextImpl.cpp @@ -47,7 +47,11 @@ LLVMContextImpl::LLVMContextImpl(LLVMContext &C) X86_FP80Ty(C, Type::X86_FP80TyID), FP128Ty(C, Type::FP128TyID), PPC_FP128Ty(C, Type::PPC_FP128TyID), X86_MMXTy(C, Type::X86_MMXTyID), X86_AMXTy(C, Type::X86_AMXTyID), Int1Ty(C, 1), Int8Ty(C, 8), - Int16Ty(C, 16), Int32Ty(C, 32), Int64Ty(C, 64), Int128Ty(C, 128) {} + Int16Ty(C, 16), Int32Ty(C, 32), Int64Ty(C, 64), Int128Ty(C, 128) { + if (OpaquePointersCL.getNumOccurrences()) { + OpaquePointers = OpaquePointersCL; + } +} LLVMContextImpl::~LLVMContextImpl() { // NOTE: We need to delete the contents of OwnedModules, but Module's dtor @@ -245,9 +249,13 @@ void LLVMContextImpl::setOptPassGate(OptPassGate& OPG) { this->OPG = &OPG; } +bool LLVMContextImpl::hasOpaquePointersValue() { + return OpaquePointers.hasValue(); +} + bool LLVMContextImpl::getOpaquePointers() { if (LLVM_UNLIKELY(!(OpaquePointers.hasValue()))) - OpaquePointers = OpaquePointersCL; + OpaquePointers = false; return *OpaquePointers; } diff --git a/llvm/lib/IR/LLVMContextImpl.h b/llvm/lib/IR/LLVMContextImpl.h index 70242f4d8f20..b87d2853ad76 100644 --- a/llvm/lib/IR/LLVMContextImpl.h +++ b/llvm/lib/IR/LLVMContextImpl.h @@ -1555,6 +1555,7 @@ public: // TODO: clean up the following after we no longer support non-opaque pointer // types. bool getOpaquePointers(); + bool hasOpaquePointersValue(); void setOpaquePointers(bool OP); private: diff --git a/llvm/test/Assembler/ptr-outside-opaque-pointers-mode.ll b/llvm/test/Assembler/ptr-outside-opaque-pointers-mode.ll index bf11dd401ebd..e26e1a75c8c2 100644 --- a/llvm/test/Assembler/ptr-outside-opaque-pointers-mode.ll +++ b/llvm/test/Assembler/ptr-outside-opaque-pointers-mode.ll @@ -1,4 +1,4 @@ -; RUN: not llvm-as < %s -disable-output 2>&1 | FileCheck %s +; RUN: not llvm-as < %s -disable-output --opaque-pointers=0 2>&1 | FileCheck %s ; CHECK: warning: ptr type is only supported in -opaque-pointers mode ; CHECK: error: expected type diff --git a/llvm/test/Other/force-opaque-ptrs-typed-dis.ll b/llvm/test/Bitcode/opaque-ptr.ll similarity index 63% rename from llvm/test/Other/force-opaque-ptrs-typed-dis.ll rename to llvm/test/Bitcode/opaque-ptr.ll index e98be24bc7cb..68b1e821fe60 100644 --- a/llvm/test/Other/force-opaque-ptrs-typed-dis.ll +++ b/llvm/test/Bitcode/opaque-ptr.ll @@ -1,4 +1,4 @@ -; RUN: llvm-as --opaque-pointers < %s | not llvm-dis 2>&1 | FileCheck %s +; RUN: llvm-as --opaque-pointers < %s | not llvm-dis --opaque-pointers=0 2>&1 | FileCheck %s ; CHECK: error: Opaque pointers are only supported in -opaque-pointers mode diff --git a/llvm/test/Other/mixed-opaque-ptrs-2.ll b/llvm/test/Other/mixed-opaque-ptrs-2.ll new file mode 100644 index 000000000000..d5fe38da514a --- /dev/null +++ b/llvm/test/Other/mixed-opaque-ptrs-2.ll @@ -0,0 +1,6 @@ +; RUN: llvm-as -disable-output %s 2>&1 +; FIXME: this should err out saying not to mix `ptr` and `foo*` +define void @f(ptr) { + %a = alloca i32* + ret void +} diff --git a/llvm/test/Other/mixed-opaque-ptrs.ll b/llvm/test/Other/mixed-opaque-ptrs.ll new file mode 100644 index 000000000000..1b67cf48474b --- /dev/null +++ b/llvm/test/Other/mixed-opaque-ptrs.ll @@ -0,0 +1,6 @@ +; RUN: not llvm-as -disable-output %s 2>&1 | FileCheck %s +; CHECK: ptr type is only supported in -opaque-pointers mode +define void @f(i32*) { + %a = alloca ptr + ret void +}