mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-11-04 05:17:07 +00:00 
			
		
		
		
	Register it with PassManager git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@184343 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			173 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
//===-- AMDGPUTargetMachine.cpp - TargetMachine for hw codegen targets-----===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
/// \file
 | 
						|
/// \brief The AMDGPU target machine contains all of the hardware specific
 | 
						|
/// information  needed to emit code for R600 and SI GPUs.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "AMDGPUTargetMachine.h"
 | 
						|
#include "AMDGPU.h"
 | 
						|
#include "R600ISelLowering.h"
 | 
						|
#include "R600InstrInfo.h"
 | 
						|
#include "R600MachineScheduler.h"
 | 
						|
#include "SIISelLowering.h"
 | 
						|
#include "SIInstrInfo.h"
 | 
						|
#include "llvm/Analysis/Passes.h"
 | 
						|
#include "llvm/Analysis/Verifier.h"
 | 
						|
#include "llvm/CodeGen/MachineFunctionAnalysis.h"
 | 
						|
#include "llvm/CodeGen/MachineModuleInfo.h"
 | 
						|
#include "llvm/CodeGen/Passes.h"
 | 
						|
#include "llvm/MC/MCAsmInfo.h"
 | 
						|
#include "llvm/PassManager.h"
 | 
						|
#include "llvm/Support/TargetRegistry.h"
 | 
						|
#include "llvm/Support/raw_os_ostream.h"
 | 
						|
#include "llvm/Transforms/IPO.h"
 | 
						|
#include "llvm/Transforms/Scalar.h"
 | 
						|
#include <llvm/CodeGen/Passes.h>
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
 | 
						|
extern "C" void LLVMInitializeR600Target() {
 | 
						|
  // Register the target
 | 
						|
  RegisterTargetMachine<AMDGPUTargetMachine> X(TheAMDGPUTarget);
 | 
						|
}
 | 
						|
 | 
						|
static ScheduleDAGInstrs *createR600MachineScheduler(MachineSchedContext *C) {
 | 
						|
  return new ScheduleDAGMI(C, new R600SchedStrategy());
 | 
						|
}
 | 
						|
 | 
						|
static MachineSchedRegistry
 | 
						|
SchedCustomRegistry("r600", "Run R600's custom scheduler",
 | 
						|
                    createR600MachineScheduler);
 | 
						|
 | 
						|
AMDGPUTargetMachine::AMDGPUTargetMachine(const Target &T, StringRef TT,
 | 
						|
    StringRef CPU, StringRef FS,
 | 
						|
  TargetOptions Options,
 | 
						|
  Reloc::Model RM, CodeModel::Model CM,
 | 
						|
  CodeGenOpt::Level OptLevel
 | 
						|
)
 | 
						|
:
 | 
						|
  LLVMTargetMachine(T, TT, CPU, FS, Options, RM, CM, OptLevel),
 | 
						|
  Subtarget(TT, CPU, FS),
 | 
						|
  Layout(Subtarget.getDataLayout()),
 | 
						|
  FrameLowering(TargetFrameLowering::StackGrowsUp, 16 // Stack Alignment
 | 
						|
                                                 , 0),
 | 
						|
  IntrinsicInfo(this),
 | 
						|
  InstrItins(&Subtarget.getInstrItineraryData()) {
 | 
						|
  // TLInfo uses InstrInfo so it must be initialized after.
 | 
						|
  if (Subtarget.getGeneration() <= AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    InstrInfo.reset(new R600InstrInfo(*this));
 | 
						|
    TLInfo.reset(new R600TargetLowering(*this));
 | 
						|
  } else {
 | 
						|
    InstrInfo.reset(new SIInstrInfo(*this));
 | 
						|
    TLInfo.reset(new SITargetLowering(*this));
 | 
						|
  }
 | 
						|
  initAsmInfo();
 | 
						|
}
 | 
						|
 | 
						|
