AsmParser: Save and restore the parsing state for types using SlotMapping.

This commit extends the 'SlotMapping' structure and includes mappings for named
and numbered types in it. The LLParser is extended accordingly to fill out
those mappings at the end of module parsing.

This information is useful when we want to parse standalone constant values
at a later stage using the 'parseConstantValue' method. The constant values
can be constant expressions, which can contain references to types. In order
to parse such constant values, we have to restore the internal named and
numbered mappings for the types in LLParser, otherwise the parser will report
a parsing error. Therefore, this commit also introduces a new method called
'restoreParsingState' to LLParser, which uses the slot mappings to restore
some of its internal parsing state.

This commit is required to serialize constant value pointers in the machine
memory operands for the MIR format.

Reviewers: Duncan P. N. Exon Smith
llvm-svn: 245740
This commit is contained in:
Alex Lorenz 2015-08-21 21:32:39 +00:00
parent 0dc7d0fbc7
commit 6fab7d4ea6
6 changed files with 77 additions and 7 deletions

View File

@ -85,8 +85,11 @@ bool parseAssemblyInto(MemoryBufferRef F, Module &M, SMDiagnostic &Err,
/// The constant value can be any LLVM constant, including a constant /// The constant value can be any LLVM constant, including a constant
/// expression. /// expression.
/// ///
/// \param Slots The optional slot mapping that will restore the parsing state
/// of the module.
/// \return null on error. /// \return null on error.
Constant *parseConstantValue(StringRef Asm, SMDiagnostic &Err, const Module &M); Constant *parseConstantValue(StringRef Asm, SMDiagnostic &Err, const Module &M,
const SlotMapping *Slots = nullptr);
} // End llvm namespace } // End llvm namespace

View File

