|  | @@ -29,6 +29,7 @@
 | 
	
		
			
				|  |  |  #include "llvm/Transforms/Utils/Local.h"
 | 
	
		
			
				|  |  |  #include <memory>
 | 
	
		
			
				|  |  |  #include <unordered_set>
 | 
	
		
			
				|  |  | +#include <array>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  using namespace llvm;
 | 
	
		
			
				|  |  |  using namespace hlsl;
 | 
	
	
		
			
				|  | @@ -37,17 +38,21 @@ class DxilOutputColorBecomesConstant : public ModulePass {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    enum VisualizerInstrumentationMode
 | 
	
		
			
				|  |  |    {
 | 
	
		
			
				|  |  | -    PRESERVE_ORIGINAL_INSTRUCTIONS,
 | 
	
		
			
				|  |  | -    REMOVE_DISCARDS_AND_OPTIONALLY_OTHER_INSTRUCTIONS
 | 
	
		
			
				|  |  | +    FromLiteralConstant,
 | 
	
		
			
				|  |  | +    FromConstantBuffer
 | 
	
		
			
				|  |  |    };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    float Red = 1.f;
 | 
	
		
			
				|  |  |    float Green = 1.f;
 | 
	
		
			
				|  |  |    float Blue = 1.f;
 | 
	
		
			
				|  |  |    float Alpha = 1.f;
 | 
	
		
			
				|  |  | -  VisualizerInstrumentationMode Mode;
 | 
	
		
			
				|  |  | +  VisualizerInstrumentationMode Mode = FromLiteralConstant;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  bool convertTarget0ToConstantValue(Function * OutputFunction, const hlsl::DxilSignature &OutputSignature, OP * HlslOP, float * color);
 | 
	
		
			
				|  |  | +  void visitOutputInstructionCallers(
 | 
	
		
			
				|  |  | +    Function * OutputFunction, 
 | 
	
		
			
				|  |  | +    const hlsl::DxilSignature &OutputSignature, 
 | 
	
		
			
				|  |  | +    OP * HlslOP, 
 | 
	
		
			
				|  |  | +    std::function<void(CallInst*)> Visitor);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  public:
 | 
	
		
			
				|  |  |    static char ID; // Pass identification, replacement for typeid
 | 
	
	
		
			
				|  | @@ -84,55 +89,35 @@ void DxilOutputColorBecomesConstant::applyOptions(PassOptions O)
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -bool DxilOutputColorBecomesConstant::convertTarget0ToConstantValue(
 | 
	
		
			
				|  |  | -  Function * OutputFunction, 
 | 
	
		
			
				|  |  | -  const hlsl::DxilSignature &OutputSignature, 
 | 
	
		
			
				|  |  | -  OP * HlslOP, 
 | 
	
		
			
				|  |  | -  float * color) {
 | 
	
		
			
				|  |  | +void DxilOutputColorBecomesConstant::visitOutputInstructionCallers(
 | 
	
		
			
				|  |  | +  Function * OutputFunction,
 | 
	
		
			
				|  |  | +  const hlsl::DxilSignature &OutputSignature,
 | 
	
		
			
				|  |  | +  OP * HlslOP,
 | 
	
		
			
				|  |  | +  std::function<void(CallInst*)> Visitor) {
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  bool Modified = false;
 | 
	
		
			
				|  |  | -    auto OutputFunctionUses = OutputFunction->uses();
 | 
	
		
			
				|  |  | +  auto OutputFunctionUses = OutputFunction->uses();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    for (Use &FunctionUse : OutputFunctionUses) {
 | 
	
		
			
				|  |  | -      iterator_range<Value::user_iterator> FunctionUsers = FunctionUse->users();
 | 
	
		
			
				|  |  | -      for (User * FunctionUser : FunctionUsers) {
 | 
	
		
			
				|  |  | -        if (isa<Instruction>(FunctionUser)) {
 | 
	
		
			
				|  |  | -          auto CallInstruction = cast<CallInst>(FunctionUser);
 | 
	
		
			
				|  |  | +  for (Use &FunctionUse : OutputFunctionUses) {
 | 
	
		
			
				|  |  | +    iterator_range<Value::user_iterator> FunctionUsers = FunctionUse->users();
 | 
	
		
			
				|  |  | +    for (User * FunctionUser : FunctionUsers) {
 | 
	
		
			
				|  |  | +      if (isa<Instruction>(FunctionUser)) {
 | 
	
		
			
				|  |  | +        auto CallInstruction = cast<CallInst>(FunctionUser);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -          // Check if the instruction writes to a render target (as opposed to a system-value, such as RenderTargetArrayIndex)
 | 
	
		
			
				|  |  | -          Value *OutputID = CallInstruction->getArgOperand(DXIL::OperandIndex::kStoreOutputIDOpIdx);
 | 
	
		
			
				|  |  | -          unsigned SignatureElementIndex = cast<ConstantInt>(OutputID)->getLimitedValue();
 | 
	
		
			
				|  |  | -          const DxilSignatureElement &SignatureElement = OutputSignature.GetElement(SignatureElementIndex);
 | 
	
		
			
				|  |  | +        // Check if the instruction writes to a render target (as opposed to a system-value, such as RenderTargetArrayIndex)
 | 
	
		
			
				|  |  | +        Value *OutputID = CallInstruction->getArgOperand(DXIL::OperandIndex::kStoreOutputIDOpIdx);
 | 
	
		
			
				|  |  | +        unsigned SignatureElementIndex = cast<ConstantInt>(OutputID)->getLimitedValue();
 | 
	
		
			
				|  |  | +        const DxilSignatureElement &SignatureElement = OutputSignature.GetElement(SignatureElementIndex);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |          // We only modify the output color for RTV0
 | 
	
		
			
				|  |  |          if (SignatureElement.GetSemantic()->GetKind() == DXIL::SemanticKind::Target &&
 | 
	
		
			
				|  |  | -          SignatureElement.GetSemanticStartIndex() == 0)
 | 
	
		
			
				|  |  | -          {
 | 
	
		
			
				|  |  | -            // The output column is the channel (red, green, blue or alpha) within the output pixel
 | 
	
		
			
				|  |  | -            Value * OutputColumnOperand = CallInstruction->getOperand(hlsl::DXIL::OperandIndex::kStoreOutputColOpIdx);
 | 
	
		
			
				|  |  | -            ConstantInt * OutputColumnConstant = cast<ConstantInt>(OutputColumnOperand);
 | 
	
		
			
				|  |  | -            APInt OutputColumn = OutputColumnConstant->getValue();
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -            Value * OutputValueOperand = CallInstruction->getOperand(hlsl::DXIL::OperandIndex::kStoreOutputValOpIdx);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -            // Replace the source operand with the appropriate constant literal value
 | 
	
		
			
				|  |  | -          if (OutputValueOperand->getType()->isFloatingPointTy())
 | 
	
		
			
				|  |  | -            {
 | 
	
		
			
				|  |  | -            Modified = true;
 | 
	
		
			
				|  |  | -              Constant * FloatConstant = HlslOP->GetFloatConst(color[*OutputColumn.getRawData()]);
 | 
	
		
			
				|  |  | -              CallInstruction->setOperand(hlsl::DXIL::OperandIndex::kStoreOutputValOpIdx, FloatConstant);
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -          else if (OutputValueOperand->getType()->isIntegerTy())
 | 
	
		
			
				|  |  | -            {
 | 
	
		
			
				|  |  | -            Modified = true;
 | 
	
		
			
				|  |  | -              Constant * pIntegerConstant = HlslOP->GetI32Const(static_cast<int>(color[*OutputColumn.getRawData()]));
 | 
	
		
			
				|  |  | -              CallInstruction->setOperand(hlsl::DXIL::OperandIndex::kStoreOutputValOpIdx, pIntegerConstant);
 | 
	
		
			
				|  |  | -            }
 | 
	
		
			
				|  |  | -          }
 | 
	
		
			
				|  |  | +            SignatureElement.GetSemanticStartIndex() == 0) {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +          // Replace the source operand with the appropriate constant value
 | 
	
		
			
				|  |  | +          Visitor(CallInstruction);
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  |        }
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  | -  return Modified;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  bool DxilOutputColorBecomesConstant::runOnModule(Module &M)
 | 
	
	
		
			
				|  | @@ -140,8 +125,6 @@ bool DxilOutputColorBecomesConstant::runOnModule(Module &M)
 | 
	
		
			
				|  |  |    // This pass finds all users of the "StoreOutput" function, and replaces their source operands with a constant
 | 
	
		
			
				|  |  |    // value. 
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  float color[4] = { Red, Green, Blue, Alpha };
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |    DxilModule &DM = M.GetOrCreateDxilModule();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    LLVMContext & Ctx = M.getContext();
 | 
	
	
		
			
				|  | @@ -150,18 +133,125 @@ bool DxilOutputColorBecomesConstant::runOnModule(Module &M)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    const hlsl::DxilSignature & OutputSignature = DM.GetOutputSignature();
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +  Function * FloatOutputFunction = HlslOP->GetOpFunc(DXIL::OpCode::StoreOutput, Type::getFloatTy(Ctx));
 | 
	
		
			
				|  |  | +  Function * IntOutputFunction = HlslOP->GetOpFunc(DXIL::OpCode::StoreOutput, Type::getInt32Ty(Ctx));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  bool hasFloatOutputs = false;
 | 
	
		
			
				|  |  | +  bool hasIntOutputs = false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  visitOutputInstructionCallers(FloatOutputFunction, OutputSignature, HlslOP, [&hasFloatOutputs](CallInst *) {
 | 
	
		
			
				|  |  | +    hasFloatOutputs = true;
 | 
	
		
			
				|  |  | +  });
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  visitOutputInstructionCallers(IntOutputFunction, OutputSignature, HlslOP, [&hasIntOutputs](CallInst *) {
 | 
	
		
			
				|  |  | +    hasIntOutputs = true;
 | 
	
		
			
				|  |  | +  });
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if (!hasFloatOutputs && !hasIntOutputs)
 | 
	
		
			
				|  |  | +  {
 | 
	
		
			
				|  |  | +    return false;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  // Otherwise, we assume the shader outputs only one or the other (because the 0th RTV can't have a mixed type)
 | 
	
		
			
				|  |  | +  DXASSERT(!hasFloatOutputs || !hasIntOutputs, "Only one or the other type of output: float or int");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  std::array<llvm::Value *, 4> ReplacementColors;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  switch (Mode)
 | 
	
		
			
				|  |  | +  {
 | 
	
		
			
				|  |  | +    case FromLiteralConstant: {
 | 
	
		
			
				|  |  | +      if (hasFloatOutputs) {
 | 
	
		
			
				|  |  | +        ReplacementColors[0] = HlslOP->GetFloatConst(Red);
 | 
	
		
			
				|  |  | +        ReplacementColors[1] = HlslOP->GetFloatConst(Green);
 | 
	
		
			
				|  |  | +        ReplacementColors[2] = HlslOP->GetFloatConst(Blue);
 | 
	
		
			
				|  |  | +        ReplacementColors[3] = HlslOP->GetFloatConst(Alpha);
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      if (hasIntOutputs) {
 | 
	
		
			
				|  |  | +        ReplacementColors[0] = HlslOP->GetI32Const(static_cast<int>(Red));
 | 
	
		
			
				|  |  | +        ReplacementColors[1] = HlslOP->GetI32Const(static_cast<int>(Green));
 | 
	
		
			
				|  |  | +        ReplacementColors[2] = HlslOP->GetI32Const(static_cast<int>(Blue));
 | 
	
		
			
				|  |  | +        ReplacementColors[3] = HlslOP->GetI32Const(static_cast<int>(Alpha));
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    break;
 | 
	
		
			
				|  |  | +    case FromConstantBuffer: {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // Setup a constant buffer with a single float4 in it:
 | 
	
		
			
				|  |  | +      SmallVector<llvm::Type*, 4> Elements { Type::getFloatTy(Ctx), Type::getFloatTy(Ctx) , Type::getFloatTy(Ctx) , Type::getFloatTy(Ctx) };
 | 
	
		
			
				|  |  | +      llvm::StructType *CBStructTy = llvm::StructType::create(Elements, "PIX_ConstantColorCB_Type");
 | 
	
		
			
				|  |  | +      std::unique_ptr<DxilCBuffer> pCBuf = llvm::make_unique<DxilCBuffer>();
 | 
	
		
			
				|  |  | +      pCBuf->SetGlobalName("PIX_ConstantColorCBName");
 | 
	
		
			
				|  |  | +      pCBuf->SetGlobalSymbol(UndefValue::get(CBStructTy));
 | 
	
		
			
				|  |  | +      pCBuf->SetID(0);
 | 
	
		
			
				|  |  | +      pCBuf->SetSpaceID((unsigned int)-2); // This is the reserved-for-tools register space
 | 
	
		
			
				|  |  | +      pCBuf->SetLowerBound(0);
 | 
	
		
			
				|  |  | +      pCBuf->SetRangeSize(1);
 | 
	
		
			
				|  |  | +      pCBuf->SetSize(4);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      ID = DM.AddCBuffer(std::move(pCBuf));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      Instruction * entryPointInstruction = &*(DM.GetEntryFunction()->begin()->begin());
 | 
	
		
			
				|  |  | +      IRBuilder<> Builder(entryPointInstruction);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // Create handle for the newly-added constant buffer (which is achieved via a function call)
 | 
	
		
			
				|  |  | +      auto ConstantBufferName = "PIX_Constant_Color_CB_Handle";
 | 
	
		
			
				|  |  | +      Function *createHandle = HlslOP->GetOpFunc(DXIL::OpCode::CreateHandle, Type::getVoidTy(Ctx));
 | 
	
		
			
				|  |  | +      Constant *CreateHandleOpcodeArg = HlslOP->GetU32Const((unsigned)DXIL::OpCode::CreateHandle);
 | 
	
		
			
				|  |  | +      Constant *CBVArg = HlslOP->GetI8Const(static_cast<std::underlying_type<DxilResourceBase::Class>::type>(DXIL::ResourceClass::CBuffer)); 
 | 
	
		
			
				|  |  | +      Constant *MetaDataArg = HlslOP->GetU32Const(ID); // position of the metadata record in the corresponding metadata list
 | 
	
		
			
				|  |  | +      Constant *IndexArg = HlslOP->GetU32Const(0); // 
 | 
	
		
			
				|  |  | +      Constant *FalseArg = HlslOP->GetI1Const(0); // non-uniform resource index: false
 | 
	
		
			
				|  |  | +      CallInst *callCreateHandle = Builder.CreateCall(createHandle, { CreateHandleOpcodeArg, CBVArg, MetaDataArg, IndexArg, FalseArg }, ConstantBufferName);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      DM.ReEmitDxilResources();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +#define PIX_CONSTANT_VALUE "PIX_Constant_Color_Value"
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // Insert the Buffer load instruction:
 | 
	
		
			
				|  |  | +      Function *CBLoad = HlslOP->GetOpFunc(OP::OpCode::CBufferLoadLegacy, hasFloatOutputs ? Type::getFloatTy(Ctx) : Type::getInt32Ty(Ctx));
 | 
	
		
			
				|  |  | +      Constant *OpArg = HlslOP->GetU32Const((unsigned)OP::OpCode::CBufferLoadLegacy);
 | 
	
		
			
				|  |  | +      Value * ResourceHandle = callCreateHandle;
 | 
	
		
			
				|  |  | +      Constant *RowIndex = HlslOP->GetU32Const(0);
 | 
	
		
			
				|  |  | +      CallInst *loadLegacy = Builder.CreateCall(CBLoad, { OpArg, ResourceHandle, RowIndex }, PIX_CONSTANT_VALUE);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // Now extract four color values:
 | 
	
		
			
				|  |  | +      ReplacementColors[0] = Builder.CreateExtractValue(loadLegacy, 0, PIX_CONSTANT_VALUE "0");
 | 
	
		
			
				|  |  | +      ReplacementColors[1] = Builder.CreateExtractValue(loadLegacy, 1, PIX_CONSTANT_VALUE "1");
 | 
	
		
			
				|  |  | +      ReplacementColors[2] = Builder.CreateExtractValue(loadLegacy, 2, PIX_CONSTANT_VALUE "2");
 | 
	
		
			
				|  |  | +      ReplacementColors[3] = Builder.CreateExtractValue(loadLegacy, 3, PIX_CONSTANT_VALUE "3");
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    break;
 | 
	
		
			
				|  |  | +    default:
 | 
	
		
			
				|  |  | +      assert(false);
 | 
	
		
			
				|  |  | +      return 0;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |    bool Modified = false;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    // The StoreOutput function can store either a float or an integer, depending on the intended output
 | 
	
		
			
				|  |  |    // render-target resource view.
 | 
	
		
			
				|  |  | -  Function * FloatOutputFunction = HlslOP->GetOpFunc(DXIL::OpCode::StoreOutput, Type::getFloatTy(Ctx));
 | 
	
		
			
				|  |  | -  if (FloatOutputFunction->getNumUses() != 0) {
 | 
	
		
			
				|  |  | -    Modified = convertTarget0ToConstantValue(FloatOutputFunction, OutputSignature, HlslOP, color);
 | 
	
		
			
				|  |  | +  if (hasFloatOutputs) {
 | 
	
		
			
				|  |  | +    visitOutputInstructionCallers(FloatOutputFunction, OutputSignature, HlslOP, 
 | 
	
		
			
				|  |  | +      [&ReplacementColors, &Modified](CallInst * CallInstruction) {
 | 
	
		
			
				|  |  | +      Modified = true;
 | 
	
		
			
				|  |  | +      // The output column is the channel (red, green, blue or alpha) within the output pixel
 | 
	
		
			
				|  |  | +      Value * OutputColumnOperand = CallInstruction->getOperand(hlsl::DXIL::OperandIndex::kStoreOutputColOpIdx);
 | 
	
		
			
				|  |  | +      ConstantInt * OutputColumnConstant = cast<ConstantInt>(OutputColumnOperand);
 | 
	
		
			
				|  |  | +      APInt OutputColumn = OutputColumnConstant->getValue();
 | 
	
		
			
				|  |  | +      CallInstruction->setOperand(hlsl::DXIL::OperandIndex::kStoreOutputValOpIdx, ReplacementColors[*OutputColumn.getRawData()]);
 | 
	
		
			
				|  |  | +    });
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -  Function * IntOutputFunction = HlslOP->GetOpFunc(DXIL::OpCode::StoreOutput, Type::getInt32Ty(Ctx));
 | 
	
		
			
				|  |  | -  if (IntOutputFunction->getNumUses() != 0) {
 | 
	
		
			
				|  |  | -    Modified = convertTarget0ToConstantValue(IntOutputFunction, OutputSignature, HlslOP, color);
 | 
	
		
			
				|  |  | +  if (hasIntOutputs) {
 | 
	
		
			
				|  |  | +    visitOutputInstructionCallers(IntOutputFunction, OutputSignature, HlslOP, 
 | 
	
		
			
				|  |  | +    [&ReplacementColors, &Modified](CallInst * CallInstruction) {
 | 
	
		
			
				|  |  | +      Modified = true;
 | 
	
		
			
				|  |  | +      // The output column is the channel (red, green, blue or alpha) within the output pixel
 | 
	
		
			
				|  |  | +      Value * OutputColumnOperand = CallInstruction->getOperand(hlsl::DXIL::OperandIndex::kStoreOutputColOpIdx);
 | 
	
		
			
				|  |  | +      ConstantInt * OutputColumnConstant = cast<ConstantInt>(OutputColumnOperand);
 | 
	
		
			
				|  |  | +      APInt OutputColumn = OutputColumnConstant->getValue();
 | 
	
		
			
				|  |  | +      CallInstruction->setOperand(hlsl::DXIL::OperandIndex::kStoreOutputValOpIdx, ReplacementColors[*OutputColumn.getRawData()]);
 | 
	
		
			
				|  |  | +    });
 | 
	
		
			
				|  |  |    }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |    return Modified;
 |