diff --git a/docs/ReleaseNotes.html b/docs/ReleaseNotes.html index 685e9de1a35..750f9825cc6 100644 --- a/docs/ReleaseNotes.html +++ b/docs/ReleaseNotes.html @@ -532,6 +532,8 @@ syntax, there are still significant gaps in that support.

  • llvm::getTrapFunctionName()
  • llvm::EnableSegmentedStacks
  • +
  • The MDBuilder class has been added to simplify the creation of + metadata.
  • ....
  • diff --git a/include/llvm/Support/MDBuilder.h b/include/llvm/Support/MDBuilder.h new file mode 100644 index 00000000000..20f5d70c554 --- /dev/null +++ b/include/llvm/Support/MDBuilder.h @@ -0,0 +1,101 @@ +//===---- llvm/Support/MDBuilder.h - Builder for LLVM metadata --*- C++ -*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file defines the MDBuilder class, which is used as a convenient way to +// create LLVM metadata with a consistent and simplified interface. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_SUPPORT_MDBUILDER_H +#define LLVM_SUPPORT_MDBUILDER_H + +#include "llvm/Constants.h" +#include "llvm/DerivedTypes.h" +#include "llvm/LLVMContext.h" +#include "llvm/Metadata.h" +#include "llvm/ADT/APInt.h" + +namespace llvm { + + class MDBuilder { + LLVMContext &Context; + + public: + MDBuilder(LLVMContext &context) : Context(context) {} + + /// CreateString - Return the given string as metadata. + MDString *CreateString(StringRef Str) const { + return MDString::get(Context, Str); + } + + //===------------------------------------------------------------------===// + // Range metadata. + //===------------------------------------------------------------------===// + + /// CreateRange - Return metadata describing the range [Lo, Hi). + MDNode *CreateRange(const APInt &Lo, const APInt &Hi) const { + assert(Lo.getBitWidth() == Hi.getBitWidth() && "Mismatched bitwidths!"); + // If the range is everything then it is useless. + if (Hi == Lo) + return 0; + + // Return the range [Lo, Hi). + Type *Ty = IntegerType::get(Context, Lo.getBitWidth()); + Value *Range[2] = { ConstantInt::get(Ty, Lo), ConstantInt::get(Ty, Hi) }; + return MDNode::get(Context, Range); + } + + + //===------------------------------------------------------------------===// + // TBAA metadata. + //===------------------------------------------------------------------===// + + /// CreateAnonymousTBAARoot - Return metadata appropriate for a TBAA root + /// node. Each returned node is distinct from all other metadata and will + /// never be identified (uniqued) with anything else. + MDNode *CreateAnonymousTBAARoot() const { + // To ensure uniqueness the root node is self-referential. + MDNode *Dummy = MDNode::getTemporary(Context, ArrayRef()); + MDNode *Root = MDNode::get(Context, Dummy); + // At this point we have + // !0 = metadata !{} <- dummy + // !1 = metadata !{metadata !0} <- root + // Replace the dummy operand with the root node itself and delete the dummy. + Root->replaceOperandWith(0, Root); + MDNode::deleteTemporary(Dummy); + // We now have + // !1 = metadata !{metadata !1} <- self-referential root + return Root; + } + + /// CreateTBAARoot - Return metadata appropriate for a TBAA root node with + /// the given name. This may be identified (uniqued) with other roots with + /// the same name. + MDNode *CreateTBAARoot(StringRef Name) const { + return MDNode::get(Context, CreateString(Name)); + } + + /// CreateTBAANode - Return metadata for a non-root TBAA node with the given + /// name, parent in the TBAA tree, and value for 'pointsToConstantMemory'. + MDNode *CreateTBAANode(StringRef Name, MDNode *Parent, + bool isConstant = false) const { + if (isConstant) { + Constant *Flags = ConstantInt::get(Type::getInt64Ty(Context), 1); + Value *Ops[3] = { CreateString(Name), Parent, Flags }; + return MDNode::get(Context, Ops); + } else { + Value *Ops[2] = { CreateString(Name), Parent }; + return MDNode::get(Context, Ops); + } + } + }; + +} // end namespace llvm + +#endif diff --git a/unittests/Support/MDBuilderTest.cpp b/unittests/Support/MDBuilderTest.cpp new file mode 100644 index 00000000000..44218d38b54 --- /dev/null +++ b/unittests/Support/MDBuilderTest.cpp @@ -0,0 +1,91 @@ +//===- llvm/unittests/Support/MDBuilderTest.cpp - MDBuilder unit tests ----===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "gtest/gtest.h" +#include "llvm/Support/MDBuilder.h" +using namespace llvm; + +namespace { + +class MDBuilderTest : public testing::Test { +protected: + LLVMContext Context; +}; + +TEST_F(MDBuilderTest, CreateString) { + MDBuilder MDHelper(Context); + MDString *Str0 = MDHelper.CreateString(""); + MDString *Str1 = MDHelper.CreateString("string"); + EXPECT_EQ(Str0->getString(), StringRef("")); + EXPECT_EQ(Str1->getString(), StringRef("string")); +} + +TEST_F(MDBuilderTest, CreateRangeMetadata) { + MDBuilder MDHelper(Context); + APInt A(8, 1), B(8, 2); + MDNode *R0 = MDHelper.CreateRange(A, A); + MDNode *R1 = MDHelper.CreateRange(A, B); + EXPECT_EQ(R0, (MDNode *)0); + EXPECT_NE(R1, (MDNode *)0); + EXPECT_EQ(R1->getNumOperands(), 2U); + EXPECT_TRUE(isa(R1->getOperand(0))); + EXPECT_TRUE(isa(R1->getOperand(1))); + ConstantInt *C0 = cast(R1->getOperand(0)); + ConstantInt *C1 = cast(R1->getOperand(1)); + EXPECT_EQ(C0->getValue(), A); + EXPECT_EQ(C1->getValue(), B); +} +TEST_F(MDBuilderTest, CreateAnonymousTBAARoot) { + MDBuilder MDHelper(Context); + MDNode *R0 = MDHelper.CreateAnonymousTBAARoot(); + MDNode *R1 = MDHelper.CreateAnonymousTBAARoot(); + EXPECT_NE(R0, R1); + EXPECT_GE(R0->getNumOperands(), 1U); + EXPECT_GE(R1->getNumOperands(), 1U); + EXPECT_EQ(R0->getOperand(0), R0); + EXPECT_EQ(R1->getOperand(0), R1); + EXPECT_TRUE(R0->getNumOperands() == 1 || R0->getOperand(1) == 0); + EXPECT_TRUE(R1->getNumOperands() == 1 || R1->getOperand(1) == 0); +} +TEST_F(MDBuilderTest, CreateTBAARoot) { + MDBuilder MDHelper(Context); + MDNode *R0 = MDHelper.CreateTBAARoot("Root"); + MDNode *R1 = MDHelper.CreateTBAARoot("Root"); + EXPECT_EQ(R0, R1); + EXPECT_GE(R0->getNumOperands(), 1U); + EXPECT_TRUE(isa(R0->getOperand(0))); + EXPECT_EQ(cast(R0->getOperand(0))->getString(), "Root"); + EXPECT_TRUE(R0->getNumOperands() == 1 || R0->getOperand(1) == 0); +} +TEST_F(MDBuilderTest, CreateTBAANode) { + MDBuilder MDHelper(Context); + MDNode *R = MDHelper.CreateTBAARoot("Root"); + MDNode *N0 = MDHelper.CreateTBAANode("Node", R); + MDNode *N1 = MDHelper.CreateTBAANode("edoN", R); + MDNode *N2 = MDHelper.CreateTBAANode("Node", R, true); + MDNode *N3 = MDHelper.CreateTBAANode("Node", R); + EXPECT_EQ(N0, N3); + EXPECT_NE(N0, N1); + EXPECT_NE(N0, N2); + EXPECT_GE(N0->getNumOperands(), 2U); + EXPECT_GE(N1->getNumOperands(), 2U); + EXPECT_GE(N2->getNumOperands(), 3U); + EXPECT_TRUE(isa(N0->getOperand(0))); + EXPECT_TRUE(isa(N1->getOperand(0))); + EXPECT_TRUE(isa(N2->getOperand(0))); + EXPECT_EQ(cast(N0->getOperand(0))->getString(), "Node"); + EXPECT_EQ(cast(N1->getOperand(0))->getString(), "edoN"); + EXPECT_EQ(cast(N2->getOperand(0))->getString(), "Node"); + EXPECT_EQ(N0->getOperand(1), R); + EXPECT_EQ(N1->getOperand(1), R); + EXPECT_EQ(N2->getOperand(1), R); + EXPECT_TRUE(isa(N2->getOperand(2))); + EXPECT_EQ(cast(N2->getOperand(2))->getZExtValue(), 1U); +} +}