@ -14,6 +14,7 @@
#ifndef LLVM_ASMPARSER_SLOTMAPPING_H #ifndef LLVM_ASMPARSER_SLOTMAPPING_H
#define LLVM_ASMPARSER_SLOTMAPPING_H #define LLVM_ASMPARSER_SLOTMAPPING_H
#include "llvm/ADT/StringMap.h"
#include "llvm/IR/TrackingMDRef.h" #include "llvm/IR/TrackingMDRef.h"
#include <map> #include <map>
#include <vector> #include <vector>
@ -21,12 +22,19 @@
namespace llvm { namespace llvm {
class GlobalValue; class GlobalValue;
class Type;
/// This struct contains the mapping from the slot numbers to unnamed metadata /// This struct contains the mappings from the slot numbers to unnamed metadata
/// nodes and global values. /// nodes, global values and types. It also contains the mapping for the named
/// types.
/// It can be used to save the parsing state of an LLVM IR module so that the
/// textual references to the values in the module can be parsed outside of the
/// module's source.
struct SlotMapping { struct SlotMapping {
std::vector<GlobalValue *> GlobalValues; std::vector<GlobalValue *> GlobalValues;
std::map<unsigned, TrackingMDNodeRef> MetadataNodes; std::map<unsigned, TrackingMDNodeRef> MetadataNodes;
StringMap<Type *> NamedTypes;
std::map<unsigned, Type *> Types;
}; };
} // end namespace llvm } // end namespace llvm

View File

@ -49,7 +49,9 @@ bool LLParser::Run() {
ValidateEndOfModule(); ValidateEndOfModule();
} }
bool LLParser::parseStandaloneConstantValue(Constant *&C) { bool LLParser::parseStandaloneConstantValue(Constant *&C,
const SlotMapping *Slots) {
restoreParsingState(Slots);
Lex.Lex(); Lex.Lex();
Type *Ty = nullptr; Type *Ty = nullptr;
@ -60,6 +62,19 @@ bool LLParser::parseStandaloneConstantValue(Constant *&C) {
return false; return false;
} }
void LLParser::restoreParsingState(const SlotMapping *Slots) {
if (!Slots)
return;
NumberedVals = Slots->GlobalValues;
NumberedMetadata = Slots->MetadataNodes;
for (const auto &I : Slots->NamedTypes)
NamedTypes.insert(
std::make_pair(I.getKey(), std::make_pair(I.second, LocTy())));
for (const auto &I : Slots->Types)
NumberedTypes.insert(
std::make_pair(I.first, std::make_pair(I.second, LocTy())));
}
/// ValidateEndOfModule - Do final validity and sanity checks at the end of the /// ValidateEndOfModule - Do final validity and sanity checks at the end of the
/// module. /// module.
bool LLParser::ValidateEndOfModule() { bool LLParser::ValidateEndOfModule() {
@ -181,6 +196,10 @@ bool LLParser::ValidateEndOfModule() {
// the mapping from LLParser as it doesn't need it anymore. // the mapping from LLParser as it doesn't need it anymore.
Slots->GlobalValues = std::move(NumberedVals); Slots->GlobalValues = std::move(NumberedVals);
Slots->MetadataNodes = std::move(NumberedMetadata); Slots->MetadataNodes = std::move(NumberedMetadata);
for (const auto &I : NamedTypes)
Slots->NamedTypes.insert(std::make_pair(I.getKey(), I.second.first));
for (const auto &I : NumberedTypes)
Slots->Types.insert(std::make_pair(I.first, I.second.first));
return false; return false;
} }

View File

@ -146,7 +146,7 @@ namespace llvm {
Slots(Slots), BlockAddressPFS(nullptr) {} Slots(Slots), BlockAddressPFS(nullptr) {}
bool Run(); bool Run();
bool parseStandaloneConstantValue(Constant *&C); bool parseStandaloneConstantValue(Constant *&C, const SlotMapping *Slots);
LLVMContext &getContext() { return Context; } LLVMContext &getContext() { return Context; }
@ -159,6 +159,10 @@ namespace llvm {
return Error(Lex.getLoc(), Msg); return Error(Lex.getLoc(), Msg);
} }
/// Restore the internal name and slot mappings using the mappings that
/// were created at an earlier parsing stage.
void restoreParsingState(const SlotMapping *Slots);
/// GetGlobalVal - Get a value with the specified name or ID, creating a /// GetGlobalVal - Get a value with the specified name or ID, creating a
/// forward reference record if needed. This can return null if the value /// forward reference record if needed. This can return null if the value
/// exists but does not have the right type. /// exists but does not have the right type.

View File

@ -68,13 +68,13 @@ std::unique_ptr<Module> llvm::parseAssemblyString(StringRef AsmString,
} }
Constant *llvm::parseConstantValue(StringRef Asm, SMDiagnostic &Err, Constant *llvm::parseConstantValue(StringRef Asm, SMDiagnostic &Err,
const Module &M) { const Module &M, const SlotMapping *Slots) {
SourceMgr SM; SourceMgr SM;
std::unique_ptr<MemoryBuffer> Buf = MemoryBuffer::getMemBuffer(Asm); std::unique_ptr<MemoryBuffer> Buf = MemoryBuffer::getMemBuffer(Asm);
SM.AddNewSourceBuffer(std::move(Buf), SMLoc()); SM.AddNewSourceBuffer(std::move(Buf), SMLoc());
Constant *C; Constant *C;
if (LLParser(Asm, SM, Err, const_cast<Module *>(&M)) if (LLParser(Asm, SM, Err, const_cast<Module *>(&M))
.parseStandaloneConstantValue(C)) .parseStandaloneConstantValue(C, Slots))
return nullptr; return nullptr;
return C; return C;
} }

View File

@ -112,4 +112,40 @@ TEST(AsmParserTest, TypeAndConstantValueParsing) {
EXPECT_EQ(Error.getMessage(), "expected end of string"); EXPECT_EQ(Error.getMessage(), "expected end of string");
} }
TEST(AsmParserTest, TypeAndConstantValueWithSlotMappingParsing) {
LLVMContext &Ctx = getGlobalContext();
SMDiagnostic Error;
StringRef Source =
"%st = type { i32, i32 }\n"
"@v = common global [50 x %st] zeroinitializer, align 16\n"
"%0 = type { i32, i32, i32, i32 }\n"
"@g = common global [50 x %0] zeroinitializer, align 16\n"
"define void @marker4(i64 %d) {\n"
"entry:\n"
" %conv = trunc i64 %d to i32\n"
" store i32 %conv, i32* getelementptr inbounds "
" ([50 x %st], [50 x %st]* @v, i64 0, i64 0, i32 0), align 16\n"
" store i32 %conv, i32* getelementptr inbounds "
" ([50 x %0], [50 x %0]* @g, i64 0, i64 0, i32 0), align 16\n"
" ret void\n"
"}";
SlotMapping Mapping;
auto Mod = parseAssemblyString(Source, Error, Ctx, &Mapping);
ASSERT_TRUE(Mod != nullptr);
auto &M = *Mod;
const Value *V;
V = parseConstantValue("i32* getelementptr inbounds ([50 x %st], [50 x %st]* "
"@v, i64 0, i64 0, i32 0)",
Error, M, &Mapping);
ASSERT_TRUE(V);
ASSERT_TRUE(isa<ConstantExpr>(V));
V = parseConstantValue("i32* getelementptr inbounds ([50 x %0], [50 x %0]* "
"@g, i64 0, i64 0, i32 0)",
Error, M, &Mapping);
ASSERT_TRUE(V);
ASSERT_TRUE(isa<ConstantExpr>(V));
}
} // end anonymous namespace } // end anonymous namespace