//===- 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 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 M = parseAssemblyString(IRString, Err, Context); ASSERT_TRUE(M); { auto *I = cast(&M->getFunction("g")->getEntryBlock().front()); ASSERT_TRUE(I->isByValArgument(0)); ASSERT_TRUE(I->getParamByValType(0)); } { auto *I = cast(&M->getFunction("h")->getEntryBlock().front()); ASSERT_TRUE(I->getParamPreallocatedType(0)); } { auto *I = cast(&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 M = parseAssemblyString(IRString, Err, Context); ASSERT_TRUE(M); { auto *I = cast(&M->getFunction("g1")->getEntryBlock().front()); ASSERT_FALSE(I->getParamAttr(0, Attribute::Range).isValid()); } { auto *I = cast(&M->getFunction("g2")->getEntryBlock().front()); ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid()); } { auto *I = cast(&M->getFunction("g3")->getEntryBlock().front()); ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid()); } { auto *I = cast(&M->getFunction("g4")->getEntryBlock().front()); ASSERT_TRUE(I->getParamAttr(0, Attribute::Range).isValid()); } } TEST(Attributes, SetIntersect) { LLVMContext C0, C1; std::optional Res; auto BuildAttr = [&](LLVMContext &C, Attribute::AttrKind Kind, uint64_t Int, Type *Ty, ConstantRange &CR, ArrayRef 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(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 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 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