mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-03 14:21:30 +00:00 
			
		
		
		
	subsequent changes are easier to review. About to fix some layering issues, and wanted to separate out the necessary churn. Also comment and sink the include of "Windows.h" in three .inc files to match the usage in Memory.inc. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@198685 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			177 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//===- Cloning.cpp - Unit tests for the Cloner ----------------------------===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "llvm/ADT/STLExtras.h"
 | 
						|
#include "llvm/ADT/SmallPtrSet.h"
 | 
						|
#include "llvm/IR/Argument.h"
 | 
						|
#include "llvm/IR/Constant.h"
 | 
						|
#include "llvm/IR/Function.h"
 | 
						|
#include "llvm/IR/IRBuilder.h"
 | 
						|
#include "llvm/IR/Instructions.h"
 | 
						|
#include "llvm/IR/LLVMContext.h"
 | 
						|
#include "llvm/Transforms/Utils/Cloning.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
class CloneInstruction : public ::testing::Test {
 | 
						|
protected:
 | 
						|
  virtual void SetUp() {
 | 
						|
    V = NULL;
 | 
						|
  }
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  T *clone(T *V1) {
 | 
						|
    Value *V2 = V1->clone();
 | 
						|
    Orig.insert(V1);
 | 
						|
    Clones.insert(V2);
 | 
						|
    return cast<T>(V2);
 | 
						|
  }
 | 
						|
 | 
						|
  void eraseClones() {
 | 
						|
    DeleteContainerPointers(Clones);
 | 
						|
  }
 | 
						|
 | 
						|
  virtual void TearDown() {
 | 
						|
    eraseClones();
 | 
						|
    DeleteContainerPointers(Orig);
 | 
						|
    delete V;
 | 
						|
  }
 | 
						|
 | 
						|
  SmallPtrSet<Value *, 4> Orig;   // Erase on exit
 | 
						|
  SmallPtrSet<Value *, 4> Clones; // Erase in eraseClones
 | 
						|
 | 
						|
  LLVMContext context;
 | 
						|
  Value *V;
 | 
						|
};
 | 
						|
 | 
						|
