123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349 |
- //===- ValueMapper.cpp - Unit tests for ValueMapper -----------------------===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/Transforms/Utils/ValueMapper.h"
- #include "llvm/IR/Constants.h"
- #include "llvm/IR/Function.h"
- #include "llvm/IR/GlobalVariable.h"
- #include "llvm/IR/LLVMContext.h"
- #include "llvm/IR/Metadata.h"
- #include "gtest/gtest.h"
- using namespace llvm;
- namespace {
- TEST(ValueMapperTest, mapMDNode) {
- LLVMContext Context;
- auto *U = MDTuple::get(Context, None);
- // The node should be unchanged.
- ValueToValueMapTy VM;
- EXPECT_EQ(U, ValueMapper(VM).mapMDNode(*U));
- }
- TEST(ValueMapperTest, mapMDNodeCycle) {
- LLVMContext Context;
- MDNode *U0;
- MDNode *U1;
- {
- Metadata *Ops[] = {nullptr};
- auto T = MDTuple::getTemporary(Context, Ops);
- Ops[0] = T.get();
- U0 = MDTuple::get(Context, Ops);
- T->replaceOperandWith(0, U0);
- U1 = MDNode::replaceWithUniqued(std::move(T));
- U0->resolveCycles();
- }
- EXPECT_TRUE(U0->isResolved());
- EXPECT_TRUE(U0->isUniqued());
- EXPECT_TRUE(U1->isResolved());
- EXPECT_TRUE(U1->isUniqued());
- EXPECT_EQ(U1, U0->getOperand(0));
- EXPECT_EQ(U0, U1->getOperand(0));
- // Cycles shouldn't be duplicated.
- {
- ValueToValueMapTy VM;
- EXPECT_EQ(U0, ValueMapper(VM).mapMDNode(*U0));
- EXPECT_EQ(U1, ValueMapper(VM).mapMDNode(*U1));
- }
- // Check the other order.
- {
- ValueToValueMapTy VM;
- EXPECT_EQ(U1, ValueMapper(VM).mapMDNode(*U1));
- EXPECT_EQ(U0, ValueMapper(VM).mapMDNode(*U0));
- }
- }
- TEST(ValueMapperTest, mapMDNodeDuplicatedCycle) {
- LLVMContext Context;
- auto *PtrTy = Type::getInt8Ty(Context)->getPointerTo();
- std::unique_ptr<GlobalVariable> G0 = std::make_unique<GlobalVariable>(
- PtrTy, false, GlobalValue::ExternalLinkage, nullptr, "G0");
- std::unique_ptr<GlobalVariable> G1 = std::make_unique<GlobalVariable>(
- PtrTy, false, GlobalValue::ExternalLinkage, nullptr, "G1");
- // Create a cycle that references G0.
- MDNode *N0; // !0 = !{!1}
- MDNode *N1; // !1 = !{!0, i8* @G0}
- {
- auto T0 = MDTuple::getTemporary(Context, nullptr);
- Metadata *Ops1[] = {T0.get(), ConstantAsMetadata::get(G0.get())};
- N1 = MDTuple::get(Context, Ops1);
- T0->replaceOperandWith(0, N1);
- N0 = MDNode::replaceWithUniqued(std::move(T0));
- }
- // Resolve N0 and N1.
- ASSERT_FALSE(N0->isResolved());
- ASSERT_FALSE(N1->isResolved());
- N0->resolveCycles();
- ASSERT_TRUE(N0->isResolved());
- ASSERT_TRUE(N1->isResolved());
- // Seed the value map to map G0 to G1 and map the nodes. The output should
- // have new nodes that reference G1 (instead of G0).
- ValueToValueMapTy VM;
- VM[G0.get()] = G1.get();
- MDNode *MappedN0 = ValueMapper(VM).mapMDNode(*N0);
- MDNode *MappedN1 = ValueMapper(VM).mapMDNode(*N1);
- EXPECT_NE(N0, MappedN0);
- EXPECT_NE(N1, MappedN1);
- EXPECT_EQ(ConstantAsMetadata::get(G1.get()), MappedN1->getOperand(1));
- // Check that the output nodes are resolved.
- EXPECT_TRUE(MappedN0->isResolved());
- EXPECT_TRUE(MappedN1->isResolved());
- }
- TEST(ValueMapperTest, mapMDNodeUnresolved) {
- LLVMContext Context;
- TempMDTuple T = MDTuple::getTemporary(Context, None);
- ValueToValueMapTy VM;
- EXPECT_EQ(T.get(), ValueMapper(VM, RF_NoModuleLevelChanges).mapMDNode(*T));
- }
- TEST(ValueMapperTest, mapMDNodeDistinct) {
- LLVMContext Context;
- auto *D = MDTuple::getDistinct(Context, None);
- {
- // The node should be cloned.
- ValueToValueMapTy VM;
- EXPECT_NE(D, ValueMapper(VM).mapMDNode(*D));
- }
- {
- // The node should be moved.
- ValueToValueMapTy VM;
- EXPECT_EQ(D, ValueMapper(VM, RF_MoveDistinctMDs).mapMDNode(*D));
- }
- }
- TEST(ValueMapperTest, mapMDNodeDistinctOperands) {
- LLVMContext Context;
- Metadata *Old = MDTuple::getDistinct(Context, None);
- auto *D = MDTuple::getDistinct(Context, Old);
- ASSERT_EQ(Old, D->getOperand(0));
- Metadata *New = MDTuple::getDistinct(Context, None);
- ValueToValueMapTy VM;
- VM.MD()[Old].reset(New);
- // Make sure operands are updated.
- EXPECT_EQ(D, ValueMapper(VM, RF_MoveDistinctMDs).mapMDNode(*D));
- EXPECT_EQ(New, D->getOperand(0));
- }
- TEST(ValueMapperTest, mapMDNodeSeeded) {
- LLVMContext Context;
- auto *D = MDTuple::getDistinct(Context, None);
- // The node should be moved.
- ValueToValueMapTy VM;
- EXPECT_EQ(None, VM.getMappedMD(D));
- VM.MD().insert(std::make_pair(D, TrackingMDRef(D)));
- EXPECT_EQ(D, *VM.getMappedMD(D));
- EXPECT_EQ(D, ValueMapper(VM).mapMDNode(*D));
- }
- TEST(ValueMapperTest, mapMDNodeSeededWithNull) {
- LLVMContext Context;
- auto *D = MDTuple::getDistinct(Context, None);
- // The node should be moved.
- ValueToValueMapTy VM;
- EXPECT_EQ(None, VM.getMappedMD(D));
- VM.MD().insert(std::make_pair(D, TrackingMDRef()));
- EXPECT_EQ(nullptr, *VM.getMappedMD(D));
- EXPECT_EQ(nullptr, ValueMapper(VM).mapMDNode(*D));
- }
- TEST(ValueMapperTest, mapMetadataNullMapGlobalWithIgnoreMissingLocals) {
- LLVMContext C;
- FunctionType *FTy =
- FunctionType::get(Type::getVoidTy(C), Type::getInt8Ty(C), false);
- std::unique_ptr<Function> F(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F"));
- ValueToValueMapTy VM;
- RemapFlags Flags = RF_IgnoreMissingLocals | RF_NullMapMissingGlobalValues;
- EXPECT_EQ(nullptr, ValueMapper(VM, Flags).mapValue(*F));
- }
- TEST(ValueMapperTest, mapMetadataMDString) {
- LLVMContext C;
- auto *S1 = MDString::get(C, "S1");
- ValueToValueMapTy VM;
- // Make sure S1 maps to itself, but isn't memoized.
- EXPECT_EQ(S1, ValueMapper(VM).mapMetadata(*S1));
- EXPECT_EQ(None, VM.getMappedMD(S1));
- // We still expect VM.MD() to be respected.
- auto *S2 = MDString::get(C, "S2");
- VM.MD()[S1].reset(S2);
- EXPECT_EQ(S2, ValueMapper(VM).mapMetadata(*S1));
- }
- TEST(ValueMapperTest, mapMetadataGetMappedMD) {
- LLVMContext C;
- auto *N0 = MDTuple::get(C, None);
- auto *N1 = MDTuple::get(C, N0);
- // Make sure hasMD and getMappedMD work correctly.
- ValueToValueMapTy VM;
- EXPECT_FALSE(VM.hasMD());
- EXPECT_EQ(N0, ValueMapper(VM).mapMetadata(*N0));
- EXPECT_EQ(N1, ValueMapper(VM).mapMetadata(*N1));
- EXPECT_TRUE(VM.hasMD());
- ASSERT_NE(None, VM.getMappedMD(N0));
- ASSERT_NE(None, VM.getMappedMD(N1));
- EXPECT_EQ(N0, *VM.getMappedMD(N0));
- EXPECT_EQ(N1, *VM.getMappedMD(N1));
- }
- TEST(ValueMapperTest, mapMetadataNoModuleLevelChanges) {
- LLVMContext C;
- auto *N0 = MDTuple::get(C, None);
- auto *N1 = MDTuple::get(C, N0);
- // Nothing should be memoized when RF_NoModuleLevelChanges.
- ValueToValueMapTy VM;
- EXPECT_FALSE(VM.hasMD());
- EXPECT_EQ(N0, ValueMapper(VM, RF_NoModuleLevelChanges).mapMetadata(*N0));
- EXPECT_EQ(N1, ValueMapper(VM, RF_NoModuleLevelChanges).mapMetadata(*N1));
- EXPECT_FALSE(VM.hasMD());
- EXPECT_EQ(None, VM.getMappedMD(N0));
- EXPECT_EQ(None, VM.getMappedMD(N1));
- }
- TEST(ValueMapperTest, mapMetadataConstantAsMetadata) {
- LLVMContext C;
- FunctionType *FTy =
- FunctionType::get(Type::getVoidTy(C), Type::getInt8Ty(C), false);
- std::unique_ptr<Function> F(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F"));
- auto *CAM = ConstantAsMetadata::get(F.get());
- {
- // ConstantAsMetadata shouldn't be memoized.
- ValueToValueMapTy VM;
- EXPECT_EQ(CAM, ValueMapper(VM).mapMetadata(*CAM));
- EXPECT_FALSE(VM.MD().count(CAM));
- EXPECT_EQ(CAM, ValueMapper(VM, RF_IgnoreMissingLocals).mapMetadata(*CAM));
- EXPECT_FALSE(VM.MD().count(CAM));
- // But it should respect a mapping that gets seeded.
- auto *N = MDTuple::get(C, None);
- VM.MD()[CAM].reset(N);
- EXPECT_EQ(N, ValueMapper(VM).mapMetadata(*CAM));
- EXPECT_EQ(N, ValueMapper(VM, RF_IgnoreMissingLocals).mapMetadata(*CAM));
- }
- std::unique_ptr<Function> F2(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F2"));
- ValueToValueMapTy VM;
- VM[F.get()] = F2.get();
- auto *F2MD = ValueMapper(VM).mapMetadata(*CAM);
- EXPECT_FALSE(VM.MD().count(CAM));
- EXPECT_TRUE(F2MD);
- EXPECT_EQ(F2.get(), cast<ConstantAsMetadata>(F2MD)->getValue());
- }
- #ifdef GTEST_HAS_DEATH_TEST
- #ifndef NDEBUG
- TEST(ValueMapperTest, mapMetadataLocalAsMetadata) {
- LLVMContext C;
- FunctionType *FTy =
- FunctionType::get(Type::getVoidTy(C), Type::getInt8Ty(C), false);
- std::unique_ptr<Function> F(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F"));
- Argument &A = *F->arg_begin();
- // mapMetadata doesn't support LocalAsMetadata. The only valid container for
- // LocalAsMetadata is a MetadataAsValue instance, so use it directly.
- auto *LAM = LocalAsMetadata::get(&A);
- ValueToValueMapTy VM;
- EXPECT_DEATH(ValueMapper(VM).mapMetadata(*LAM), "Unexpected local metadata");
- EXPECT_DEATH(ValueMapper(VM, RF_IgnoreMissingLocals).mapMetadata(*LAM),
- "Unexpected local metadata");
- }
- #endif
- #endif
- TEST(ValueMapperTest, mapValueLocalAsMetadata) {
- LLVMContext C;
- FunctionType *FTy =
- FunctionType::get(Type::getVoidTy(C), Type::getInt8Ty(C), false);
- std::unique_ptr<Function> F(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F"));
- Argument &A = *F->arg_begin();
- auto *LAM = LocalAsMetadata::get(&A);
- auto *MAV = MetadataAsValue::get(C, LAM);
- // The principled answer to a LocalAsMetadata of an unmapped SSA value would
- // be to return nullptr (regardless of RF_IgnoreMissingLocals).
- //
- // However, algorithms that use RemapInstruction assume that each instruction
- // only references SSA values from previous instructions. Arguments of
- // such as "metadata i32 %x" don't currently successfully maintain that
- // property. To keep RemapInstruction from crashing we need a non-null
- // return here, but we also shouldn't reference the unmapped local. Use
- // "metadata !{}".
- auto *N0 = MDTuple::get(C, None);
- auto *N0AV = MetadataAsValue::get(C, N0);
- ValueToValueMapTy VM;
- EXPECT_EQ(N0AV, ValueMapper(VM).mapValue(*MAV));
- EXPECT_EQ(nullptr, ValueMapper(VM, RF_IgnoreMissingLocals).mapValue(*MAV));
- EXPECT_FALSE(VM.count(MAV));
- EXPECT_FALSE(VM.count(&A));
- EXPECT_EQ(None, VM.getMappedMD(LAM));
- VM[MAV] = MAV;
- EXPECT_EQ(MAV, ValueMapper(VM).mapValue(*MAV));
- EXPECT_EQ(MAV, ValueMapper(VM, RF_IgnoreMissingLocals).mapValue(*MAV));
- EXPECT_TRUE(VM.count(MAV));
- EXPECT_FALSE(VM.count(&A));
- VM[MAV] = &A;
- EXPECT_EQ(&A, ValueMapper(VM).mapValue(*MAV));
- EXPECT_EQ(&A, ValueMapper(VM, RF_IgnoreMissingLocals).mapValue(*MAV));
- EXPECT_TRUE(VM.count(MAV));
- EXPECT_FALSE(VM.count(&A));
- }
- TEST(ValueMapperTest, mapValueLocalAsMetadataToConstant) {
- LLVMContext Context;
- auto *Int8 = Type::getInt8Ty(Context);
- FunctionType *FTy = FunctionType::get(Type::getVoidTy(Context), Int8, false);
- std::unique_ptr<Function> F(
- Function::Create(FTy, GlobalValue::ExternalLinkage, "F"));
- // Map a local value to a constant.
- Argument &A = *F->arg_begin();
- Constant &C = *ConstantInt::get(Int8, 42);
- ValueToValueMapTy VM;
- VM[&A] = &C;
- // Look up the metadata-as-value wrapper. Don't crash.
- auto *MDA = MetadataAsValue::get(Context, ValueAsMetadata::get(&A));
- auto *MDC = MetadataAsValue::get(Context, ValueAsMetadata::get(&C));
- EXPECT_TRUE(isa<LocalAsMetadata>(MDA->getMetadata()));
- EXPECT_TRUE(isa<ConstantAsMetadata>(MDC->getMetadata()));
- EXPECT_EQ(&C, ValueMapper(VM).mapValue(A));
- EXPECT_EQ(MDC, ValueMapper(VM).mapValue(*MDA));
- }
- } // end namespace
|