forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			324 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			324 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- CGBuilder.h - Choose IRBuilder implementation  ----------*- C++ -*-===//
 | |
| //
 | |
| // 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
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #ifndef LLVM_CLANG_LIB_CODEGEN_CGBUILDER_H
 | |
| #define LLVM_CLANG_LIB_CODEGEN_CGBUILDER_H
 | |
| 
 | |
| #include "llvm/IR/DataLayout.h"
 | |
| #include "llvm/IR/IRBuilder.h"
 | |
| #include "Address.h"
 | |
| #include "CodeGenTypeCache.h"
 | |
| 
 | |
| namespace clang {
 | |
| namespace CodeGen {
 | |
| 
 | |
| class CodeGenFunction;
 | |
| 
 | |
| /// This is an IRBuilder insertion helper that forwards to
 | |
| /// CodeGenFunction::InsertHelper, which adds necessary metadata to
 | |
| /// instructions.
 | |
| class CGBuilderInserter final : public llvm::IRBuilderDefaultInserter {
 | |
| public:
 | |
|   CGBuilderInserter() = default;
 | |
|   explicit CGBuilderInserter(CodeGenFunction *CGF) : CGF(CGF) {}
 | |
| 
 | |
|   /// This forwards to CodeGenFunction::InsertHelper.
 | |
|   void InsertHelper(llvm::Instruction *I, const llvm::Twine &Name,
 | |
|                     llvm::BasicBlock *BB,
 | |
|                     llvm::BasicBlock::iterator InsertPt) const override;
 | |
| private:
 | |
|   CodeGenFunction *CGF = nullptr;
 | |
| };
 | |
| 
 | |
| typedef CGBuilderInserter CGBuilderInserterTy;
 | |
| 
 | |
| typedef llvm::IRBuilder<llvm::ConstantFolder, CGBuilderInserterTy>
 | |
|     CGBuilderBaseTy;
 | |
| 
 | |
| class CGBuilderTy : public CGBuilderBaseTy {
 | |
|   /// Storing a reference to the type cache here makes it a lot easier
 | |
|   /// to build natural-feeling, target-specific IR.
 | |
|   const CodeGenTypeCache &TypeCache;
 | |
| public:
 | |
|   CGBuilderTy(const CodeGenTypeCache &TypeCache, llvm::LLVMContext &C)
 | |
|     : CGBuilderBaseTy(C), TypeCache(TypeCache) {}
 | |
|   CGBuilderTy(const CodeGenTypeCache &TypeCache,
 | |
|               llvm::LLVMContext &C, const llvm::ConstantFolder &F,
 | |
|               const CGBuilderInserterTy &Inserter)
 | |
|     : CGBuilderBaseTy(C, F, Inserter), TypeCache(TypeCache) {}
 | |
|   CGBuilderTy(const CodeGenTypeCache &TypeCache, llvm::Instruction *I)
 | |
|     : CGBuilderBaseTy(I), TypeCache(TypeCache) {}
 | |
|   CGBuilderTy(const CodeGenTypeCache &TypeCache, llvm::BasicBlock *BB)
 | |
|     : CGBuilderBaseTy(BB), TypeCache(TypeCache) {}
 | |
| 
 | |
|   llvm::ConstantInt *getSize(CharUnits N) {
 | |
|     return llvm::ConstantInt::get(TypeCache.SizeTy, N.getQuantity());
 | |
|   }
 | |
|   llvm::ConstantInt *getSize(uint64_t N) {
 | |
|     return llvm::ConstantInt::get(TypeCache.SizeTy, N);
 | |
|   }
 | |
| 
 | |
|   // Note that we intentionally hide the CreateLoad APIs that don't
 | |
|   // take an alignment.
 | |
|   llvm::LoadInst *CreateLoad(Address Addr, const llvm::Twine &Name = "") {
 | |
|     return CreateAlignedLoad(Addr.getPointer(),
 | |
|                              Addr.getAlignment().getAsAlign(), Name);
 | |
|   }
 | |
|   llvm::LoadInst *CreateLoad(Address Addr, const char *Name) {
 | |
|     // This overload is required to prevent string literals from
 | |
|     // ending up in the IsVolatile overload.
 | |
|     return CreateAlignedLoad(Addr.getPointer(),
 | |
|                              Addr.getAlignment().getAsAlign(), Name);
 | |
|   }
 | |
|   llvm::LoadInst *CreateLoad(Address Addr, bool IsVolatile,
 | |
|                              const llvm::Twine &Name = "") {
 | |
|     return CreateAlignedLoad(
 | |
|         Addr.getPointer(), Addr.getAlignment().getAsAlign(), IsVolatile, Name);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateAlignedLoad;
 | |
|   llvm::LoadInst *CreateAlignedLoad(llvm::Value *Addr, CharUnits Align,
 | |
|                                     const llvm::Twine &Name = "") {
 | |
|     return CreateAlignedLoad(Addr, Align.getAsAlign(), Name);
 | |
|   }
 | |
|   llvm::LoadInst *CreateAlignedLoad(llvm::Value *Addr, CharUnits Align,
 | |
|                                     const char *Name) {
 | |
|     return CreateAlignedLoad(Addr, Align.getAsAlign(), Name);
 | |
|   }
 | |
|   llvm::LoadInst *CreateAlignedLoad(llvm::Type *Ty, llvm::Value *Addr,
 | |
|                                     CharUnits Align,
 | |
|                                     const llvm::Twine &Name = "") {
 | |
|     assert(Addr->getType()->getPointerElementType() == Ty);
 | |
|     return CreateAlignedLoad(Addr, Align.getAsAlign(), Name);
 | |
|   }
 | |
| 
 | |
|   // Note that we intentionally hide the CreateStore APIs that don't
 | |
|   // take an alignment.
 | |
|   llvm::StoreInst *CreateStore(llvm::Value *Val, Address Addr,
 | |
|                                bool IsVolatile = false) {
 | |
|     return CreateAlignedStore(Val, Addr.getPointer(),
 | |
|                               Addr.getAlignment().getAsAlign(), IsVolatile);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateAlignedStore;
 | |
|   llvm::StoreInst *CreateAlignedStore(llvm::Value *Val, llvm::Value *Addr,
 | |
|                                       CharUnits Align, bool IsVolatile = false) {
 | |
|     return CreateAlignedStore(Val, Addr, Align.getAsAlign(), IsVolatile);
 | |
|   }
 | |
| 
 | |
|   // FIXME: these "default-aligned" APIs should be removed,
 | |
|   // but I don't feel like fixing all the builtin code right now.
 | |
|   llvm::StoreInst *CreateDefaultAlignedStore(llvm::Value *Val,
 | |
|                                              llvm::Value *Addr,
 | |
|                                              bool IsVolatile = false) {
 | |
|     return CGBuilderBaseTy::CreateStore(Val, Addr, IsVolatile);
 | |
|   }
 | |
| 
 | |
|   /// Emit a load from an i1 flag variable.
 | |
|   llvm::LoadInst *CreateFlagLoad(llvm::Value *Addr,
 | |
|                                  const llvm::Twine &Name = "") {
 | |
|     assert(Addr->getType()->getPointerElementType() == getInt1Ty());
 | |
|     return CreateAlignedLoad(getInt1Ty(), Addr, CharUnits::One(), Name);
 | |
|   }
 | |
| 
 | |
|   /// Emit a store to an i1 flag variable.
 | |
|   llvm::StoreInst *CreateFlagStore(bool Value, llvm::Value *Addr) {
 | |
|     assert(Addr->getType()->getPointerElementType() == getInt1Ty());
 | |
|     return CreateAlignedStore(getInt1(Value), Addr, CharUnits::One());
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateBitCast;
 | |
|   Address CreateBitCast(Address Addr, llvm::Type *Ty,
 | |
|                         const llvm::Twine &Name = "") {
 | |
|     return Address(CreateBitCast(Addr.getPointer(), Ty, Name),
 | |
|                    Addr.getAlignment());
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateAddrSpaceCast;
 | |
|   Address CreateAddrSpaceCast(Address Addr, llvm::Type *Ty,
 | |
|                               const llvm::Twine &Name = "") {
 | |
|     return Address(CreateAddrSpaceCast(Addr.getPointer(), Ty, Name),
 | |
|                    Addr.getAlignment());
 | |
|   }
 | |
| 
 | |
|   /// Cast the element type of the given address to a different type,
 | |
|   /// preserving information like the alignment and address space.
 | |
|   Address CreateElementBitCast(Address Addr, llvm::Type *Ty,
 | |
|                                const llvm::Twine &Name = "") {
 | |
|     auto PtrTy = Ty->getPointerTo(Addr.getAddressSpace());
 | |
|     return CreateBitCast(Addr, PtrTy, Name);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreatePointerBitCastOrAddrSpaceCast;
 | |
|   Address CreatePointerBitCastOrAddrSpaceCast(Address Addr, llvm::Type *Ty,
 | |
|                                               const llvm::Twine &Name = "") {
 | |
|     llvm::Value *Ptr =
 | |
|       CreatePointerBitCastOrAddrSpaceCast(Addr.getPointer(), Ty, Name);
 | |
|     return Address(Ptr, Addr.getAlignment());
 | |
|   }
 | |
| 
 | |
|   /// Given
 | |
|   ///   %addr = {T1, T2...}* ...
 | |
|   /// produce
 | |
|   ///   %name = getelementptr inbounds %addr, i32 0, i32 index
 | |
|   ///
 | |
|   /// This API assumes that drilling into a struct like this is always an
 | |
|   /// inbounds operation.
 | |
|   using CGBuilderBaseTy::CreateStructGEP;
 | |
|   Address CreateStructGEP(Address Addr, unsigned Index,
 | |
|                           const llvm::Twine &Name = "") {
 | |
|     llvm::StructType *ElTy = cast<llvm::StructType>(Addr.getElementType());
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
|     const llvm::StructLayout *Layout = DL.getStructLayout(ElTy);
 | |
|     auto Offset = CharUnits::fromQuantity(Layout->getElementOffset(Index));
 | |
| 
 | |
|     return Address(CreateStructGEP(Addr.getElementType(),
 | |
|                                    Addr.getPointer(), Index, Name),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Offset));
 | |
|   }
 | |
| 
 | |
|   /// Given
 | |
|   ///   %addr = [n x T]* ...
 | |
|   /// produce
 | |
|   ///   %name = getelementptr inbounds %addr, i64 0, i64 index
 | |
|   /// where i64 is actually the target word size.
 | |
|   ///
 | |
|   /// This API assumes that drilling into an array like this is always
 | |
|   /// an inbounds operation.
 | |
|   Address CreateConstArrayGEP(Address Addr, uint64_t Index,
 | |
|                               const llvm::Twine &Name = "") {
 | |
|     llvm::ArrayType *ElTy = cast<llvm::ArrayType>(Addr.getElementType());
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
|     CharUnits EltSize =
 | |
|         CharUnits::fromQuantity(DL.getTypeAllocSize(ElTy->getElementType()));
 | |
| 
 | |
|     return Address(
 | |
|         CreateInBoundsGEP(Addr.getPointer(),
 | |
|                           {getSize(CharUnits::Zero()), getSize(Index)}, Name),
 | |
|         Addr.getAlignment().alignmentAtOffset(Index * EltSize));
 | |
|   }
 | |
| 
 | |
|   /// Given
 | |
|   ///   %addr = T* ...
 | |
|   /// produce
 | |
|   ///   %name = getelementptr inbounds %addr, i64 index
 | |
|   /// where i64 is actually the target word size.
 | |
|   Address CreateConstInBoundsGEP(Address Addr, uint64_t Index,
 | |
|                                  const llvm::Twine &Name = "") {
 | |
|     llvm::Type *ElTy = Addr.getElementType();
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
|     CharUnits EltSize = CharUnits::fromQuantity(DL.getTypeAllocSize(ElTy));
 | |
| 
 | |
|     return Address(CreateInBoundsGEP(Addr.getElementType(), Addr.getPointer(),
 | |
|                                      getSize(Index), Name),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Index * EltSize));
 | |
|   }
 | |
| 
 | |
|   /// Given
 | |
|   ///   %addr = T* ...
 | |
|   /// produce
 | |
|   ///   %name = getelementptr inbounds %addr, i64 index
 | |
|   /// where i64 is actually the target word size.
 | |
|   Address CreateConstGEP(Address Addr, uint64_t Index,
 | |
|                          const llvm::Twine &Name = "") {
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
|     CharUnits EltSize =
 | |
|         CharUnits::fromQuantity(DL.getTypeAllocSize(Addr.getElementType()));
 | |
| 
 | |
|     return Address(CreateGEP(Addr.getElementType(), Addr.getPointer(),
 | |
|                              getSize(Index), Name),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Index * EltSize));
 | |
|   }
 | |
| 
 | |
|   /// Given a pointer to i8, adjust it by a given constant offset.
 | |
|   Address CreateConstInBoundsByteGEP(Address Addr, CharUnits Offset,
 | |
|                                      const llvm::Twine &Name = "") {
 | |
|     assert(Addr.getElementType() == TypeCache.Int8Ty);
 | |
|     return Address(CreateInBoundsGEP(Addr.getPointer(), getSize(Offset), Name),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Offset));
 | |
|   }
 | |
|   Address CreateConstByteGEP(Address Addr, CharUnits Offset,
 | |
|                              const llvm::Twine &Name = "") {
 | |
|     assert(Addr.getElementType() == TypeCache.Int8Ty);
 | |
|     return Address(CreateGEP(Addr.getPointer(), getSize(Offset), Name),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Offset));
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateConstInBoundsGEP2_32;
 | |
|   Address CreateConstInBoundsGEP2_32(Address Addr, unsigned Idx0, unsigned Idx1,
 | |
|                                      const llvm::Twine &Name = "") {
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
| 
 | |
|     auto *GEP = cast<llvm::GetElementPtrInst>(CreateConstInBoundsGEP2_32(
 | |
|         Addr.getElementType(), Addr.getPointer(), Idx0, Idx1, Name));
 | |
|     llvm::APInt Offset(
 | |
|         DL.getIndexSizeInBits(Addr.getType()->getPointerAddressSpace()), 0,
 | |
|         /*isSigned=*/true);
 | |
|     if (!GEP->accumulateConstantOffset(DL, Offset))
 | |
|       llvm_unreachable("offset of GEP with constants is always computable");
 | |
|     return Address(GEP, Addr.getAlignment().alignmentAtOffset(
 | |
|                             CharUnits::fromQuantity(Offset.getSExtValue())));
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateMemCpy;
 | |
|   llvm::CallInst *CreateMemCpy(Address Dest, Address Src, llvm::Value *Size,
 | |
|                                bool IsVolatile = false) {
 | |
|     return CreateMemCpy(Dest.getPointer(), Dest.getAlignment().getAsAlign(),
 | |
|                         Src.getPointer(), Src.getAlignment().getAsAlign(), Size,
 | |
|                         IsVolatile);
 | |
|   }
 | |
|   llvm::CallInst *CreateMemCpy(Address Dest, Address Src, uint64_t Size,
 | |
|                                bool IsVolatile = false) {
 | |
|     return CreateMemCpy(Dest.getPointer(), Dest.getAlignment().getAsAlign(),
 | |
|                         Src.getPointer(), Src.getAlignment().getAsAlign(), Size,
 | |
|                         IsVolatile);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateMemCpyInline;
 | |
|   llvm::CallInst *CreateMemCpyInline(Address Dest, Address Src, uint64_t Size) {
 | |
|     return CreateMemCpyInline(
 | |
|         Dest.getPointer(), Dest.getAlignment().getAsAlign(), Src.getPointer(),
 | |
|         Src.getAlignment().getAsAlign(), getInt64(Size));
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateMemMove;
 | |
|   llvm::CallInst *CreateMemMove(Address Dest, Address Src, llvm::Value *Size,
 | |
|                                 bool IsVolatile = false) {
 | |
|     return CreateMemMove(Dest.getPointer(), Dest.getAlignment().getAsAlign(),
 | |
|                          Src.getPointer(), Src.getAlignment().getAsAlign(),
 | |
|                          Size, IsVolatile);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreateMemSet;
 | |
|   llvm::CallInst *CreateMemSet(Address Dest, llvm::Value *Value,
 | |
|                                llvm::Value *Size, bool IsVolatile = false) {
 | |
|     return CreateMemSet(Dest.getPointer(), Value, Size,
 | |
|                         Dest.getAlignment().getAsAlign(), IsVolatile);
 | |
|   }
 | |
| 
 | |
|   using CGBuilderBaseTy::CreatePreserveStructAccessIndex;
 | |
|   Address CreatePreserveStructAccessIndex(Address Addr,
 | |
|                                           unsigned Index,
 | |
|                                           unsigned FieldIndex,
 | |
|                                           llvm::MDNode *DbgInfo) {
 | |
|     llvm::StructType *ElTy = cast<llvm::StructType>(Addr.getElementType());
 | |
|     const llvm::DataLayout &DL = BB->getParent()->getParent()->getDataLayout();
 | |
|     const llvm::StructLayout *Layout = DL.getStructLayout(ElTy);
 | |
|     auto Offset = CharUnits::fromQuantity(Layout->getElementOffset(Index));
 | |
| 
 | |
|     return Address(CreatePreserveStructAccessIndex(ElTy, Addr.getPointer(),
 | |
|                                                    Index, FieldIndex, DbgInfo),
 | |
|                    Addr.getAlignment().alignmentAtOffset(Offset));
 | |
|   }
 | |
| };
 | |
| 
 | |
| }  // end namespace CodeGen
 | |
| }  // end namespace clang
 | |
| 
 | |
| #endif
 |