AMDGPUTargetMachine::~AMDGPUTargetMachine() {
 | 
						|
}
 | 
						|
 | 
						|
namespace {
 | 
						|
class AMDGPUPassConfig : public TargetPassConfig {
 | 
						|
public:
 | 
						|
  AMDGPUPassConfig(AMDGPUTargetMachine *TM, PassManagerBase &PM)
 | 
						|
    : TargetPassConfig(TM, PM) {
 | 
						|
    const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
    if (ST.getGeneration() <= AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
      enablePass(&MachineSchedulerID);
 | 
						|
      MachineSchedRegistry::setDefault(createR600MachineScheduler);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  AMDGPUTargetMachine &getAMDGPUTargetMachine() const {
 | 
						|
    return getTM<AMDGPUTargetMachine>();
 | 
						|
  }
 | 
						|
 | 
						|
  virtual bool addPreISel();
 | 
						|
  virtual bool addInstSelector();
 | 
						|
  virtual bool addPreRegAlloc();
 | 
						|
  virtual bool addPostRegAlloc();
 | 
						|
  virtual bool addPreSched2();
 | 
						|
  virtual bool addPreEmitPass();
 | 
						|
};
 | 
						|
} // End of anonymous namespace
 | 
						|
 | 
						|
TargetPassConfig *AMDGPUTargetMachine::createPassConfig(PassManagerBase &PM) {
 | 
						|
  return new AMDGPUPassConfig(this, PM);
 | 
						|
}
 | 
						|
 | 
						|
bool
 | 
						|
AMDGPUPassConfig::addPreISel() {
 | 
						|
  const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
  if (ST.getGeneration() > AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    addPass(createStructurizeCFGPass());
 | 
						|
    addPass(createSIAnnotateControlFlowPass());
 | 
						|
  } else {
 | 
						|
    addPass(createR600TextureIntrinsicsReplacer());
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool AMDGPUPassConfig::addInstSelector() {
 | 
						|
  addPass(createAMDGPUISelDag(getAMDGPUTargetMachine()));
 | 
						|
 | 
						|
  const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
  if (ST.getGeneration() <= AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    // This callbacks this pass uses are not implemented yet on SI.
 | 
						|
    addPass(createAMDGPUIndirectAddressingPass(*TM));
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool AMDGPUPassConfig::addPreRegAlloc() {
 | 
						|
  addPass(createAMDGPUConvertToISAPass(*TM));
 | 
						|
  const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
 | 
						|
  if (ST.getGeneration() <= AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    addPass(createR600VectorRegMerger(*TM));
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool AMDGPUPassConfig::addPostRegAlloc() {
 | 
						|
  const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
 | 
						|
  if (ST.getGeneration() > AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    addPass(createSIInsertWaits(*TM));
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool AMDGPUPassConfig::addPreSched2() {
 | 
						|
 | 
						|
  addPass(&IfConverterID);
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool AMDGPUPassConfig::addPreEmitPass() {
 | 
						|
  const AMDGPUSubtarget &ST = TM->getSubtarget<AMDGPUSubtarget>();
 | 
						|
  if (ST.getGeneration() <= AMDGPUSubtarget::NORTHERN_ISLANDS) {
 | 
						|
    addPass(createAMDGPUCFGPreparationPass(*TM));
 | 
						|
    addPass(createAMDGPUCFGStructurizerPass(*TM));
 | 
						|
    addPass(createR600EmitClauseMarkers(*TM));
 | 
						|
    addPass(createR600ExpandSpecialInstrsPass(*TM));
 | 
						|
    addPass(&FinalizeMachineBundlesID);
 | 
						|
    addPass(createR600Packetizer(*TM));
 | 
						|
    addPass(createR600ControlFlowFinalizer(*TM));
 | 
						|
  } else {
 | 
						|
    addPass(createSILowerControlFlowPass(*TM));
 | 
						|
  }
 | 
						|
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 |