CIRCT 22.0.0git
Loading...
Searching...
No Matches
PassiveWires.cpp
Go to the documentation of this file.
1//===- PassiveWires.cpp - Make Wires Passive --------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file defines the PassiveWires pass. This pass eliminated flips from
10// wires with aggregate types. Since flips only determine connect direction,
11// they are unnecessary on wires and just get in the way.
12//
13//===----------------------------------------------------------------------===//
14
18#include "circt/Support/Debug.h"
19#include "mlir/IR/ImplicitLocOpBuilder.h"
20#include "mlir/Pass/Pass.h"
21#include "llvm/Support/Debug.h"
22
23#define DEBUG_TYPE "firrtl-passive-wires"
24
25namespace circt {
26namespace firrtl {
27#define GEN_PASS_DEF_PASSIVEWIRES
28#include "circt/Dialect/FIRRTL/Passes.h.inc"
29} // namespace firrtl
30} // namespace circt
31
32using namespace circt;
33using namespace firrtl;
34
35static bool hasFlip(Type t) {
36 if (auto type = type_dyn_cast<FIRRTLBaseType>(t))
37 return !type.isPassive();
38 return false;
39}
40
41//===----------------------------------------------------------------------===//
42// Pass Infrastructure
43//===----------------------------------------------------------------------===//
44
45namespace {
46struct PassiveWiresPass
47 : public circt::firrtl::impl::PassiveWiresBase<PassiveWiresPass> {
48 void runOnOperation() override;
49};
50} // end anonymous namespace
51
52// This is the main entrypoint for the lowering pass.
53void PassiveWiresPass::runOnOperation() {
55
56 auto module = getOperation();
57
58 // First, expand any connects to resolve flips.
59 SmallVector<Operation *> worklist;
60 module.walk([&](Operation *op) -> WalkResult {
61 if (auto wire = dyn_cast<WireOp>(op)) {
62 if (hasFlip(wire.getType(0)))
63 worklist.push_back(wire);
64 return WalkResult::advance();
65 }
66 if (!isa<ConnectOp, MatchingConnectOp>(op))
67 return WalkResult::advance();
68 // connect/matchingconnect
69 if (!hasFlip(op->getOperand(0).getType()))
70 return WalkResult::advance();
71
72 mlir::ImplicitLocOpBuilder builder(op->getLoc(), op);
73 // This will "blow out" a connect to passive pieces
74 emitConnect(builder, op->getOperand(0), op->getOperand(1));
75 op->erase();
76 return WalkResult::advance();
77 });
78
79 // Second, remove flips from most things.
80 while (!worklist.empty()) {
81 auto *op = worklist.back();
82 worklist.pop_back();
83 auto r = op->getResult(0);
84 if (!hasFlip(r.getType()))
85 continue;
86 for (auto users : r.getUsers())
87 worklist.push_back(users);
88 // In-place updates is safe as consumers don't care about flip.
89 r.setType(type_cast<FIRRTLBaseType>(r.getType()).getPassiveType());
90 }
91}
static bool hasFlip(Type t)
#define CIRCT_DEBUG_SCOPED_PASS_LOGGER(PASS)
Definition Debug.h:70
void emitConnect(OpBuilder &builder, Location loc, Value lhs, Value rhs)
Emit a connect between two values.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.