13 #include "mlir/Dialect/ControlFlow/IR/ControlFlowOps.h"
14 #include "mlir/IR/Dominance.h"
15 #include "mlir/IR/PatternMatch.h"
16 #include "mlir/IR/Region.h"
17 #include "mlir/Pass/Pass.h"
18 #include "mlir/Support/LLVM.h"
19 #include "mlir/Transforms/RegionUtils.h"
20 #include "llvm/ADT/STLExtras.h"
25 #define GEN_PASS_DEF_TEMPORALCODEMOTION
26 #include "circt/Dialect/LLHD/Transforms/Passes.h.inc"
30 using namespace circt;
39 DenseMap<Block *, Value> &mem) {
40 Location loc = driveBlock->getTerminator()->getLoc();
41 if (mem.count(driveBlock))
42 return mem[driveBlock];
44 SmallVector<Block *> worklist;
45 worklist.push_back(driveBlock);
47 while (!worklist.empty()) {
48 Block *curr = worklist.back();
50 if (curr == dominator || curr->getPredecessors().empty()) {
58 bool addedSomething =
false;
59 for (
auto *predBlock : curr->getPredecessors()) {
60 if (!mem.count(predBlock)) {
61 worklist.push_back(predBlock);
62 addedSomething =
true;
70 for (
auto *predBlock : curr->getPredecessors()) {
71 if (predBlock->getTerminator()->getNumSuccessors() != 1) {
72 auto condBr = cast<cf::CondBranchOp>(predBlock->getTerminator());
73 Value cond = condBr.getCondition();
74 if (condBr.getFalseDest() == curr) {
78 Value next = builder.create<
comb::AndOp>(loc, mem[predBlock], cond);
79 runner = builder.create<
comb::OrOp>(loc, runner, next);
81 runner = builder.create<
comb::OrOp>(loc, runner, mem[predBlock]);
88 return mem[driveBlock];
94 Operation *moveBefore,
95 DenseMap<Block *, Value> &mem) {
96 OpBuilder builder(drvOp);
97 builder.setInsertionPoint(moveBefore);
98 Block *drvParentBlock = drvOp->getBlock();
103 builder, drvParentBlock, dominator, mem);
105 if (drvOp.getEnable())
106 finalValue = builder.create<
comb::AndOp>(drvOp.getLoc(), drvOp.getEnable(),
109 drvOp.getEnableMutable().assign(finalValue);
110 drvOp->moveBefore(moveBefore);
114 struct TemporalCodeMotionPass
115 :
public llhd::impl::TemporalCodeMotionBase<TemporalCodeMotionPass> {
116 void runOnOperation()
override;
117 LogicalResult runOnProcess(llhd::ProcessOp procOp);
121 void TemporalCodeMotionPass::runOnOperation() {
122 for (
auto proc : getOperation().getOps<llhd::ProcessOp>())
123 (void)runOnProcess(proc);
126 LogicalResult TemporalCodeMotionPass::runOnProcess(llhd::ProcessOp procOp) {
153 bool seenWait =
false;
154 WalkResult walkResult = procOp.walk([&](llhd::WaitOp op) -> WalkResult {
160 int trId = trAnalysis.
getBlockTR(op->getBlock());
165 return WalkResult::advance();
167 if (walkResult.wasInterrupted())
175 for (
unsigned currTR = 0; currTR < numTRs; ++currTR) {
180 assert((numTRSuccs == 1 ||
182 "only TRs with a single TR as possible successor are "
183 "supported for now.");
196 auxBlock->addArguments(
197 succTREntry->getArgumentTypes(),
198 SmallVector<Location>(succTREntry->getNumArguments(), procOp.getLoc()));
201 procOp.getBody().getBlocks().insertAfter(
207 for (
auto [i, succ] : llvm::enumerate(exit->getSuccessors()))
208 if (trAnalysis.
getBlockTR(succ) !=
static_cast<int>(currTR))
209 exit->getTerminator()->setSuccessor(auxBlock, i);
214 b.setInsertionPointToEnd(auxBlock);
215 b.create<cf::BranchOp>(procOp.getLoc(), succTREntry,
216 auxBlock->getArguments());
223 DenseMap<Operation *, Block *> drvPos;
228 OpBuilder builder(procOp);
230 for (
unsigned currTR = 0; currTR < numTRs; ++currTR) {
231 DenseMap<Block *, Value> mem;
239 DominanceInfo dom(procOp);
240 Block *dominator = exitingBlock;
244 procOp.walk([&](llhd::DrvOp op) {
245 if (trAnalysis.
getBlockTR(op.getOperation()->getBlock()) ==
246 static_cast<int>(currTR)) {
247 Block *parentBlock = op.getOperation()->getBlock();
248 drvPos[op] = parentBlock;
249 dominator = dom.findNearestCommonDominator(dominator, parentBlock);
254 Operation *moveBefore = exitingBlock->getTerminator();
255 exitingBlock->walk([&](llhd::DrvOp op) { moveBefore = op; });
258 "could not find nearest common dominator for TR exiting "
259 "block and the block containing drv");
263 if (trAnalysis.
getBlockTR(dominator) !=
static_cast<int>(currTR))
266 std::queue<Block *> workQueue;
267 SmallPtrSet<Block *, 32> workDone;
269 if (entryBlock != exitingBlock)
270 workQueue.push(entryBlock);
272 while (!workQueue.empty()) {
273 Block *block = workQueue.front();
275 workDone.insert(block);
277 builder.setInsertionPoint(moveBefore);
278 SmallVector<llhd::DrvOp> drives(block->getOps<llhd::DrvOp>());
279 for (
auto drive : drives)
282 for (Block *succ : block->getSuccessors()) {
283 if (succ == exitingBlock ||
284 trAnalysis.
getBlockTR(succ) !=
static_cast<int>(currTR))
287 if (llvm::all_of(succ->getPredecessors(), [&](Block *block) {
288 return workDone.contains(block);
290 workQueue.push(succ);
295 if (entryBlock != exitingBlock) {
296 entryBlock->getTerminator()->erase();
297 entryBlock->getOperations().splice(entryBlock->end(),
298 exitingBlock->getOperations());
302 IRRewriter rewriter(procOp);
303 (void)mlir::eraseUnreachableBlocks(rewriter, procOp->getRegions());
310 DominanceInfo dom(procOp);
311 for (
unsigned currTR = 0; currTR < numTRs; ++currTR) {
316 DenseMap<std::pair<Value, Value>, llhd::DrvOp> sigToDrv;
318 SmallVector<llhd::DrvOp> drives(exitingBlock->getOps<llhd::DrvOp>());
319 for (
auto op : drives) {
320 auto sigTimePair = std::make_pair(op.getSignal(), op.getTime());
321 if (!sigToDrv.count(sigTimePair)) {
322 sigToDrv[sigTimePair] = op;
326 OpBuilder builder(op);
327 if (op.getEnable()) {
329 auto firstDrive = sigToDrv[sigTimePair];
331 op.getLoc(), op.getEnable(), op.getValue(), firstDrive.getValue());
332 op.getValueMutable().assign(muxValue);
335 if (firstDrive.getEnable()) {
336 Value orVal = builder.create<
comb::OrOp>(op.getLoc(), op.getEnable(),
337 firstDrive.getEnable());
338 op.getEnableMutable().assign(orVal);
341 op.getEnableMutable().clear();
345 sigToDrv[sigTimePair]->erase();
346 sigToDrv[sigTimePair] = op;
353 std::unique_ptr<OperationPass<hw::HWModuleOp>>
355 return std::make_unique<TemporalCodeMotionPass>();
assert(baseType &&"element must be base type")
static void moveDriveOpBefore(llhd::DrvOp drvOp, Block *dominator, Operation *moveBefore, DenseMap< Block *, Value > &mem)
More a 'llhd.drv' operation before the 'moveBefore' operation by adjusting the 'enable' operand.
static Value getBranchDecisionsFromDominatorToTarget(OpBuilder &builder, Block *driveBlock, Block *dominator, DenseMap< Block *, Value > &mem)
Explore all paths from the 'driveBlock' to the 'dominator' block and construct a boolean expression a...
def create(data_type, value)
std::unique_ptr< OperationPass< hw::HWModuleOp > > createTemporalCodeMotionPass()
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
int getBlockTR(Block *) const
unsigned getNumTemporalRegions() const
bool isOwnTRSuccessor(int tr)
bool hasSingleExitBlock(int tr) const
SmallVector< Block *, 8 > getExitingBlocksInTR(int) const
SmallVector< int, 8 > getTRSuccessors(int)
Block * getTREntryBlock(int)
unsigned getNumTRSuccessors(int tr)