//===- IntegerDivision.cpp - Unit tests for the integer division code -----===// // // 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/Transforms/Utils/IntegerDivision.h" #include "llvm/IR/BasicBlock.h" #include "llvm/IR/Function.h" #include "llvm/IR/GlobalValue.h" #include "llvm/IR/IRBuilder.h" #include "llvm/IR/Module.h" #include "gtest/gtest.h" using namespace llvm; namespace { TEST(IntegerDivision, SDiv) { LLVMContext C; Module M("test division", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt32Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt32Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Div = Builder.CreateSDiv(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::SDiv); Value *Ret = Builder.CreateRet(Div); expandDivision(cast(Div)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::AShr); Instruction* Quotient = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Quotient && Quotient->getOpcode() == Instruction::Sub); } TEST(IntegerDivision, UDiv) { LLVMContext C; Module M("test division", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt32Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt32Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Div = Builder.CreateUDiv(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::UDiv); Value *Ret = Builder.CreateRet(Div); expandDivision(cast(Div)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::ICmp); Instruction* Quotient = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Quotient && Quotient->getOpcode() == Instruction::PHI); } TEST(IntegerDivision, SRem) { LLVMContext C; Module M("test remainder", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt32Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt32Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Rem = Builder.CreateSRem(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::SRem); Value *Ret = Builder.CreateRet(Rem); expandRemainder(cast(Rem)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::AShr); Instruction* Remainder = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Remainder && Remainder->getOpcode() == Instruction::Sub); } TEST(IntegerDivision, URem) { LLVMContext C; Module M("test remainder", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt32Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt32Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Rem = Builder.CreateURem(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::URem); Value *Ret = Builder.CreateRet(Rem); expandRemainder(cast(Rem)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::ICmp); Instruction* Remainder = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Remainder && Remainder->getOpcode() == Instruction::Sub); } TEST(IntegerDivision, SDiv64) { LLVMContext C; Module M("test division", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt64Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt64Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Div = Builder.CreateSDiv(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::SDiv); Value *Ret = Builder.CreateRet(Div); expandDivision(cast(Div)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::AShr); Instruction* Quotient = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Quotient && Quotient->getOpcode() == Instruction::Sub); } TEST(IntegerDivision, UDiv64) { LLVMContext C; Module M("test division", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt64Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt64Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Div = Builder.CreateUDiv(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::UDiv); Value *Ret = Builder.CreateRet(Div); expandDivision(cast(Div)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::ICmp); Instruction* Quotient = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Quotient && Quotient->getOpcode() == Instruction::PHI); } TEST(IntegerDivision, SRem64) { LLVMContext C; Module M("test remainder", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt64Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt64Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Rem = Builder.CreateSRem(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::SRem); Value *Ret = Builder.CreateRet(Rem); expandRemainder(cast(Rem)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::AShr); Instruction* Remainder = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Remainder && Remainder->getOpcode() == Instruction::Sub); } TEST(IntegerDivision, URem64) { LLVMContext C; Module M("test remainder", C); IRBuilder<> Builder(C); SmallVector ArgTys(2, Builder.getInt64Ty()); Function *F = Function::Create(FunctionType::get(Builder.getInt64Ty(), ArgTys, false), GlobalValue::ExternalLinkage, "F", &M); assert(F->arg_size() == 2); BasicBlock *BB = BasicBlock::Create(C, "", F); Builder.SetInsertPoint(BB); Function::arg_iterator AI = F->arg_begin(); Value *A = &*AI++; Value *B = &*AI++; Value *Rem = Builder.CreateURem(A, B); EXPECT_TRUE(BB->front().getOpcode() == Instruction::URem); Value *Ret = Builder.CreateRet(Rem); expandRemainder(cast(Rem)); EXPECT_TRUE(BB->front().getOpcode() == Instruction::ICmp); Instruction* Remainder = dyn_cast(cast(Ret)->getOperand(0)); EXPECT_TRUE(Remainder && Remainder->getOpcode() == Instruction::Sub); } }