260 строки
8.9 KiB
C++
260 строки
8.9 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// //
|
|
// WaveSensitivityAnalysis.cpp //
|
|
// Copyright (C) Microsoft Corporation. All rights reserved. //
|
|
// This file is distributed under the University of Illinois Open Source //
|
|
// License. See LICENSE.TXT for details. //
|
|
// //
|
|
// This file provides support for doing analysis that are aware of wave //
|
|
// intrinsics. //
|
|
// //
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "dxc/DXIL/DxilInstructions.h"
|
|
#include "dxc/DXIL/DxilOperations.h"
|
|
#include "dxc/HLSL/DxilGenerationPass.h"
|
|
#include "dxc/Support/Global.h"
|
|
|
|
#include "llvm/Analysis/PostDominators.h"
|
|
#include "llvm/IR/Constants.h"
|
|
#include "llvm/IR/Instructions.h"
|
|
|
|
using namespace llvm;
|
|
using std::map;
|
|
|
|
namespace hlsl {
|
|
|
|
// WaveSensitivityAnalysis is created to validate Gradient operations.
|
|
// Gradient operations require all neighbor lanes to be active when calculated,
|
|
// compiler will enable lanes to meet this requirement. If a wave operation
|
|
// contributed to gradient operation, it will get unexpected result because the
|
|
// active lanes are modified.
|
|
// To avoid unexpected result, validation will fail if gradient operations
|
|
// are dependent on wave-sensitive data or control flow.
|
|
|
|
class WaveSensitivityAnalyzer : public WaveSensitivityAnalysis {
|
|
private:
|
|
enum WaveSensitivity { KnownSensitive, KnownNotSensitive, Unknown };
|
|
PostDominatorTree *pPDT;
|
|
map<Instruction *, WaveSensitivity> InstState;
|
|
map<BasicBlock *, WaveSensitivity> BBState;
|
|
std::vector<Instruction *> InstWorkList;
|
|
std::vector<PHINode *>
|
|
UnknownPhis; // currently unknown phis. Indicate cycles after Analyze
|
|
std::vector<BasicBlock *> BBWorkList;
|
|
bool CheckBBState(BasicBlock *BB, WaveSensitivity WS);
|
|
WaveSensitivity GetInstState(Instruction *I);
|
|
void UpdateBlock(BasicBlock *BB, WaveSensitivity WS);
|
|
void UpdateInst(Instruction *I, WaveSensitivity WS);
|
|
void VisitInst(Instruction *I);
|
|
|
|
public:
|
|
WaveSensitivityAnalyzer(PostDominatorTree &PDT) : pPDT(&PDT) {}
|
|
void Analyze(Function *F);
|
|
void Analyze();
|
|
bool IsWaveSensitive(Instruction *op);
|
|
};
|
|
|
|
WaveSensitivityAnalysis *
|
|
WaveSensitivityAnalysis::create(PostDominatorTree &PDT) {
|
|
return new WaveSensitivityAnalyzer(PDT);
|
|
}
|
|
|
|
// Analyze the given function's instructions as wave-sensitive or not
|
|
void WaveSensitivityAnalyzer::Analyze(Function *F) {
|
|
// Add all blocks but the entry in reverse order so they come out in order
|
|
auto it = F->getBasicBlockList().end();
|
|
for (it--; it != F->getBasicBlockList().begin(); it--)
|
|
BBWorkList.emplace_back(&*it);
|
|
// Add entry block as non-sensitive
|
|
UpdateBlock(&*it, KnownNotSensitive);
|
|
|
|
// First analysis
|
|
Analyze();
|
|
|
|
// If any phis with explored preds remain unknown
|
|
// it has to be in a loop that don't include wave sensitivity
|
|
// Update each as such and redo Analyze to mark the descendents
|
|
while (!UnknownPhis.empty() || !InstWorkList.empty() || !BBWorkList.empty()) {
|
|
while (!UnknownPhis.empty()) {
|
|
PHINode *Phi = UnknownPhis.back();
|
|
UnknownPhis.pop_back();
|
|
// UnknownPhis might have actually known phis that were changed. skip them
|
|
if (Unknown == GetInstState(Phi)) {
|
|
// If any of the preds have not been visited, we can't assume a cycle
|
|
// yet
|
|
bool allPredsVisited = true;
|
|
for (unsigned i = 0; i < Phi->getNumIncomingValues(); i++) {
|
|
if (!BBState.count(Phi->getIncomingBlock(i))) {
|
|
allPredsVisited = false;
|
|
break;
|
|
}
|
|
}
|
|
#ifndef NDEBUG
|
|
for (unsigned i = 0; i < Phi->getNumIncomingValues(); i++) {
|
|
if (Instruction *IArg =
|
|
dyn_cast<Instruction>(Phi->getIncomingValue(i))) {
|
|
DXASSERT_LOCALVAR(IArg, GetInstState(IArg) != KnownSensitive,
|
|
"Unknown wave-status Phi argument should not be "
|
|
"able to be known sensitive");
|
|
}
|
|
}
|
|
#endif
|
|
if (allPredsVisited)
|
|
UpdateInst(Phi, KnownNotSensitive);
|
|
}
|
|
}
|
|
Analyze();
|
|
}
|
|
#ifndef NDEBUG
|
|
for (BasicBlock &BB : *F) {
|
|
for (Instruction &I : BB) {
|
|
DXASSERT_LOCALVAR(I, Unknown != GetInstState(&I),
|
|
"Wave sensitivity analysis exited without finding "
|
|
"results for all instructions");
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// Analyze the member instruction and BBlock worklists
|
|
void WaveSensitivityAnalyzer::Analyze() {
|
|
while (!InstWorkList.empty() || !BBWorkList.empty()) {
|
|
// Process the instruction work list.
|
|
while (!InstWorkList.empty()) {
|
|
Instruction *I = InstWorkList.back();
|
|
InstWorkList.pop_back();
|
|
|
|
// "I" got into the work list because it made a transition.
|
|
for (User *U : I->users()) {
|
|
Instruction *UI = cast<Instruction>(U);
|
|
VisitInst(UI);
|
|
}
|
|
}
|
|
|
|
// Process one entry of the basic block work list.
|
|
if (!BBWorkList.empty()) {
|
|
BasicBlock *BB = BBWorkList.back();
|
|
BBWorkList.pop_back();
|
|
|
|
// Notify all instructions in this basic block that they need to
|
|
// be reevaluated (eg, a block previously though to be insensitive
|
|
// is now sensitive).
|
|
for (BasicBlock::iterator I = BB->begin(), E = BB->end(); I != E; ++I)
|
|
VisitInst(I);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool WaveSensitivityAnalyzer::CheckBBState(BasicBlock *BB, WaveSensitivity WS) {
|
|
auto c = BBState.find(BB);
|
|
if (c == BBState.end()) {
|
|
return WS == Unknown;
|
|
} else {
|
|
return (*c).second == WS;
|
|
}
|
|
}
|
|
|
|
WaveSensitivityAnalyzer::WaveSensitivity
|
|
WaveSensitivityAnalyzer::GetInstState(Instruction *I) {
|
|
auto c = InstState.find(I);
|
|
if (c == InstState.end())
|
|
return Unknown;
|
|
return (*c).second;
|
|
}
|
|
|
|
void WaveSensitivityAnalyzer::UpdateBlock(BasicBlock *BB, WaveSensitivity WS) {
|
|
auto c = BBState.find(BB);
|
|
// Do not update if an entry is already found and it hasn't changed or
|
|
// has already been marked as wave sensitive (an insensitive term might
|
|
// try to mark it as such, but this effectively implements the 'any pred'
|
|
// rule).
|
|
if (c != BBState.end() &&
|
|
((*c).second == WS || (*c).second == KnownSensitive))
|
|
return;
|
|
BBState[BB] = WS;
|
|
BBWorkList.push_back(BB);
|
|
}
|
|
|
|
void WaveSensitivityAnalyzer::UpdateInst(Instruction *I, WaveSensitivity WS) {
|
|
auto c = InstState.find(I);
|
|
if (c == InstState.end() || (*c).second != WS) {
|
|
InstState[I] = WS;
|
|
InstWorkList.push_back(I);
|
|
if (TerminatorInst *TI = dyn_cast<TerminatorInst>(I)) {
|
|
BasicBlock *CurBB = TI->getParent();
|
|
for (unsigned i = 0; i < TI->getNumSuccessors(); ++i) {
|
|
BasicBlock *BB = TI->getSuccessor(i);
|
|
// Only propagate WS when BB not post dom CurBB.
|
|
WaveSensitivity TmpWS = pPDT->properlyDominates(BB, CurBB)
|
|
? WaveSensitivity::KnownNotSensitive
|
|
: WS;
|
|
UpdateBlock(BB, TmpWS);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void WaveSensitivityAnalyzer::VisitInst(Instruction *I) {
|
|
unsigned firstArg = 0;
|
|
if (CallInst *CI = dyn_cast<CallInst>(I)) {
|
|
if (OP::IsDxilOpFuncCallInst(CI)) {
|
|
firstArg = 1;
|
|
OP::OpCode opcode = OP::GetDxilOpFuncCallInst(CI);
|
|
if (OP::IsDxilOpWave(opcode)) {
|
|
UpdateInst(I, KnownSensitive);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (CheckBBState(I->getParent(), KnownSensitive)) {
|
|
UpdateInst(I, KnownSensitive);
|
|
return;
|
|
}
|
|
|
|
// Catch control flow wave sensitive for phi.
|
|
if (PHINode *Phi = dyn_cast<PHINode>(I)) {
|
|
for (unsigned i = 0; i < Phi->getNumIncomingValues(); i++) {
|
|
BasicBlock *BB = Phi->getIncomingBlock(i);
|
|
WaveSensitivity WS = GetInstState(BB->getTerminator());
|
|
if (WS == KnownSensitive) {
|
|
UpdateInst(I, KnownSensitive);
|
|
return;
|
|
} else if (Unknown == GetInstState(I)) {
|
|
UnknownPhis.emplace_back(Phi);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool allKnownNotSensitive = true;
|
|
for (unsigned i = firstArg; i < I->getNumOperands(); ++i) {
|
|
Value *V = I->getOperand(i);
|
|
if (Instruction *IArg = dyn_cast<Instruction>(V)) {
|
|
WaveSensitivity WS = GetInstState(IArg);
|
|
if (WS == KnownSensitive) {
|
|
UpdateInst(I, KnownSensitive);
|
|
return;
|
|
} else if (WS == Unknown) {
|
|
allKnownNotSensitive = false;
|
|
}
|
|
}
|
|
}
|
|
if (allKnownNotSensitive) {
|
|
UpdateInst(I, KnownNotSensitive);
|
|
}
|
|
}
|
|
|
|
bool WaveSensitivityAnalyzer::IsWaveSensitive(Instruction *op) {
|
|
auto c = InstState.find(op);
|
|
if (c == InstState.end()) {
|
|
DXASSERT(false,
|
|
"Instruction sensitivity not foud. Analysis didn't complete!");
|
|
return false;
|
|
}
|
|
DXASSERT((*c).second != Unknown, "else analysis is missing a case");
|
|
return (*c).second == KnownSensitive;
|
|
}
|
|
|
|
} // namespace hlsl
|