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(loc, adaptor.getAntecedent(), rewriter);
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(loc, adaptor.getInput(), rewriter);
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);
176struct LowerLTLToCorePass
177 :
public circt::impl::LowerLTLToCoreBase<LowerLTLToCorePass> {
178 LowerLTLToCorePass() =
default;
179 void runOnOperation()
override;
184void LowerLTLToCorePass::runOnOperation() {
188 ConversionTarget target(getContext());
189 target.addLegalDialect<hw::HWDialect>();
190 target.addLegalDialect<comb::CombDialect>();
191 target.addLegalDialect<sv::SVDialect>();
192 target.addLegalDialect<seq::SeqDialect>();
193 target.addLegalDialect<ltl::LTLDialect>();
194 target.addLegalDialect<verif::VerifDialect>();
195 target.addIllegalOp<verif::HasBeenResetOp>();
197 auto isLegal = [](Operation *op) {
198 auto hasNonAssertUsers = std::any_of(
199 op->getUsers().begin(), op->getUsers().end(), [](Operation *user) {
200 return !isa<verif::AssertOp, verif::ClockedAssertOp>(user);
202 auto hasIntegerResultTypes =
203 std::all_of(op->getResultTypes().begin(), op->getResultTypes().end(),
204 [](Type type) { return isa<IntegerType>(type); });
207 if (hasNonAssertUsers && !hasIntegerResultTypes)
212 op->getOperands().begin(), op->getOperands().end(),
213 [](Value operand) { return !isa<IntegerType>(operand.getType()); });
215 target.addDynamicallyLegalOp<ltl::ImplicationOp>(isLegal);
216 target.addDynamicallyLegalOp<ltl::NotOp>(isLegal);
217 target.addDynamicallyLegalOp<ltl::AndOp>(isLegal);
218 target.addDynamicallyLegalOp<ltl::OrOp>(isLegal);
221 mlir::TypeConverter converter;
224 converter.addConversion([](IntegerType type) {
return type; });
225 converter.addConversion([](ltl::PropertyType type) {
226 return IntegerType::get(type.getContext(), 1);
228 converter.addConversion([](ltl::SequenceType type) {
229 return IntegerType::get(type.getContext(), 1);
233 converter.addTargetMaterialization(
234 [&](mlir::OpBuilder &builder, mlir::Type resultType,
235 mlir::ValueRange inputs, mlir::Location loc) -> mlir::Value {
236 if (inputs.size() != 1)
238 return UnrealizedConversionCastOp::create(builder, loc, resultType,
243 converter.addSourceMaterialization(
244 [&](mlir::OpBuilder &builder, mlir::Type resultType,
245 mlir::ValueRange inputs, mlir::Location loc) -> mlir::Value {
246 if (inputs.size() != 1)
248 return UnrealizedConversionCastOp::create(builder, loc, resultType,
254 RewritePatternSet
patterns(&getContext());
255 patterns.add<HasBeenResetOpConversion, LTLImplicationConversion,
256 LTLNotConversion, LTLAndOpConversion, LTLOrOpConversion>(
260 applyPartialConversion(getOperation(), target, std::move(
patterns))))
261 return signalPassFailure();
264 getOperation().walk([&](Operation *op) {
265 if (!isa<verif::AssertOp, verif::ClockedAssertOp>(op))
267 Value prop = op->getOperand(0);
268 if (
auto cast = prop.getDefiningOp<UnrealizedConversionCastOp>()) {
271 if (auto intType = dyn_cast<IntegerType>(cast.getOperandTypes()[0]);
272 intType && intType.getWidth() == 1)
273 op->setOperand(0, cast.getInputs()[0]);
280 return std::make_unique<LowerLTLToCorePass>();
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)