//===--- TBAATest.cpp - Mixed TBAA unit tests -----------------------------===// // // 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/Analysis/AliasAnalysisEvaluator.h" #include "llvm/Analysis/Passes.h" #include "llvm/IR/Constants.h" #include "llvm/IR/Instructions.h" #include "llvm/IR/LLVMContext.h" #include "llvm/IR/LegacyPassManager.h" #include "llvm/IR/MDBuilder.h" #include "llvm/IR/Module.h" #include "llvm/IR/Verifier.h" #include "llvm/Support/CommandLine.h" #include "gtest/gtest.h" namespace llvm { namespace { class TBAATest : public testing::Test { protected: TBAATest() : M("TBAATest", C), MD(C) {} LLVMContext C; Module M; MDBuilder MD; }; static StoreInst *getFunctionWithSingleStore(Module *M, StringRef Name) { auto &C = M->getContext(); FunctionType *FTy = FunctionType::get(Type::getVoidTy(C), {}); auto *F = Function::Create(FTy, Function::ExternalLinkage, Name, M); auto *BB = BasicBlock::Create(C, "entry", F); auto *IntType = Type::getInt32Ty(C); auto *PtrType = Type::getInt32PtrTy(C); auto *SI = new StoreInst(ConstantInt::get(IntType, 42), ConstantPointerNull::get(PtrType), BB); ReturnInst::Create(C, nullptr, BB); return SI; } TEST_F(TBAATest, checkVerifierBehaviorForOldTBAA) { auto *SI = getFunctionWithSingleStore(&M, "f1"); auto *F = SI->getFunction(); // C++ unit test case to avoid going through the auto upgrade logic. auto *RootMD = MD.createTBAARoot("Simple C/C++ TBAA"); auto *MD1 = MD.createTBAANode("omnipotent char", RootMD); auto *MD2 = MD.createTBAANode("int", MD1); SI->setMetadata(LLVMContext::MD_tbaa, MD2); SmallVector ErrorMsg; raw_svector_ostream Outs(ErrorMsg); StringRef ExpectedFailureMsg( "Old-style TBAA is no longer allowed, use struct-path TBAA instead"); EXPECT_TRUE(verifyFunction(*F, &Outs)); EXPECT_TRUE(StringRef(ErrorMsg.begin(), ErrorMsg.size()) .startswith(ExpectedFailureMsg)); } TEST_F(TBAATest, checkTBAAMerging) { auto *SI = getFunctionWithSingleStore(&M, "f2"); auto *F = SI->getFunction(); auto *RootMD = MD.createTBAARoot("tbaa-root"); auto *MD1 = MD.createTBAANode("scalar-a", RootMD); auto *StructTag1 = MD.createTBAAStructTagNode(MD1, MD1, 0); auto *MD2 = MD.createTBAANode("scalar-b", RootMD); auto *StructTag2 = MD.createTBAAStructTagNode(MD2, MD2, 0); auto *GenericMD = MDNode::getMostGenericTBAA(StructTag1, StructTag2); EXPECT_EQ(GenericMD, nullptr); // Despite GenericMD being nullptr, we expect the setMetadata call to be well // defined and produce a well-formed function. SI->setMetadata(LLVMContext::MD_tbaa, GenericMD); EXPECT_TRUE(!verifyFunction(*F)); } } // end anonymous namspace } // end llvm namespace