//===- 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 G0 = std::make_unique( PtrTy, false, GlobalValue::ExternalLinkage, nullptr, "G0"); std::unique_ptr G1 = std::make_unique( 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 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 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 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(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 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 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 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(MDA->getMetadata())); EXPECT_TRUE(isa(MDC->getMetadata())); EXPECT_EQ(&C, ValueMapper(VM).mapValue(A)); EXPECT_EQ(MDC, ValueMapper(VM).mapValue(*MDA)); } } // end namespace