llvm-project/llvm/unittests/IR/AttributesTest.cpp
Nikita Popov 22e9024c9f
[IR] Introduce captures attribute (#116990)
This introduces the `captures` attribute as described in:
https://discourse.llvm.org/t/rfc-improvements-to-capture-tracking/81420

This initial patch only introduces the IR/bitcode support for the
attribute and its in-memory representation as `CaptureInfo`. This will
be followed by a patch to upgrade and remove the `nocapture` attribute,
and then by actual inference/analysis support.

Based on the RFC feedback, I've used a syntax similar to the `memory`
attribute, though the only "location" that can be specified is `ret`.

I've added some pretty extensive documentation to LangRef on the
semantics. One non-obvious bit here is that using ptrtoint will not
result in a "return-only" capture, even if the ptrtoint result is only
used in the return value. Without this requirement we wouldn't be able
to continue ordinary capture analysis on the return value.
2025-01-13 14:40:25 +01:00

792 lines
27 KiB
C++

//===- llvm/unittest/IR/AttributesTest.cpp - Attributes 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/IR/Attributes.h"
#include "llvm-c/Core.h"
#include "llvm/ADT/FloatingPointMode.h"
#include "llvm/AsmParser/Parser.h"
#include "llvm/IR/AttributeMask.h"
#include "llvm/IR/ConstantRange.h"
#include "llvm/IR/DerivedTypes.h"
#include "llvm/IR/InstrTypes.h"
#include "llvm/IR/LLVMContext.h"
#include "llvm/IR/Module.h"
#include "llvm/Support/SourceMgr.h"
#include "gtest/gtest.h"
using namespace llvm;
namespace {
TEST(Attributes, Uniquing) {
LLVMContext C;
Attribute AttrA = Attribute::get(C, Attribute::AlwaysInline);
Attribute AttrB = Attribute::get(C, Attribute::AlwaysInline);
EXPECT_EQ(AttrA, AttrB);
AttributeList ASs[] = {AttributeList::get(C, 1, Attribute::ZExt),
AttributeList::get(C, 2, Attribute::SExt)};
AttributeList SetA = AttributeList::get(C, ASs);
AttributeList SetB = AttributeList::get(C, ASs);
EXPECT_EQ(SetA, SetB);
}
TEST(Attributes, Ordering) {
LLVMContext C;
Attribute Align4 = Attribute::get(C, Attribute::Alignment, 4);
Attribute Align5 = Attribute::get(C, Attribute::Alignment, 5);
Attribute Deref4 = Attribute::get(C, Attribute::Dereferenceable, 4);
Attribute Deref5 = Attribute::get(C, Attribute::Dereferenceable, 5);
EXPECT_TRUE(Align4 < Align5);
EXPECT_TRUE(Align4 < Deref4);
EXPECT_TRUE(Align4 < Deref5);
EXPECT_TRUE(Align5 < Deref4);
EXPECT_EQ(Deref5.cmpKind(Deref4), 0);
EXPECT_EQ(Align4.cmpKind(Align5), 0);
Attribute ByVal = Attribute::get(C, Attribute::ByVal, Type::getInt32Ty(C));
EXPECT_FALSE(ByVal < Attribute::get(C, Attribute::ZExt));
EXPECT_TRUE(ByVal < Align4);
EXPECT_FALSE(ByVal < ByVal);
AttributeList ASs[] = {AttributeList::get(C, 2, Attribute::ZExt),
AttributeList::get(C, 1, Attribute::SExt)};
AttributeList SetA = AttributeList::get(C, ASs);
AttributeList SetB =
SetA.removeParamAttributes(C, 0, ASs[1].getParamAttrs(0));
EXPECT_NE(SetA, SetB);
}
TEST(Attributes, AddAttributes) {
LLVMContext C;
AttributeList AL;
AttrBuilder B(C);
B.addAttribute(Attribute::NoReturn);
AL = AL.addFnAttributes(C, AttrBuilder(C, AttributeSet::get(C, B)));
EXPECT_TRUE(AL.hasFnAttr(Attribute::NoReturn));
B.clear();
B.addAttribute(Attribute::SExt);
AL = AL.addRetAttributes(C, B);
EXPECT_TRUE(AL.hasRetAttr(Attribute::SExt));
EXPECT_TRUE(AL.hasFnAttr(Attribute::NoReturn));
}
TEST(Attributes, RemoveAlign) {
LLVMContext C;
Attribute AlignAttr = Attribute::getWithAlignment(C, Align(8));
Attribute StackAlignAttr = Attribute::getWithStackAlignment(C, Align(32));
AttrBuilder B_align_readonly(C);
B_align_readonly.addAttribute(AlignAttr);
B_align_readonly.addAttribute(Attribute::ReadOnly);
AttributeMask B_align;
B_align.addAttribute(AlignAttr);
AttrBuilder B_stackalign_optnone(C);
B_stackalign_optnone.addAttribute(StackAlignAttr);
B_stackalign_optnone.addAttribute(Attribute::OptimizeNone);
AttributeMask B_stackalign;
B_stackalign.addAttribute(StackAlignAttr);
AttributeSet AS = AttributeSet::get(C, B_align_readonly);
EXPECT_TRUE(AS.getAlignment() == MaybeAlign(8));
EXPECT_TRUE(AS.hasAttribute(Attribute::ReadOnly));
AS = AS.removeAttribute(C, Attribute::Alignment);
EXPECT_FALSE(AS.hasAttribute(Attribute::Alignment));
EXPECT_TRUE(AS.hasAttribute(Attribute::ReadOnly));
AS = AttributeSet::get(C, B_align_readonly);
AS = AS.removeAttributes(C, B_align);
EXPECT_TRUE(AS.getAlignment() == std::nullopt);
EXPECT_TRUE(AS.hasAttribute(Attribute::ReadOnly));
AttributeList AL;
AL = AL.addParamAttributes(C, 0, B_align_readonly);
AL = AL.addRetAttributes(C, B_stackalign_optnone);
EXPECT_TRUE(AL.hasRetAttrs());
EXPECT_TRUE(AL.hasRetAttr(Attribute::StackAlignment));
EXPECT_TRUE(AL.hasRetAttr(Attribute::OptimizeNone));
EXPECT_TRUE(AL.getRetStackAlignment() == MaybeAlign(32));
EXPECT_TRUE(AL.hasParamAttrs(0));
EXPECT_TRUE(AL.hasParamAttr(0, Attribute::Alignment));
EXPECT_TRUE(AL.hasParamAttr(0, Attribute::ReadOnly));
EXPECT_TRUE(AL.getParamAlignment(0) == MaybeAlign(8));
AL = AL.removeParamAttribute(C, 0, Attribute::Alignment);
EXPECT_FALSE(AL.hasParamAttr(0, Attribute::Alignment));
EXPECT_TRUE(AL.hasParamAttr(0, Attribute::ReadOnly));
EXPECT_TRUE(AL.hasRetAttr(Attribute::StackAlignment));
EXPECT_TRUE(AL.hasRetAttr(Attribute::OptimizeNone));
EXPECT_TRUE(AL.getRetStackAlignment() == MaybeAlign(32));
AL = AL.removeRetAttribute(C, Attribute::StackAlignment);
EXPECT_FALSE(AL.hasParamAttr(0, Attribute::Alignment));
EXPECT_TRUE(AL.hasParamAttr(0, Attribute::ReadOnly));
EXPECT_FALSE(AL.hasRetAttr(Attribute::StackAlignment));
EXPECT_TRUE(AL.hasRetAttr(Attribute::OptimizeNone));
AttributeList AL2;
AL2 = AL2.addParamAttributes(C, 0, B_align_readonly);
AL2 = AL2.addRetAttributes(C, B_stackalign_optnone);
AL2 = AL2.removeParamAttributes(C, 0, B_align);
EXPECT_FALSE(AL2.hasParamAttr(0, Attribute::Alignment));
EXPECT_TRUE(AL2.hasParamAttr(0, Attribute::ReadOnly));
EXPECT_TRUE(AL2.hasRetAttr(Attribute::StackAlignment));
EXPECT_TRUE(AL2.hasRetAttr(Attribute::OptimizeNone));
EXPECT_TRUE(AL2.getRetStackAlignment() == MaybeAlign(32));
AL2 = AL2.removeRetAttributes(C, B_stackalign);
EXPECT_FALSE(AL2.hasParamAttr(0, Attribute::Alignment));
EXPECT_TRUE(AL2.hasParamAttr(0, Attribute::ReadOnly));
EXPECT_FALSE(AL2.hasRetAttr(Attribute::StackAlignment));
EXPECT_TRUE(AL2.hasRetAttr(Attribute::OptimizeNone));
}
TEST(Attributes, AddMatchingAlignAttr) {
LLVMContext C;
AttributeList AL;
AL = AL.addParamAttribute(C, 0, Attribute::getWithAlignment(C, Align(8)));
AL = AL.addParamAttribute(C, 1, Attribute::getWithAlignment(C, Align(32)));
EXPECT_EQ(Align(8), AL.getParamAlignment(0));
EXPECT_EQ(Align(32), AL.getParamAlignment(1));
AttrBuilder B(C);
B.addAttribute(Attribute::NonNull);
B.addAlignmentAttr(8);
AL = AL.addParamAttributes(C, 0, B);
EXPECT_EQ(Align(8), AL.getParamAlignment(0));
EXPECT_EQ(Align(32), AL.getParamAlignment(1));
EXPECT_TRUE(AL.hasParamAttr(0, Attribute::NonNull));
}
TEST(Attributes, EmptyGet) {
LLVMContext C;
AttributeList EmptyLists[] = {AttributeList(), AttributeList()};
AttributeList AL = AttributeList::get(C, EmptyLists);
EXPECT_TRUE(AL.isEmpty());
}
TEST(Attributes, OverflowGet) {
LLVMContext C;
std::pair<unsigned, Attribute> Attrs[] = {
{AttributeList::ReturnIndex, Attribute::get(C, Attribute::SExt)},
{AttributeList::FunctionIndex, Attribute::get(C, Attribute::ReadOnly)}};
AttributeList AL = AttributeList::get(C, Attrs);
EXPECT_EQ(2U, AL.getNumAttrSets());
}
TEST(Attributes, StringRepresentation) {
LLVMContext C;
StructType *Ty = StructType::create(Type::getInt32Ty(C), "mystruct");
// Insufficiently careful printing can result in byval(%mystruct = { i32 })
Attribute A = Attribute::getWithByValType(C, Ty);
EXPECT_EQ(A.getAsString(), "byval(%mystruct)");
A = Attribute::getWithByValType(C, Type::getInt32Ty(C));
EXPECT_EQ(A.getAsString(), "byval(i32)");
}
TEST(Attributes, HasParentContext) {
LLVMContext C1, C2;
{
Attribute Attr1 = Attribute::get(C1, Attribute::AlwaysInline);
Attribute Attr2 = Attribute::get(C2, Attribute::AlwaysInline);
EXPECT_TRUE(Attr1.hasParentContext(C1));
EXPECT_FALSE(Attr1.hasParentContext(C2));
EXPECT_FALSE(Attr2.hasParentContext(C1));
EXPECT_TRUE(Attr2.hasParentContext(C2));
}
{
AttributeSet AS1 = AttributeSet::get(
C1, ArrayRef(Attribute::get(C1, Attribute::NoReturn)));
AttributeSet AS2 = AttributeSet::get(
C2, ArrayRef(Attribute::get(C2, Attribute::NoReturn)));
EXPECT_TRUE(AS1.hasParentContext(C1));
EXPECT_FALSE(AS1.hasParentContext(C2));
EXPECT_FALSE(AS2.hasParentContext(C1));
EXPECT_TRUE(AS2.hasParentContext(C2));
}
{
AttributeList AL1 = AttributeList::get(C1, 1, Attribute::ZExt);
AttributeList AL2 = AttributeList::get(C2, 1, Attribute::ZExt);
EXPECT_TRUE(AL1.hasParentContext(C1));
EXPECT_FALSE(AL1.hasParentContext(C2));
EXPECT_FALSE(AL2.hasParentContext(C1));
EXPECT_TRUE(AL2.hasParentContext(C2));
}
}
TEST(Attributes, AttributeListPrinting) {
LLVMContext C;
{
std::string S;
raw_string_ostream OS(S);
AttributeList AL;
AL.addFnAttribute(C, Attribute::AlwaysInline).print(OS);
EXPECT_EQ(S, "AttributeList[\n"
" { function => alwaysinline }\n"
"]\n");
}
{
std::string S;
raw_string_ostream OS(S);
AttributeList AL;
AL.addRetAttribute(C, Attribute::SExt).print(OS);
EXPECT_EQ(S, "AttributeList[\n"
" { return => signext }\n"
"]\n");
}
{
std::string S;
raw_string_ostream OS(S);
AttributeList AL;
AL.addParamAttribute(C, 5, Attribute::ZExt).print(OS);
EXPECT_EQ(S, "AttributeList[\n"
" { arg(5) => zeroext }\n"
"]\n");
}
}
TEST(Attributes, MismatchedABIAttrs) {
const char *IRString = R"IR(
declare void @f1(i32* byval(i32))
define void @g() {
call void @f1(i32* null)
ret void
}
declare void @f2(i32* preallocated(i32))
define void @h() {
call void @f2(i32* null)
ret void
}
declare void @f3(i32* inalloca(i32))
define void @i() {
call void @f3(i32* null)
ret void
}
)IR";
SMDiagnostic Err;
LLVMContext Context;
std::unique_ptr<Module> M = parseAssemblyString(IRString, Err, Context);
ASSERT_TRUE(M);
{
auto *I = cast<CallBase>(&M->getFunction("g")->getEntryBlock().front());
ASSERT_TRUE(I->isByValArgument(0));
ASSERT_TRUE(I->getParamByValType(0));
}
{
auto *I = cast<CallBase>(&M->getFunction("h")->getEntryBlock().front());
ASSERT_TRUE(I->getParamPreallocatedType(0));
}
{
auto *I = cast<CallBase>(&M->getFunction("i")->getEntryBlock().front());
ASSERT_TRUE(I->isInAllocaArgument(0));
ASSERT_TRUE(I->getParamInAllocaType(0));
}
}
TEST(Attributes, RemoveParamAttributes) {
LLVMContext C;
AttributeList AL;
AL = AL.addParamAttribute(C, 1, Attribute::NoUndef);
EXPECT_EQ(AL.getNumAttrSets(), 4U);
AL = AL.addParamAttribute(C, 3, Attribute::NonNull);
EXPECT_EQ(AL.getNumAttrSets(), 6U);
AL = AL.removeParamAttributes(C, 3);
EXPECT_EQ(AL.getNumAttrSets(), 4U);
AL = AL.removeParamAttribute(C, 1, Attribute::NoUndef);
EXPECT_EQ(AL.getNumAttrSets(), 0U);
}
TEST(Attributes, ConstantRangeAttributeCAPI) {
LLVMContext C;
{
const unsigned NumBits = 8;
const uint64_t LowerWords[] = {0};
const uint64_t UpperWords[] = {42};
ConstantRange Range(APInt(NumBits, ArrayRef(LowerWords)),
APInt(NumBits, ArrayRef(UpperWords)));
Attribute RangeAttr = Attribute::get(C, Attribute::Range, Range);
auto OutAttr = unwrap(LLVMCreateConstantRangeAttribute(
wrap(&C), Attribute::Range, NumBits, LowerWords, UpperWords));
EXPECT_EQ(OutAttr, RangeAttr);
}
{
const unsigned NumBits = 128;
const uint64_t LowerWords[] = {1, 1};
const uint64_t UpperWords[] = {42, 42};
ConstantRange Range(APInt(NumBits, ArrayRef(LowerWords)),
APInt(NumBits, ArrayRef(UpperWords)));
Attribute RangeAttr = Attribute::get(C, Attribute::Range, Range);
auto OutAttr = unwrap(LLVMCreateConstantRangeAttribute(
wrap(&C), Attribute::Range, NumBits, LowerWords, UpperWords));
EXPECT_EQ(OutAttr, RangeAttr);
}
}
TEST(Attributes, CalleeAttributes) {
const char *IRString = R"IR(
declare void @f1(i32 %i)
declare void @f2(i32 range(i32 1, 2) %i)
define void @g1(i32 %i) {
call void @f1(i32 %i)
ret void
}
define void @g2(i32 %i) {
call void @f2(i32 %i)
ret void
}
define void @g3(i32 %i) {
call void @f1(i32 range(i32 3, 4) %i)
ret void
}
define void @g4(i32 %i) {
call void @f2(i32 range(i32 3, 4) %i)
ret void
}
)IR";
SMDiagnostic Err;
LLVMContext Context;
std::unique_ptr<Module> M = parseAssemblyString(IRString, Err, Context);
ASSERT_TRUE(M);
{
auto *I = cast<CallBase>(&M->getFunction("g1")->getEntryBlock().front());
ASSERT_FALSE(I->getParamAttr(0, Attribute::Range).isValid());
}
{
auto *I = cast<CallBase>(&M->getFunction("g2")->getEntryBlock().front());
ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid());
}
{
auto *I = cast<CallBase>(&M->getFunction("g3")->getEntryBlock().front());
ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid());
}
{
auto *I = cast<CallBase>(&M->getFunction("g4")->getEntryBlock().front());
ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid());
}
}
TEST(Attributes, SetIntersect) {
LLVMContext C0, C1;
std::optional<AttributeSet> Res;
auto BuildAttr = [&](LLVMContext &C, Attribute::AttrKind Kind, uint64_t Int,
Type *Ty, ConstantRange &CR,
ArrayRef<ConstantRange> CRList) {
if (Attribute::isEnumAttrKind(Kind))
return Attribute::get(C, Kind);
if (Attribute::isTypeAttrKind(Kind))
return Attribute::get(C, Kind, Ty);
if (Attribute::isIntAttrKind(Kind))
return Attribute::get(C, Kind, Int);
if (Attribute::isConstantRangeAttrKind(Kind))
return Attribute::get(C, Kind, CR);
if (Attribute::isConstantRangeListAttrKind(Kind))
return Attribute::get(C, Kind, CRList);
std::abort();
};
for (unsigned i = Attribute::AttrKind::None + 1,
e = Attribute::AttrKind::EndAttrKinds;
i < e; ++i) {
Attribute::AttrKind Kind = static_cast<Attribute::AttrKind>(i);
Attribute::AttrKind Other =
Kind == Attribute::NoUndef ? Attribute::NonNull : Attribute::NoUndef;
AttributeSet AS0, AS1;
AttrBuilder AB0(C0);
AttrBuilder AB1(C1);
uint64_t V0, V1;
V0 = 0;
V1 = 0;
if (Attribute::intersectWithCustom(Kind)) {
switch (Kind) {
case Attribute::Alignment:
V0 = 2;
V1 = 4;
break;
case Attribute::Memory:
V0 = MemoryEffects::readOnly().toIntValue();
V1 = MemoryEffects::none().toIntValue();
break;
case Attribute::NoFPClass:
V0 = FPClassTest::fcNan | FPClassTest::fcInf;
V1 = FPClassTest::fcNan;
break;
case Attribute::Range:
break;
case Attribute::Captures:
V0 = CaptureInfo(CaptureComponents::AddressIsNull,
CaptureComponents::None)
.toIntValue();
V1 = CaptureInfo(CaptureComponents::None,
CaptureComponents::ReadProvenance)
.toIntValue();
break;
default:
ASSERT_FALSE(true);
}
} else {
V0 = (i & 2) + 1;
V1 = (2 - (i & 2)) + 1;
}
ConstantRange CR0(APInt(32, 0), APInt(32, 10));
ConstantRange CR1(APInt(32, 15), APInt(32, 20));
ConstantRange CRL0[] = {CR0};
ConstantRange CRL1[] = {CR0, CR1};
Type *T0 = Type::getInt32Ty(C0);
Type *T1 = Type::getInt64Ty(C0);
Attribute Attr0 = BuildAttr(C0, Kind, V0, T0, CR0, CRL0);
Attribute Attr1 = BuildAttr(
C1, Attribute::isEnumAttrKind(Kind) ? Other : Kind, V1, T1, CR1, CRL1);
bool CanDrop = Attribute::intersectWithAnd(Kind) ||
Attribute::intersectWithMin(Kind) ||
Attribute::intersectWithCustom(Kind);
AB0.addAttribute(Attr0);
AB1.addAttribute(Attr1);
Res = AS0.intersectWith(C0, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_EQ(AS0, *Res);
AS0 = AttributeSet::get(C0, AB0);
Res = AS0.intersectWith(C0, AS1);
ASSERT_EQ(Res.has_value(), CanDrop);
if (CanDrop)
ASSERT_FALSE(Res->hasAttributes());
AS1 = AttributeSet::get(C1, AB0);
Res = AS0.intersectWith(C0, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_EQ(AS0, *Res);
AS1 = AttributeSet::get(C1, AB1);
Res = AS0.intersectWith(C0, AS1);
if (!CanDrop) {
ASSERT_FALSE(Res.has_value());
continue;
}
if (Attribute::intersectWithAnd(Kind)) {
ASSERT_TRUE(Res.has_value());
ASSERT_FALSE(Res->hasAttributes());
AS1 = AS1.addAttribute(C1, Kind);
Res = AS0.intersectWith(C0, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttributes());
ASSERT_TRUE(Res->hasAttribute(Kind));
ASSERT_FALSE(Res->hasAttribute(Other));
} else if (Attribute::intersectWithMin(Kind)) {
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttributes());
ASSERT_TRUE(Res->hasAttribute(Kind));
ASSERT_EQ(Res->getAttribute(Kind).getValueAsInt(), std::min(V0, V1));
} else if (Attribute::intersectWithCustom(Kind)) {
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttributes());
ASSERT_TRUE(Res->hasAttribute(Kind));
switch (Kind) {
case Attribute::Alignment:
ASSERT_EQ(Res->getAlignment().valueOrOne(), MaybeAlign(2).valueOrOne());
break;
case Attribute::Memory:
ASSERT_EQ(Res->getMemoryEffects(), MemoryEffects::readOnly());
break;
case Attribute::NoFPClass:
ASSERT_EQ(Res->getNoFPClass(), FPClassTest::fcNan);
break;
case Attribute::Range:
ASSERT_EQ(Res->getAttribute(Kind).getRange(),
ConstantRange(APInt(32, 0), APInt(32, 20)));
break;
case Attribute::Captures:
ASSERT_EQ(Res->getCaptureInfo(),
CaptureInfo(CaptureComponents::AddressIsNull,
CaptureComponents::ReadProvenance));
break;
default:
ASSERT_FALSE(true);
}
}
AS0 = AS0.addAttribute(C0, Attribute::AlwaysInline);
ASSERT_FALSE(AS0.intersectWith(C0, AS1).has_value());
}
}
TEST(Attributes, SetIntersectByValAlign) {
LLVMContext C;
AttributeSet AS0, AS1;
Attribute ByVal = Attribute::get(C, Attribute::ByVal, Type::getInt32Ty(C));
Attribute Align0 = Attribute::get(C, Attribute::Alignment, 4);
Attribute Align1 = Attribute::get(C, Attribute::Alignment, 8);
{
AttrBuilder AB0(C), AB1(C);
AB0.addAttribute(Align0);
AB1.addAttribute(Align1);
AB0.addAttribute(Attribute::NoUndef);
AS0 = AttributeSet::get(C, AB0);
AS1 = AttributeSet::get(C, AB1);
auto Res = AS0.intersectWith(C, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttribute(Attribute::Alignment));
}
{
AttrBuilder AB0(C), AB1(C);
AB0.addAttribute(Align0);
AB0.addAttribute(ByVal);
AB1.addAttribute(Align1);
AB1.addAttribute(ByVal);
AB0.addAttribute(Attribute::NoUndef);
AS0 = AttributeSet::get(C, AB0);
AS1 = AttributeSet::get(C, AB1);
auto Res = AS0.intersectWith(C, AS1);
ASSERT_FALSE(Res.has_value());
}
{
AttrBuilder AB0(C), AB1(C);
AB0.addAttribute(Align0);
AB0.addAttribute(ByVal);
AB1.addAttribute(ByVal);
AB0.addAttribute(Attribute::NoUndef);
AS0 = AttributeSet::get(C, AB0);
AS1 = AttributeSet::get(C, AB1);
ASSERT_FALSE(AS0.intersectWith(C, AS1).has_value());
ASSERT_FALSE(AS1.intersectWith(C, AS0).has_value());
}
{
AttrBuilder AB0(C), AB1(C);
AB0.addAttribute(ByVal);
AB1.addAttribute(ByVal);
AB0.addAttribute(Attribute::NoUndef);
AS0 = AttributeSet::get(C, AB0);
AS1 = AttributeSet::get(C, AB1);
auto Res = AS0.intersectWith(C, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttribute(Attribute::ByVal));
}
{
AttrBuilder AB0(C), AB1(C);
AB0.addAttribute(ByVal);
AB0.addAttribute(Align0);
AB1.addAttribute(ByVal);
AB1.addAttribute(Align0);
AB0.addAttribute(Attribute::NoUndef);
AS0 = AttributeSet::get(C, AB0);
AS1 = AttributeSet::get(C, AB1);
auto Res = AS0.intersectWith(C, AS1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasAttribute(Attribute::ByVal));
ASSERT_TRUE(Res->hasAttribute(Attribute::Alignment));
}
}
TEST(Attributes, ListIntersectDifferingMustPreserve) {
LLVMContext C;
std::optional<AttributeList> Res;
{
AttributeList AL0;
AttributeList AL1;
AL1 = AL1.addFnAttribute(C, Attribute::ReadOnly);
AL0 = AL0.addParamAttribute(C, 0, Attribute::SExt);
Res = AL0.intersectWith(C, AL1);
ASSERT_FALSE(Res.has_value());
Res = AL1.intersectWith(C, AL0);
ASSERT_FALSE(Res.has_value());
}
{
AttributeList AL0;
AttributeList AL1;
AL1 = AL1.addFnAttribute(C, Attribute::AlwaysInline);
AL0 = AL0.addParamAttribute(C, 0, Attribute::ReadOnly);
Res = AL0.intersectWith(C, AL1);
ASSERT_FALSE(Res.has_value());
Res = AL1.intersectWith(C, AL0);
ASSERT_FALSE(Res.has_value());
AL0 = AL0.addFnAttribute(C, Attribute::AlwaysInline);
AL1 = AL1.addParamAttribute(C, 1, Attribute::SExt);
Res = AL0.intersectWith(C, AL1);
ASSERT_FALSE(Res.has_value());
Res = AL1.intersectWith(C, AL0);
ASSERT_FALSE(Res.has_value());
}
}
TEST(Attributes, ListIntersect) {
LLVMContext C;
AttributeList AL0;
AttributeList AL1;
std::optional<AttributeList> Res;
AL0 = AL0.addRetAttribute(C, Attribute::NoUndef);
AL1 = AL1.addRetAttribute(C, Attribute::NoUndef);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_EQ(AL0, *Res);
AL0 = AL0.addParamAttribute(C, 1, Attribute::NoUndef);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(1, Attribute::NoUndef));
AL1 = AL1.addParamAttribute(C, 2, Attribute::NoUndef);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NoUndef));
AL0 = AL0.addParamAttribute(C, 2, Attribute::NoUndef);
AL1 = AL1.addParamAttribute(C, 1, Attribute::NoUndef);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_EQ(AL0, *Res);
AL0 = AL0.addParamAttribute(C, 2, Attribute::NonNull);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_NE(AL0, *Res);
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
AL0 = AL0.addRetAttribute(C, Attribute::NonNull);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_NE(AL0, *Res);
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
AL0 = AL0.addFnAttribute(C, Attribute::ReadOnly);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_NE(AL0, *Res);
ASSERT_FALSE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
AL1 = AL1.addFnAttribute(C, Attribute::ReadOnly);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_NE(AL0, *Res);
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
AL1 = AL1.addFnAttribute(C, Attribute::AlwaysInline);
Res = AL0.intersectWith(C, AL1);
ASSERT_FALSE(Res.has_value());
AL0 = AL0.addFnAttribute(C, Attribute::AlwaysInline);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasFnAttr(Attribute::AlwaysInline));
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
AL1 = AL1.addParamAttribute(C, 2, Attribute::ReadNone);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasFnAttr(Attribute::AlwaysInline));
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::ReadNone));
AL1 = AL1.addParamAttribute(C, 3, Attribute::ReadNone);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasFnAttr(Attribute::AlwaysInline));
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::ReadNone));
ASSERT_FALSE(Res->hasParamAttr(3, Attribute::ReadNone));
AL0 = AL0.addParamAttribute(C, 3, Attribute::ReadNone);
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasFnAttr(Attribute::AlwaysInline));
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::ReadNone));
ASSERT_TRUE(Res->hasParamAttr(3, Attribute::ReadNone));
AL0 = AL0.addParamAttribute(
C, {3}, Attribute::get(C, Attribute::ByVal, Type::getInt32Ty(C)));
Res = AL0.intersectWith(C, AL1);
ASSERT_FALSE(Res.has_value());
AL1 = AL1.addParamAttribute(
C, {3}, Attribute::get(C, Attribute::ByVal, Type::getInt32Ty(C)));
Res = AL0.intersectWith(C, AL1);
ASSERT_TRUE(Res.has_value());
ASSERT_TRUE(Res->hasFnAttr(Attribute::AlwaysInline));
ASSERT_TRUE(Res->hasFnAttr(Attribute::ReadOnly));
ASSERT_TRUE(Res->hasRetAttr(Attribute::NoUndef));
ASSERT_FALSE(Res->hasRetAttr(Attribute::NonNull));
ASSERT_TRUE(Res->hasParamAttr(1, Attribute::NoUndef));
ASSERT_TRUE(Res->hasParamAttr(2, Attribute::NoUndef));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::NonNull));
ASSERT_FALSE(Res->hasParamAttr(2, Attribute::ReadNone));
ASSERT_TRUE(Res->hasParamAttr(3, Attribute::ReadNone));
ASSERT_TRUE(Res->hasParamAttr(3, Attribute::ByVal));
}
} // end anonymous namespace