//===- llvm/unittest/TypeBuilderTest.cpp - TypeBuilder tests --------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/IR/TypeBuilder.h" #include "llvm/ADT/ArrayRef.h" #include "llvm/IR/LLVMContext.h" #include "gtest/gtest.h" using namespace llvm; namespace { TEST(TypeBuilderTest, Void) { EXPECT_EQ(Type::getVoidTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getVoidTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); // Special cases for C compatibility: EXPECT_EQ(Type::getInt8PtrTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8PtrTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8PtrTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8PtrTy(getGlobalContext()), (TypeBuilder::get( getGlobalContext()))); } TEST(TypeBuilderTest, HostIntegers) { EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt16Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt16Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt32Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt32Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt64Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt64Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(IntegerType::get(getGlobalContext(), sizeof(size_t) * CHAR_BIT), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(IntegerType::get(getGlobalContext(), sizeof(ptrdiff_t) * CHAR_BIT), (TypeBuilder::get(getGlobalContext()))); } TEST(TypeBuilderTest, CrossCompilableIntegers) { EXPECT_EQ(IntegerType::get(getGlobalContext(), 1), (TypeBuilder, true>::get(getGlobalContext()))); EXPECT_EQ(IntegerType::get(getGlobalContext(), 1), (TypeBuilder, false>::get(getGlobalContext()))); EXPECT_EQ(IntegerType::get(getGlobalContext(), 72), (TypeBuilder, true>::get(getGlobalContext()))); EXPECT_EQ(IntegerType::get(getGlobalContext(), 72), (TypeBuilder, false>::get(getGlobalContext()))); } TEST(TypeBuilderTest, Float) { EXPECT_EQ(Type::getFloatTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getDoubleTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); // long double isn't supported yet. EXPECT_EQ(Type::getFloatTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getFloatTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getDoubleTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getDoubleTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getX86_FP80Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getX86_FP80Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getFP128Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getFP128Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getPPC_FP128Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getPPC_FP128Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); } TEST(TypeBuilderTest, Derived) { EXPECT_EQ(PointerType::getUnqual(Type::getInt8PtrTy(getGlobalContext())), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 7), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 0), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(PointerType::getUnqual(Type::getInt8PtrTy(getGlobalContext())), (TypeBuilder**, false>::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 7), (TypeBuilder[7], false>::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 0), (TypeBuilder[], false>::get(getGlobalContext()))); EXPECT_EQ(PointerType::getUnqual(Type::getInt8PtrTy(getGlobalContext())), (TypeBuilder**, true>::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 7), (TypeBuilder[7], true>::get(getGlobalContext()))); EXPECT_EQ(ArrayType::get(Type::getInt8Ty(getGlobalContext()), 0), (TypeBuilder[], true>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, false>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, false>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, false>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, true>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, true>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8Ty(getGlobalContext()), (TypeBuilder, true>::get(getGlobalContext()))); EXPECT_EQ(Type::getInt8PtrTy(getGlobalContext()), (TypeBuilder::get(getGlobalContext()))); } TEST(TypeBuilderTest, Functions) { std::vector params; EXPECT_EQ(FunctionType::get(Type::getVoidTy(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); params.push_back(TypeBuilder::get(getGlobalContext())); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); params.push_back(TypeBuilder::get(getGlobalContext())); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); params.push_back(TypeBuilder::get(getGlobalContext())); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); params.push_back(TypeBuilder::get(getGlobalContext())); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); params.push_back(TypeBuilder::get(getGlobalContext())); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, false), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(FunctionType::get(Type::getInt8Ty(getGlobalContext()), params, true), (TypeBuilder::get(getGlobalContext()))); } TEST(TypeBuilderTest, Context) { // We used to cache TypeBuilder results in static local variables. This // produced the same type for different contexts, which of course broke // things. LLVMContext context1; EXPECT_EQ(&context1, &(TypeBuilder, true>::get(context1))->getContext()); LLVMContext context2; EXPECT_EQ(&context2, &(TypeBuilder, true>::get(context2))->getContext()); } struct MyType { int a; int *b; void *array[1]; }; struct MyPortableType { int32_t a; int32_t *b; void *array[1]; }; } // anonymous namespace namespace llvm { template class TypeBuilder { public: static StructType *get(LLVMContext &Context) { // Using the static result variable ensures that the type is // only looked up once. std::vector st; st.push_back(TypeBuilder::get(Context)); st.push_back(TypeBuilder::get(Context)); st.push_back(TypeBuilder::get(Context)); static StructType *const result = StructType::get(Context, st); return result; } // You may find this a convenient place to put some constants // to help with getelementptr. They don't have any effect on // the operation of TypeBuilder. enum Fields { FIELD_A, FIELD_B, FIELD_ARRAY }; }; template class TypeBuilder { public: static StructType *get(LLVMContext &Context) { // Using the static result variable ensures that the type is // only looked up once. std::vector st; st.push_back(TypeBuilder, cross>::get(Context)); st.push_back(TypeBuilder*, cross>::get(Context)); st.push_back(TypeBuilder*[], cross>::get(Context)); static StructType *const result = StructType::get(Context, st); return result; } // You may find this a convenient place to put some constants // to help with getelementptr. They don't have any effect on // the operation of TypeBuilder. enum Fields { FIELD_A, FIELD_B, FIELD_ARRAY }; }; } // namespace llvm namespace { TEST(TypeBuilderTest, Extensions) { EXPECT_EQ(PointerType::getUnqual(StructType::get( TypeBuilder::get(getGlobalContext()), TypeBuilder::get(getGlobalContext()), TypeBuilder::get(getGlobalContext()), (void*)0)), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(PointerType::getUnqual(StructType::get( TypeBuilder, false>::get(getGlobalContext()), TypeBuilder*, false>::get(getGlobalContext()), TypeBuilder*[], false>::get(getGlobalContext()), (void*)0)), (TypeBuilder::get(getGlobalContext()))); EXPECT_EQ(PointerType::getUnqual(StructType::get( TypeBuilder, false>::get(getGlobalContext()), TypeBuilder*, false>::get(getGlobalContext()), TypeBuilder*[], false>::get(getGlobalContext()), (void*)0)), (TypeBuilder::get(getGlobalContext()))); } } // anonymous namespace