TEST_F(CloneInstruction, OverflowBits) {
 | 
						|
  V = new Argument(Type::getInt32Ty(context));
 | 
						|
 | 
						|
  BinaryOperator *Add = BinaryOperator::Create(Instruction::Add, V, V);
 | 
						|
  BinaryOperator *Sub = BinaryOperator::Create(Instruction::Sub, V, V);
 | 
						|
  BinaryOperator *Mul = BinaryOperator::Create(Instruction::Mul, V, V);
 | 
						|
 | 
						|
  BinaryOperator *AddClone = this->clone(Add);
 | 
						|
  BinaryOperator *SubClone = this->clone(Sub);
 | 
						|
  BinaryOperator *MulClone = this->clone(Mul);
 | 
						|
 | 
						|
  EXPECT_FALSE(AddClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(AddClone->hasNoSignedWrap());
 | 
						|
  EXPECT_FALSE(SubClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(SubClone->hasNoSignedWrap());
 | 
						|
  EXPECT_FALSE(MulClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(MulClone->hasNoSignedWrap());
 | 
						|
 | 
						|
  eraseClones();
 | 
						|
 | 
						|
  Add->setHasNoUnsignedWrap();
 | 
						|
  Sub->setHasNoUnsignedWrap();
 | 
						|
  Mul->setHasNoUnsignedWrap();
 | 
						|
 | 
						|
  AddClone = this->clone(Add);
 | 
						|
  SubClone = this->clone(Sub);
 | 
						|
  MulClone = this->clone(Mul);
 | 
						|
 | 
						|
  EXPECT_TRUE(AddClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(AddClone->hasNoSignedWrap());
 | 
						|
  EXPECT_TRUE(SubClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(SubClone->hasNoSignedWrap());
 | 
						|
  EXPECT_TRUE(MulClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_FALSE(MulClone->hasNoSignedWrap());
 | 
						|
 | 
						|
  eraseClones();
 | 
						|
 | 
						|
  Add->setHasNoSignedWrap();
 | 
						|
  Sub->setHasNoSignedWrap();
 | 
						|
  Mul->setHasNoSignedWrap();
 | 
						|
 | 
						|
  AddClone = this->clone(Add);
 | 
						|
  SubClone = this->clone(Sub);
 | 
						|
  MulClone = this->clone(Mul);
 | 
						|
 | 
						|
  EXPECT_TRUE(AddClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(AddClone->hasNoSignedWrap());
 | 
						|
  EXPECT_TRUE(SubClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(SubClone->hasNoSignedWrap());
 | 
						|
  EXPECT_TRUE(MulClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(MulClone->hasNoSignedWrap());
 | 
						|
 | 
						|
  eraseClones();
 | 
						|
 | 
						|
  Add->setHasNoUnsignedWrap(false);
 | 
						|
  Sub->setHasNoUnsignedWrap(false);
 | 
						|
  Mul->setHasNoUnsignedWrap(false);
 | 
						|
 | 
						|
  AddClone = this->clone(Add);
 | 
						|
  SubClone = this->clone(Sub);
 | 
						|
  MulClone = this->clone(Mul);
 | 
						|
 | 
						|
  EXPECT_FALSE(AddClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(AddClone->hasNoSignedWrap());
 | 
						|
  EXPECT_FALSE(SubClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(SubClone->hasNoSignedWrap());
 | 
						|
  EXPECT_FALSE(MulClone->hasNoUnsignedWrap());
 | 
						|
  EXPECT_TRUE(MulClone->hasNoSignedWrap());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(CloneInstruction, Inbounds) {
 | 
						|
  V = new Argument(Type::getInt32PtrTy(context));
 | 
						|
 | 
						|
  Constant *Z = Constant::getNullValue(Type::getInt32Ty(context));
 | 
						|
  std::vector<Value *> ops;
 | 
						|
  ops.push_back(Z);
 | 
						|
  GetElementPtrInst *GEP = GetElementPtrInst::Create(V, ops);
 | 
						|
  EXPECT_FALSE(this->clone(GEP)->isInBounds());
 | 
						|
 | 
						|
  GEP->setIsInBounds();
 | 
						|
  EXPECT_TRUE(this->clone(GEP)->isInBounds());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(CloneInstruction, Exact) {
 | 
						|
  V = new Argument(Type::getInt32Ty(context));
 | 
						|
 | 
						|
  BinaryOperator *SDiv = BinaryOperator::Create(Instruction::SDiv, V, V);
 | 
						|
  EXPECT_FALSE(this->clone(SDiv)->isExact());
 | 
						|
 | 
						|
  SDiv->setIsExact(true);
 | 
						|
  EXPECT_TRUE(this->clone(SDiv)->isExact());
 | 
						|
}
 | 
						|
 | 
						|
TEST_F(CloneInstruction, Attributes) {
 | 
						|
  Type *ArgTy1[] = { Type::getInt32PtrTy(context) };
 | 
						|
  FunctionType *FT1 =  FunctionType::get(Type::getVoidTy(context), ArgTy1, false);
 | 
						|
 | 
						|
  Function *F1 = Function::Create(FT1, Function::ExternalLinkage);
 | 
						|
  BasicBlock *BB = BasicBlock::Create(context, "", F1);
 | 
						|
  IRBuilder<> Builder(BB);
 | 
						|
  Builder.CreateRetVoid();
 | 
						|
 | 
						|
  Function *F2 = Function::Create(FT1, Function::ExternalLinkage);
 | 
						|
 | 
						|
  Attribute::AttrKind AK[] = { Attribute::NoCapture };
 | 
						|
  AttributeSet AS = AttributeSet::get(context, 0, AK);
 | 
						|
  Argument *A = F1->arg_begin();
 | 
						|
  A->addAttr(AS);
 | 
						|
 | 
						|
  SmallVector<ReturnInst*, 4> Returns;
 | 
						|
  ValueToValueMapTy VMap;
 | 
						|
  VMap[A] = UndefValue::get(A->getType());
 | 
						|
 | 
						|
  CloneFunctionInto(F2, F1, VMap, false, Returns);
 | 
						|
  EXPECT_FALSE(F2->arg_begin()->hasNoCaptureAttr());
 | 
						|
 | 
						|
  delete F1;
 | 
						|
  delete F2;
 | 
						|
}
 | 
						|
 | 
						|
}
 |