25#include "mlir/Dialect/Func/IR/FuncOps.h"
26#include "mlir/Dialect/LLVMIR/LLVMDialect.h"
27#include "mlir/Dialect/SCF/IR/SCF.h"
28#include "mlir/IR/BuiltinTypes.h"
29#include "mlir/Pass/Pass.h"
30#include "mlir/Transforms/DialectConversion.h"
31#include "llvm/Support/MathExtras.h"
34#define GEN_PASS_DEF_LOWERLTLTOCORE
35#include "circt/Conversion/Passes.h.inc"
53 matchAndRewrite(verif::HasBeenResetOp op, OpAdaptor adaptor,
54 ConversionPatternRewriter &rewriter)
const override {
55 auto i1 = rewriter.getI1Type();
57 Value constZero = seq::createConstantInitialValue(
58 rewriter, op->getLoc(), rewriter.getIntegerAttr(i1, 0));
70 comb::OrOp::create(rewriter, op.getLoc(), adaptor.getReset(),
reg);
74 Value reset, resetval;
78 rewriter, op.getLoc(), orReset,
79 rewriter.createOrFold<seq::ToClockOp>(op.getLoc(), adaptor.getClock()),
80 rewriter.getStringAttr(
"hbr"), reset, resetval, constZero,
87 Value notReset = comb::XorOp::create(rewriter, op.getLoc(),
88 adaptor.getReset(), constOne);
95struct LTLImplicationConversion
100 matchAndRewrite(ltl::ImplicationOp op, OpAdaptor adaptor,
101 ConversionPatternRewriter &rewriter)
const override {
103 if (!isa<IntegerType>(op.getAntecedent().getType()) ||
104 !isa<IntegerType>(op.getConsequent().getType()))
107 auto loc = op.getLoc();
108 auto notA = comb::createOrFoldNot(rewriter, loc, adaptor.getAntecedent());
110 comb::OrOp::create(rewriter, loc, notA, adaptor.getConsequent());
111 rewriter.replaceOp(op, orOp);
120 matchAndRewrite(ltl::NotOp op, OpAdaptor adaptor,
121 ConversionPatternRewriter &rewriter)
const override {
123 if (!isa<IntegerType>(op.getInput().getType()))
125 auto loc = op.getLoc();
126 auto inverted = comb::createOrFoldNot(rewriter, loc, adaptor.getInput());
127 rewriter.replaceOp(op, inverted);
136 matchAndRewrite(ltl::AndOp op, OpAdaptor adaptor,
137 ConversionPatternRewriter &rewriter)
const override {
139 if (!isa<IntegerType>(op->getOperandTypes()[0]) ||
140 !isa<IntegerType>(op->getOperandTypes()[1]))
142 auto loc = op.getLoc();
145 comb::AndOp::create(rewriter, loc, adaptor.getOperands(),
false);
146 rewriter.replaceOp(op, andOp);
155 matchAndRewrite(ltl::OrOp op, OpAdaptor adaptor,
156 ConversionPatternRewriter &rewriter)
const override {
158 if (!isa<IntegerType>(op->getOperandTypes()[0]) ||
159 !isa<IntegerType>(op->getOperandTypes()[1]))
161 auto loc = op.getLoc();
163 auto orOp = comb::OrOp::create(rewriter, loc, adaptor.getOperands(),
false);
164 rewriter.replaceOp(op, orOp);
170 LTLPastOpConversion(MLIRContext *
context,
bool assumeFirstClock)
172 assumeFirstClock(assumeFirstClock) {}
175 matchAndRewrite(ltl::PastOp op, OpAdaptor adaptor,
176 ConversionPatternRewriter &rewriter)
const override {
178 if (!adaptor.getClk()) {
179 if (!assumeFirstClock)
182 auto module = op->getParentOfType<HWModuleOp>();
183 std::optional<size_t> clockArgNum;
185 if (port.isInput() && isa<seq::ClockType>(port.type)) {
186 clockArgNum = port.argNum;
191 clock =
module.getArgumentForInput(*clockArgNum);
194 auto toClockOps =
module.getOps<seq::ToClockOp>();
195 if (toClockOps.empty())
197 clock = (*toClockOps.begin()).getResult();
200 clock = seq::ToClockOp::create(rewriter, op.getLoc(), adaptor.getClk());
203 Value cur = adaptor.getInput();
207 seq::ShiftRegOp::create(rewriter, op.getLoc(), op.getDelayAttr(), cur,
208 clock, ce, {}, {}, {}, {}, {});
209 rewriter.replaceOp(op, shiftreg);
213 bool assumeFirstClock;
223struct LowerLTLToCorePass
224 :
public circt::impl::LowerLTLToCoreBase<LowerLTLToCorePass> {
225 LowerLTLToCorePass() =
default;
226 void runOnOperation()
override;
231void LowerLTLToCorePass::runOnOperation() {
234 if (!assumeFirstClock) {
235 auto res = getOperation().walk([&](ltl::PastOp op) {
238 "ltl.past operations without a clock operand are not supported.");
239 return WalkResult::interrupt();
241 return WalkResult::advance();
243 if (res.wasInterrupted())
244 return signalPassFailure();
249 ConversionTarget target(getContext());
250 target.addLegalDialect<hw::HWDialect>();
251 target.addLegalDialect<comb::CombDialect>();
252 target.addLegalDialect<sv::SVDialect>();
253 target.addLegalDialect<seq::SeqDialect>();
254 target.addLegalDialect<ltl::LTLDialect>();
255 target.addLegalDialect<verif::VerifDialect>();
256 target.addIllegalOp<verif::HasBeenResetOp>();
257 target.addIllegalOp<ltl::PastOp>();
259 auto isLegal = [](Operation *op) {
260 auto hasNonAssertUsers = std::any_of(
261 op->getUsers().begin(), op->getUsers().end(), [](Operation *user) {
262 return !isa<verif::AssertOp, verif::ClockedAssertOp>(user);
264 auto hasIntegerResultTypes =
265 std::all_of(op->getResultTypes().begin(), op->getResultTypes().end(),
266 [](Type type) { return isa<IntegerType>(type); });
269 if (hasNonAssertUsers && !hasIntegerResultTypes)
274 op->getOperands().begin(), op->getOperands().end(),
275 [](Value operand) { return !isa<IntegerType>(operand.getType()); });
277 target.addDynamicallyLegalOp<ltl::ImplicationOp>(isLegal);
278 target.addDynamicallyLegalOp<ltl::NotOp>(isLegal);
279 target.addDynamicallyLegalOp<ltl::AndOp>(isLegal);
280 target.addDynamicallyLegalOp<ltl::OrOp>(isLegal);
283 mlir::TypeConverter converter;
286 converter.addConversion([](IntegerType type) {
return type; });
287 converter.addConversion([](ltl::PropertyType type) {
288 return IntegerType::get(type.getContext(), 1);
290 converter.addConversion([](ltl::SequenceType type) {
291 return IntegerType::get(type.getContext(), 1);
295 converter.addTargetMaterialization(
296 [&](mlir::OpBuilder &builder, mlir::Type resultType,
297 mlir::ValueRange inputs, mlir::Location loc) -> mlir::Value {
298 if (inputs.size() != 1)
300 return UnrealizedConversionCastOp::create(builder, loc, resultType,
305 converter.addSourceMaterialization(
306 [&](mlir::OpBuilder &builder, mlir::Type resultType,
307 mlir::ValueRange inputs, mlir::Location loc) -> mlir::Value {
308 if (inputs.size() != 1)
310 return UnrealizedConversionCastOp::create(builder, loc, resultType,
316 RewritePatternSet
patterns(&getContext());
317 patterns.add<HasBeenResetOpConversion, LTLImplicationConversion,
318 LTLNotConversion, LTLAndOpConversion, LTLOrOpConversion>(
323 applyPartialConversion(getOperation(), target, std::move(
patterns))))
324 return signalPassFailure();
327 getOperation().walk([&](Operation *op) {
328 if (!isa<verif::AssertOp, verif::ClockedAssertOp>(op))
330 Value prop = op->getOperand(0);
331 if (
auto cast = prop.getDefiningOp<UnrealizedConversionCastOp>()) {
334 if (auto intType = dyn_cast<IntegerType>(cast.getOperandTypes()[0]);
335 intType && intType.getWidth() == 1)
336 op->setOperand(0, cast.getInputs()[0]);
343 return std::make_unique<LowerLTLToCorePass>();
static std::unique_ptr< Context > context
static SmallVector< PortInfo > getPortList(ModuleTy &mod)
Instantiate one of these and use it to build typed backedges.
Backedge get(mlir::Type resultType, mlir::LocationAttr optionalLoc={})
Create a typed backedge.
Backedge is a wrapper class around a Value.
create(cls, result_type, reset=None, reset_value=None, name=None, sym_name=None, **kwargs)
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
std::unique_ptr< mlir::Pass > createLowerLTLToCorePass()
reg(value, clock, reset=None, reset_value=None, name=None, sym_name=None)