CIRCT 20.0.0git
Loading...
Searching...
No Matches
SeqToSV.cpp
Go to the documentation of this file.
1//===- LowerSeqToSV.cpp - Seq to SV lowering ------------------------------===//
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 transform translate Seq ops to SV.
10//
11//===----------------------------------------------------------------------===//
12
14#include "FirMemLowering.h"
15#include "FirRegLowering.h"
25#include "mlir/IR/Builders.h"
26#include "mlir/IR/ImplicitLocOpBuilder.h"
27#include "mlir/IR/Threading.h"
28#include "mlir/Pass/Pass.h"
29#include "mlir/Transforms/DialectConversion.h"
30
31#define DEBUG_TYPE "lower-seq-to-sv"
32
33using namespace circt;
34using namespace seq;
35using hw::HWModuleOp;
36using llvm::MapVector;
37
38namespace circt {
39#define GEN_PASS_DEF_LOWERSEQTOSV
40#include "circt/Conversion/Passes.h.inc"
41
42struct SeqToSVPass : public impl::LowerSeqToSVBase<SeqToSVPass> {
43
44 void runOnOperation() override;
45
46 using LowerSeqToSVBase<SeqToSVPass>::lowerToAlwaysFF;
47 using LowerSeqToSVBase<SeqToSVPass>::disableRegRandomization;
48 using LowerSeqToSVBase<SeqToSVPass>::emitSeparateAlwaysBlocks;
49 using LowerSeqToSVBase<SeqToSVPass>::LowerSeqToSVBase;
50 using LowerSeqToSVBase<SeqToSVPass>::numSubaccessRestored;
51};
52} // namespace circt
53
54namespace {
55struct ModuleLoweringState {
56 ModuleLoweringState(HWModuleOp module)
57 : immutableValueLowering(module), module(module) {}
58
59 struct ImmutableValueLowering {
60 ImmutableValueLowering(hw::HWModuleOp module) : module(module) {}
61
62 // Lower initial ops.
63 LogicalResult lower();
64 LogicalResult lower(seq::InitialOp initialOp);
65
66 Value
67 lookupImmutableValue(mlir::TypedValue<seq::ImmutableType> immut) const {
68 return mapping.lookup(immut);
69 }
70
71 sv::InitialOp getSVInitial() const { return svInitialOp; }
72
73 private:
74 sv::InitialOp svInitialOp = {};
75 // A mapping from a dummy immutable value to the actual initial value
76 // defined in SV initial op.
77 MapVector<mlir::TypedValue<seq::ImmutableType>, Value> mapping;
78
79 hw::HWModuleOp module;
80 } immutableValueLowering;
81
82 struct FragmentInfo {
83 bool needsRegFragment = false;
84 } fragment;
85
86 HWModuleOp module;
87};
88
89LogicalResult ModuleLoweringState::ImmutableValueLowering::lower() {
90 auto result = mergeInitialOps(module.getBodyBlock());
91 if (failed(result))
92 return failure();
93
94 auto initialOp = *result;
95 if (!initialOp)
96 return success();
97
98 return lower(initialOp);
99}
100
101LogicalResult
102ModuleLoweringState::ImmutableValueLowering::lower(seq::InitialOp initialOp) {
103 OpBuilder builder = OpBuilder::atBlockBegin(module.getBodyBlock());
104 if (!svInitialOp)
105 svInitialOp = builder.create<sv::InitialOp>(initialOp->getLoc());
106 // Initial ops are merged to single one and must not have operands.
107 assert(initialOp.getNumOperands() == 0 &&
108 "initial op should have no operands");
109
110 auto loc = initialOp.getLoc();
111 llvm::SmallVector<Value> results;
112
113 auto yieldOp = cast<seq::YieldOp>(initialOp.getBodyBlock()->getTerminator());
114
115 for (auto [result, operand] :
116 llvm::zip(initialOp.getResults(), yieldOp->getOperands())) {
117 auto placeholder =
118 builder
119 .create<mlir::UnrealizedConversionCastOp>(
120 loc, ArrayRef<Type>{result.getType()}, ArrayRef<Value>{})
121 ->getResult(0);
122 result.replaceAllUsesWith(placeholder);
123 mapping.insert(
124 {cast<mlir::TypedValue<seq ::ImmutableType>>(placeholder), operand});
125 }
126
127 svInitialOp.getBodyBlock()->getOperations().splice(
128 svInitialOp.end(), initialOp.getBodyBlock()->getOperations());
129
130 assert(initialOp->use_empty());
131 initialOp.erase();
132 yieldOp->erase();
133 return success();
134}
135
136/// Lower CompRegOp to `sv.reg` and `sv.alwaysff`. Use a posedge clock and
137/// synchronous reset.
138template <typename OpTy>
139class CompRegLower : public OpConversionPattern<OpTy> {
140public:
141 CompRegLower(
142 TypeConverter &typeConverter, MLIRContext *context, bool lowerToAlwaysFF,
143 const MapVector<StringAttr, ModuleLoweringState> &moduleLoweringStates)
144 : OpConversionPattern<OpTy>(typeConverter, context),
145 lowerToAlwaysFF(lowerToAlwaysFF),
146 moduleLoweringStates(moduleLoweringStates) {}
147
148 using OpAdaptor = typename OpConversionPattern<OpTy>::OpAdaptor;
149
150 LogicalResult
151 matchAndRewrite(OpTy reg, OpAdaptor adaptor,
152 ConversionPatternRewriter &rewriter) const final {
153 Location loc = reg.getLoc();
154
155 auto regTy =
156 ConversionPattern::getTypeConverter()->convertType(reg.getType());
157 auto svReg = rewriter.create<sv::RegOp>(loc, regTy, reg.getNameAttr(),
158 reg.getInnerSymAttr());
159
160 svReg->setDialectAttrs(reg->getDialectAttrs());
161
163
164 auto regVal = rewriter.create<sv::ReadInOutOp>(loc, svReg);
165
166 auto assignValue = [&] {
167 createAssign(rewriter, reg.getLoc(), svReg, reg);
168 };
169 auto assignReset = [&] {
170 rewriter.create<sv::PAssignOp>(loc, svReg, adaptor.getResetValue());
171 };
172
173 // Registers written in an `always_ff` process may not have any assignments
174 // outside of that process.
175 // For some tools this also prohibits inititalization.
176 bool mayLowerToAlwaysFF = lowerToAlwaysFF && !reg.getInitialValue();
177
178 if (adaptor.getReset() && adaptor.getResetValue()) {
179 if (mayLowerToAlwaysFF) {
180 rewriter.create<sv::AlwaysFFOp>(
181 loc, sv::EventControl::AtPosEdge, adaptor.getClk(),
182 sv::ResetType::SyncReset, sv::EventControl::AtPosEdge,
183 adaptor.getReset(), assignValue, assignReset);
184 } else {
185 rewriter.create<sv::AlwaysOp>(
186 loc, sv::EventControl::AtPosEdge, adaptor.getClk(), [&] {
187 rewriter.create<sv::IfOp>(loc, adaptor.getReset(), assignReset,
188 assignValue);
189 });
190 }
191 } else {
192 if (mayLowerToAlwaysFF) {
193 rewriter.create<sv::AlwaysFFOp>(loc, sv::EventControl::AtPosEdge,
194 adaptor.getClk(), assignValue);
195 } else {
196 rewriter.create<sv::AlwaysOp>(loc, sv::EventControl::AtPosEdge,
197 adaptor.getClk(), assignValue);
198 }
199 }
200
201 // Lower initial values.
202 if (auto init = reg.getInitialValue()) {
203 auto module = reg->template getParentOfType<hw::HWModuleOp>();
204 const auto &initial =
205 moduleLoweringStates.find(module.getModuleNameAttr())
206 ->second.immutableValueLowering;
207
208 Value initialValue = initial.lookupImmutableValue(init);
209
210 if (auto op = initialValue.getDefiningOp();
211 op && op->hasTrait<mlir::OpTrait::ConstantLike>()) {
212 auto clonedConstant = rewriter.clone(*op);
213 rewriter.moveOpBefore(clonedConstant, svReg);
214 svReg.getInitMutable().assign(clonedConstant->getResult(0));
215 } else {
216 OpBuilder::InsertionGuard guard(rewriter);
217 auto in = initial.getSVInitial();
218 rewriter.setInsertionPointToEnd(in.getBodyBlock());
219 rewriter.create<sv::BPAssignOp>(reg->getLoc(), svReg, initialValue);
220 }
221 }
222
223 rewriter.replaceOp(reg, regVal);
224 return success();
225 }
226
227 // Helper to create an assignment based on the register type.
228 void createAssign(ConversionPatternRewriter &rewriter, Location loc,
229 sv::RegOp svReg, OpAdaptor reg) const;
230
231private:
232 bool lowerToAlwaysFF;
233 const MapVector<StringAttr, ModuleLoweringState> &moduleLoweringStates;
234};
235
236/// Create the assign.
237template <>
238void CompRegLower<CompRegOp>::createAssign(ConversionPatternRewriter &rewriter,
239 Location loc, sv::RegOp svReg,
240 OpAdaptor reg) const {
241 rewriter.create<sv::PAssignOp>(loc, svReg, reg.getInput());
242}
243/// Create the assign inside of an if block.
244template <>
245void CompRegLower<CompRegClockEnabledOp>::createAssign(
246 ConversionPatternRewriter &rewriter, Location loc, sv::RegOp svReg,
247 OpAdaptor reg) const {
248 rewriter.create<sv::IfOp>(loc, reg.getClockEnable(), [&]() {
249 rewriter.create<sv::PAssignOp>(loc, svReg, reg.getInput());
250 });
251}
252
253/// Lower FromImmutable to `sv.reg` and `sv.initial`.
254class FromImmutableLowering : public OpConversionPattern<FromImmutableOp> {
255public:
256 FromImmutableLowering(
257 TypeConverter &typeConverter, MLIRContext *context,
258 const MapVector<StringAttr, ModuleLoweringState> &moduleLoweringStates)
259 : OpConversionPattern<FromImmutableOp>(typeConverter, context),
260 moduleLoweringStates(moduleLoweringStates) {}
261
262 using OpAdaptor = typename OpConversionPattern<FromImmutableOp>::OpAdaptor;
263
264 LogicalResult
265 matchAndRewrite(FromImmutableOp fromImmutableOp, OpAdaptor adaptor,
266 ConversionPatternRewriter &rewriter) const final {
267 Location loc = fromImmutableOp.getLoc();
268
269 auto regTy = ConversionPattern::getTypeConverter()->convertType(
270 fromImmutableOp.getType());
271 auto svReg = rewriter.create<sv::RegOp>(loc, regTy);
272
273 auto regVal = rewriter.create<sv::ReadInOutOp>(loc, svReg);
274
275 // Lower initial values.
276 auto module = fromImmutableOp->template getParentOfType<hw::HWModuleOp>();
277 const auto &initial = moduleLoweringStates.find(module.getModuleNameAttr())
278 ->second.immutableValueLowering;
279
280 Value initialValue =
281 initial.lookupImmutableValue(fromImmutableOp.getInput());
282
283 OpBuilder::InsertionGuard guard(rewriter);
284 auto in = initial.getSVInitial();
285 rewriter.setInsertionPointToEnd(in.getBodyBlock());
286 rewriter.create<sv::BPAssignOp>(fromImmutableOp->getLoc(), svReg,
287 initialValue);
288
289 rewriter.replaceOp(fromImmutableOp, regVal);
290 return success();
291 }
292
293private:
294 const MapVector<StringAttr, ModuleLoweringState> &moduleLoweringStates;
295};
296// Lower seq.clock_gate to a fairly standard clock gate implementation.
297//
298class ClockGateLowering : public OpConversionPattern<ClockGateOp> {
299public:
301 using OpAdaptor = typename OpConversionPattern<ClockGateOp>::OpAdaptor;
302 LogicalResult
303 matchAndRewrite(ClockGateOp clockGate, OpAdaptor adaptor,
304 ConversionPatternRewriter &rewriter) const final {
305 auto loc = clockGate.getLoc();
306 Value clk = adaptor.getInput();
307
308 // enable in
309 Value enable = adaptor.getEnable();
310 if (auto te = adaptor.getTestEnable())
311 enable = rewriter.create<comb::OrOp>(loc, enable, te);
312
313 // Enable latch.
314 Value enableLatch = rewriter.create<sv::RegOp>(
315 loc, rewriter.getI1Type(), rewriter.getStringAttr("cg_en_latch"));
316
317 // Latch the enable signal using an always @* block.
318 rewriter.create<sv::AlwaysOp>(
319 loc, llvm::SmallVector<sv::EventControl>{}, llvm::SmallVector<Value>{},
320 [&]() {
321 rewriter.create<sv::IfOp>(
322 loc, comb::createOrFoldNot(loc, clk, rewriter), [&]() {
323 rewriter.create<sv::PAssignOp>(loc, enableLatch, enable);
324 });
325 });
326
327 // Create the gated clock signal.
328 rewriter.replaceOpWithNewOp<comb::AndOp>(
329 clockGate, clk, rewriter.create<sv::ReadInOutOp>(loc, enableLatch));
330 return success();
331 }
332};
333
334// Lower seq.clock_inv to a regular inverter.
335//
336class ClockInverterLowering : public OpConversionPattern<ClockInverterOp> {
337public:
338 using OpConversionPattern<ClockInverterOp>::OpConversionPattern;
339
340 LogicalResult
341 matchAndRewrite(ClockInverterOp op, OpAdaptor adaptor,
342 ConversionPatternRewriter &rewriter) const final {
343 auto loc = op.getLoc();
344 Value clk = adaptor.getInput();
345
346 StringAttr name = op->getAttrOfType<StringAttr>("sv.namehint");
347 Value one = rewriter.create<hw::ConstantOp>(loc, APInt(1, 1));
348 auto newOp = rewriter.replaceOpWithNewOp<comb::XorOp>(op, clk, one);
349 if (name)
350 rewriter.modifyOpInPlace(newOp,
351 [&] { newOp->setAttr("sv.namehint", name); });
352 return success();
353 }
354};
355
356// Lower seq.clock_mux to a `comb.mux` op
357//
358class ClockMuxLowering : public OpConversionPattern<ClockMuxOp> {
359public:
361 using OpConversionPattern<ClockMuxOp>::OpAdaptor;
362
363 LogicalResult
364 matchAndRewrite(ClockMuxOp clockMux, OpAdaptor adaptor,
365 ConversionPatternRewriter &rewriter) const final {
366 rewriter.replaceOpWithNewOp<comb::MuxOp>(clockMux, adaptor.getCond(),
367 adaptor.getTrueClock(),
368 adaptor.getFalseClock(), true);
369 return success();
370 }
371};
372
373/// Map `seq.clock` to `i1`.
374struct SeqToSVTypeConverter : public TypeConverter {
375 SeqToSVTypeConverter() {
376 addConversion([&](Type type) { return type; });
377 addConversion([&](seq::ImmutableType type) { return type.getInnerType(); });
378 addConversion([&](seq::ClockType type) {
379 return IntegerType::get(type.getContext(), 1);
380 });
381 addConversion([&](hw::StructType structTy) {
382 bool changed = false;
383
384 SmallVector<hw::StructType::FieldInfo> newFields;
385 for (auto field : structTy.getElements()) {
386 auto &newField = newFields.emplace_back();
387 newField.name = field.name;
388 newField.type = convertType(field.type);
389 if (field.type != newField.type)
390 changed = true;
391 }
392
393 if (!changed)
394 return structTy;
395
396 return hw::StructType::get(structTy.getContext(), newFields);
397 });
398 addConversion([&](hw::ArrayType arrayTy) {
399 auto elementTy = arrayTy.getElementType();
400 auto newElementTy = convertType(elementTy);
401 if (elementTy != newElementTy)
402 return hw::ArrayType::get(newElementTy, arrayTy.getNumElements());
403 return arrayTy;
404 });
405
406 addTargetMaterialization([&](mlir::OpBuilder &builder,
407 mlir::Type resultType, mlir::ValueRange inputs,
408 mlir::Location loc) -> mlir::Value {
409 if (inputs.size() != 1)
410 return Value();
411 return builder
412 .create<mlir::UnrealizedConversionCastOp>(loc, resultType, inputs[0])
413 ->getResult(0);
414 });
415
416 addSourceMaterialization([&](mlir::OpBuilder &builder,
417 mlir::Type resultType, mlir::ValueRange inputs,
418 mlir::Location loc) -> mlir::Value {
419 if (inputs.size() != 1)
420 return Value();
421 return builder
422 .create<mlir::UnrealizedConversionCastOp>(loc, resultType, inputs[0])
423 ->getResult(0);
424 });
425 }
426};
427
428/// Eliminate no-op clock casts.
429template <typename T>
430class ClockCastLowering : public OpConversionPattern<T> {
431public:
433
434 LogicalResult
435 matchAndRewrite(T op, typename T::Adaptor adaptor,
436 ConversionPatternRewriter &rewriter) const final {
437 // If the cast had a better name than its input, propagate it.
438 if (Operation *inputOp = adaptor.getInput().getDefiningOp())
439 if (!isa<mlir::UnrealizedConversionCastOp>(inputOp))
440 if (auto name = chooseName(op, inputOp))
441 rewriter.modifyOpInPlace(
442 inputOp, [&] { inputOp->setAttr("sv.namehint", name); });
443
444 rewriter.replaceOp(op, adaptor.getInput());
445 return success();
446 }
447};
448
449// Lower seq.const_clock to `hw.constant`
450//
451class ClockConstLowering : public OpConversionPattern<ConstClockOp> {
452public:
454 using OpConversionPattern<ConstClockOp>::OpAdaptor;
455
456 LogicalResult
457 matchAndRewrite(ConstClockOp clockConst, OpAdaptor adaptor,
458 ConversionPatternRewriter &rewriter) const final {
459 rewriter.replaceOpWithNewOp<hw::ConstantOp>(
460 clockConst, APInt(1, clockConst.getValue() == ClockConst::High));
461 return success();
462 }
463};
464
465class AggregateConstantPattern
466 : public OpConversionPattern<hw::AggregateConstantOp> {
467public:
468 using OpConversionPattern<hw::AggregateConstantOp>::OpConversionPattern;
469 using OpConversionPattern<hw::AggregateConstantOp>::OpAdaptor;
470
471 LogicalResult
472 matchAndRewrite(hw::AggregateConstantOp aggregateConstant, OpAdaptor adaptor,
473 ConversionPatternRewriter &rewriter) const final {
474 auto newType = typeConverter->convertType(aggregateConstant.getType());
475 auto newAttr = aggregateConstant.getFieldsAttr().replace(
476 [](seq::ClockConstAttr clockConst) {
477 return mlir::IntegerAttr::get(
478 mlir::IntegerType::get(clockConst.getContext(), 1),
479 APInt(1, clockConst.getValue() == ClockConst::High));
480 });
481 rewriter.replaceOpWithNewOp<hw::AggregateConstantOp>(
482 aggregateConstant, newType, cast<ArrayAttr>(newAttr));
483 return success();
484 }
485};
486
487/// Lower `seq.clock_div` to a behavioural clock divider
488///
489class ClockDividerLowering : public OpConversionPattern<ClockDividerOp> {
490public:
491 using OpConversionPattern<ClockDividerOp>::OpConversionPattern;
492 using OpConversionPattern<ClockDividerOp>::OpAdaptor;
493
494 LogicalResult
495 matchAndRewrite(ClockDividerOp clockDiv, OpAdaptor adaptor,
496 ConversionPatternRewriter &rewriter) const final {
497 Location loc = clockDiv.getLoc();
498
499 Value one;
500 if (clockDiv.getPow2()) {
501 one = rewriter.create<hw::ConstantOp>(loc, APInt(1, 1));
502 }
503
504 Value output = clockDiv.getInput();
505
506 SmallVector<Value> regs;
507 for (unsigned i = 0; i < clockDiv.getPow2(); ++i) {
508 Value reg = rewriter.create<sv::RegOp>(
509 loc, rewriter.getI1Type(),
510 rewriter.getStringAttr("clock_out_" + std::to_string(i)));
511 regs.push_back(reg);
512
513 rewriter.create<sv::AlwaysOp>(
514 loc, sv::EventControl::AtPosEdge, output, [&] {
515 Value outputVal = rewriter.create<sv::ReadInOutOp>(loc, reg);
516 Value inverted = rewriter.create<comb::XorOp>(loc, outputVal, one);
517 rewriter.create<sv::BPAssignOp>(loc, reg, inverted);
518 });
519
520 output = rewriter.create<sv::ReadInOutOp>(loc, reg);
521 }
522
523 if (!regs.empty()) {
524 Value zero = rewriter.create<hw::ConstantOp>(loc, APInt(1, 0));
525 rewriter.create<sv::InitialOp>(loc, [&] {
526 for (Value reg : regs) {
527 rewriter.create<sv::BPAssignOp>(loc, reg, zero);
528 }
529 });
530 }
531
532 rewriter.replaceOp(clockDiv, output);
533 return success();
534 }
535};
536
537} // namespace
538
539// NOLINTBEGIN(misc-no-recursion)
540static bool isLegalType(Type ty) {
541 if (hw::type_isa<ClockType>(ty))
542 return false;
543
544 if (auto arrayTy = hw::type_dyn_cast<hw::ArrayType>(ty))
545 return isLegalType(arrayTy.getElementType());
546
547 if (auto structTy = hw::type_dyn_cast<hw::StructType>(ty)) {
548 for (auto field : structTy.getElements())
549 if (!isLegalType(field.type))
550 return false;
551 return true;
552 }
553
554 return true;
555}
556// NOLINTEND(misc-no-recursion)
557
558static bool isLegalOp(Operation *op) {
559 if (auto module = dyn_cast<hw::HWModuleLike>(op)) {
560 for (auto port : module.getHWModuleType().getPorts())
561 if (!isLegalType(port.type))
562 return false;
563 return true;
564 }
565
566 if (auto hwAggregateConstantOp = dyn_cast<hw::AggregateConstantOp>(op)) {
567 bool foundClockAttr = false;
568 hwAggregateConstantOp.getFieldsAttr().walk(
569 [&](seq::ClockConstAttr attr) { foundClockAttr = true; });
570 if (foundClockAttr)
571 return false;
572 }
573
574 bool allOperandsLowered = llvm::all_of(
575 op->getOperands(), [](auto op) { return isLegalType(op.getType()); });
576 bool allResultsLowered = llvm::all_of(op->getResults(), [](auto result) {
577 return isLegalType(result.getType());
578 });
579 return allOperandsLowered && allResultsLowered;
580}
581
583 auto circuit = getOperation();
584 MLIRContext *context = &getContext();
585
586 auto modules = llvm::to_vector(circuit.getOps<HWModuleOp>());
587
588 FirMemLowering memLowering(circuit);
589
590 // Identify memories and group them by module.
591 auto uniqueMems = memLowering.collectMemories(modules);
592 MapVector<HWModuleOp, SmallVector<FirMemLowering::MemoryConfig>> memsByModule;
593 SmallVector<HWModuleGeneratedOp> generatedModules;
594 for (auto &[config, memOps] : uniqueMems) {
595 // Create the `HWModuleGeneratedOp`s for each unique configuration.
596 auto genOp = memLowering.createMemoryModule(config, memOps);
597 generatedModules.push_back(genOp);
598
599 // Group memories by their parent module for parallelism.
600 for (auto memOp : memOps) {
601 auto parent = memOp->getParentOfType<HWModuleOp>();
602 memsByModule[parent].emplace_back(&config, genOp, memOp);
603 }
604 }
605
606 // Lower memories and registers in modules in parallel.
607 std::atomic<bool> needsRegRandomization = false;
608 std::atomic<bool> needsMemRandomization = false;
609
610 MapVector<StringAttr, ModuleLoweringState> moduleLoweringStates;
611 for (auto module : circuit.getOps<HWModuleOp>())
612 moduleLoweringStates.try_emplace(module.getModuleNameAttr(),
613 ModuleLoweringState(module));
614
615 auto result = mlir::failableParallelForEach(
616 &getContext(), moduleLoweringStates, [&](auto &moduleAndState) {
617 auto &state = moduleAndState.second;
618 auto module = state.module;
619 SeqToSVTypeConverter typeConverter;
620 FirRegLowering regLowering(typeConverter, module,
621 disableRegRandomization,
622 emitSeparateAlwaysBlocks);
623 regLowering.lower();
624 if (regLowering.needsRegRandomization()) {
625 if (!disableRegRandomization) {
626 state.fragment.needsRegFragment = true;
627 }
628 needsRegRandomization = true;
629 }
630 numSubaccessRestored += regLowering.numSubaccessRestored;
631
632 if (auto *it = memsByModule.find(module); it != memsByModule.end()) {
633 memLowering.lowerMemoriesInModule(module, it->second);
634 // Generated memories need register randomization since `HWMemSimImpl`
635 // may add registers.
636 needsMemRandomization = true;
637 needsRegRandomization = true;
638 }
639 return state.immutableValueLowering.lower();
640 });
641
642 if (failed(result))
643 return signalPassFailure();
644
645 auto randomInitFragmentName =
646 FlatSymbolRefAttr::get(context, "RANDOM_INIT_FRAGMENT");
647 auto randomInitRegFragmentName =
648 FlatSymbolRefAttr::get(context, "RANDOM_INIT_REG_FRAGMENT");
649 auto randomInitMemFragmentName =
650 FlatSymbolRefAttr::get(context, "RANDOM_INIT_MEM_FRAGMENT");
651
652 for (auto &[_, state] : moduleLoweringStates) {
653 const auto &info = state.fragment;
654 // Do not add fragments if not needed.
655 if (!info.needsRegFragment) {
656 continue;
657 }
658
659 SmallVector<Attribute> fragmentAttrs;
660 auto module = state.module;
661 if (auto others =
662 module->getAttrOfType<ArrayAttr>(emit::getFragmentsAttrName()))
663 fragmentAttrs = llvm::to_vector(others);
664
665 if (info.needsRegFragment) {
666 fragmentAttrs.push_back(randomInitRegFragmentName);
667 fragmentAttrs.push_back(randomInitFragmentName);
668 }
669
670 module->setAttr(emit::getFragmentsAttrName(),
671 ArrayAttr::get(context, fragmentAttrs));
672 }
673
674 // Set fragments for generated modules.
675 SmallVector<Attribute> genModFragments;
676 if (!disableRegRandomization)
677 genModFragments.push_back(randomInitRegFragmentName);
678 if (!disableMemRandomization)
679 genModFragments.push_back(randomInitMemFragmentName);
680 if (!genModFragments.empty()) {
681 genModFragments.push_back(randomInitFragmentName);
682 auto fragmentAttr = ArrayAttr::get(context, genModFragments);
683 for (auto genOp : generatedModules)
684 genOp->setAttr(emit::getFragmentsAttrName(), fragmentAttr);
685 }
686
687 // Mark all ops which can have clock types as illegal.
688 SeqToSVTypeConverter typeConverter;
689 ConversionTarget target(*context);
690 target.addIllegalDialect<SeqDialect>();
691 target.markUnknownOpDynamicallyLegal(isLegalOp);
692
693 RewritePatternSet patterns(context);
694 patterns.add<CompRegLower<CompRegOp>>(typeConverter, context, lowerToAlwaysFF,
695 moduleLoweringStates);
696 patterns.add<CompRegLower<CompRegClockEnabledOp>>(
697 typeConverter, context, lowerToAlwaysFF, moduleLoweringStates);
698 patterns.add<FromImmutableLowering>(typeConverter, context,
699 moduleLoweringStates);
700 patterns.add<ClockCastLowering<seq::FromClockOp>>(typeConverter, context);
701 patterns.add<ClockCastLowering<seq::ToClockOp>>(typeConverter, context);
702 patterns.add<ClockGateLowering>(typeConverter, context);
703 patterns.add<ClockInverterLowering>(typeConverter, context);
704 patterns.add<ClockMuxLowering>(typeConverter, context);
705 patterns.add<ClockDividerLowering>(typeConverter, context);
706 patterns.add<ClockConstLowering>(typeConverter, context);
707 patterns.add<TypeConversionPattern>(typeConverter, context);
708 patterns.add<AggregateConstantPattern>(typeConverter, context);
709
710 if (failed(applyPartialConversion(circuit, target, std::move(patterns))))
711 signalPassFailure();
712
713 auto loc = UnknownLoc::get(context);
714 auto b = ImplicitLocOpBuilder::atBlockBegin(loc, circuit.getBody());
715 if (needsRegRandomization || needsMemRandomization) {
716 b.create<sv::MacroDeclOp>("ENABLE_INITIAL_REG_");
717 b.create<sv::MacroDeclOp>("ENABLE_INITIAL_MEM_");
718 if (needsRegRandomization) {
719 b.create<sv::MacroDeclOp>("FIRRTL_BEFORE_INITIAL");
720 b.create<sv::MacroDeclOp>("FIRRTL_AFTER_INITIAL");
721 }
722 if (needsMemRandomization)
723 b.create<sv::MacroDeclOp>("RANDOMIZE_MEM_INIT");
724 b.create<sv::MacroDeclOp>("RANDOMIZE_REG_INIT");
725 b.create<sv::MacroDeclOp>("RANDOMIZE");
726 b.create<sv::MacroDeclOp>("RANDOMIZE_DELAY");
727 b.create<sv::MacroDeclOp>("RANDOM");
728 b.create<sv::MacroDeclOp>("INIT_RANDOM");
729 b.create<sv::MacroDeclOp>("INIT_RANDOM_PROLOG_");
730 }
731
732 bool hasRegRandomization = needsRegRandomization && !disableRegRandomization;
733 bool hasMemRandomization = needsMemRandomization && !disableMemRandomization;
734 if (!hasRegRandomization && !hasMemRandomization)
735 return;
736
737 // Build macros for FIRRTL-style register and memory initialization.
738 // Insert them at the start of the module, after any other verbatims.
739 for (Operation &op : *circuit.getBody()) {
740 if (!isa<sv::VerbatimOp, sv::IfDefOp>(&op)) {
741 b.setInsertionPoint(&op);
742 break;
743 }
744 }
745
746 // Create SYNTHESIS/VERILATOR macros if other passes have not done so already.
747 {
748 StringSet<> symbols;
749 for (auto sym : circuit.getOps<sv::MacroDeclOp>())
750 symbols.insert(sym.getName());
751 if (!symbols.count("SYNTHESIS"))
752 b.create<sv::MacroDeclOp>("SYNTHESIS");
753 if (!symbols.count("VERILATOR"))
754 b.create<sv::MacroDeclOp>("VERILATOR");
755 }
756
757 // TODO: We could have an operation for macros and uses of them, and
758 // even turn them into symbols so we can DCE unused macro definitions.
759 auto emitGuardedDefine = [&](StringRef guard, StringRef defName,
760 StringRef defineTrue = "",
761 StringRef defineFalse = StringRef()) {
762 if (!defineFalse.data()) {
763 assert(defineTrue.data() && "didn't define anything");
764 b.create<sv::IfDefOp>(
765 guard, [&]() { b.create<sv::MacroDefOp>(defName, defineTrue); });
766 } else {
767 b.create<sv::IfDefOp>(
768 guard,
769 [&]() {
770 if (defineTrue.data())
771 b.create<sv::MacroDefOp>(defName, defineTrue);
772 },
773 [&]() { b.create<sv::MacroDefOp>(defName, defineFalse); });
774 }
775 };
776
777 // Helper function to emit #ifndef guard.
778 auto emitGuard = [&](const char *guard, llvm::function_ref<void(void)> body) {
779 b.create<sv::IfDefOp>(
780 guard, []() {}, body);
781 };
782
783 b.create<emit::FragmentOp>(randomInitFragmentName.getAttr(), [&] {
784 b.create<sv::VerbatimOp>(
785 "// Standard header to adapt well known macros for "
786 "register randomization.");
787
788 b.create<sv::VerbatimOp>(
789 "\n// RANDOM may be set to an expression that produces a 32-bit "
790 "random unsigned value.");
791 emitGuardedDefine("RANDOM", "RANDOM", StringRef(), "$random");
792
793 b.create<sv::VerbatimOp>(
794 "\n// Users can define INIT_RANDOM as general code that gets "
795 "injected "
796 "into the\n// initializer block for modules with registers.");
797 emitGuardedDefine("INIT_RANDOM", "INIT_RANDOM", StringRef(), "");
798
799 b.create<sv::VerbatimOp>(
800 "\n// If using random initialization, you can also define "
801 "RANDOMIZE_DELAY to\n// customize the delay used, otherwise 0.002 "
802 "is used.");
803 emitGuardedDefine("RANDOMIZE_DELAY", "RANDOMIZE_DELAY", StringRef(),
804 "0.002");
805
806 b.create<sv::VerbatimOp>(
807 "\n// Define INIT_RANDOM_PROLOG_ for use in our modules below.");
808 emitGuard("INIT_RANDOM_PROLOG_", [&]() {
809 b.create<sv::IfDefOp>(
810 "RANDOMIZE",
811 [&]() {
812 emitGuardedDefine("VERILATOR", "INIT_RANDOM_PROLOG_",
813 "`INIT_RANDOM",
814 "`INIT_RANDOM #`RANDOMIZE_DELAY begin end");
815 },
816 [&]() { b.create<sv::MacroDefOp>("INIT_RANDOM_PROLOG_", ""); });
817 });
818 });
819
820 if (hasMemRandomization) {
821 b.create<emit::FragmentOp>(randomInitMemFragmentName.getAttr(), [&] {
822 b.create<sv::VerbatimOp>("\n// Include rmemory initializers in init "
823 "blocks unless synthesis is set");
824 emitGuard("RANDOMIZE", [&]() {
825 emitGuardedDefine("RANDOMIZE_MEM_INIT", "RANDOMIZE");
826 });
827 emitGuard("SYNTHESIS", [&] {
828 emitGuardedDefine("ENABLE_INITIAL_MEM_", "ENABLE_INITIAL_MEM_",
829 StringRef(), "");
830 });
831 b.create<sv::VerbatimOp>("");
832 });
833 }
834
835 if (hasRegRandomization) {
836 b.create<emit::FragmentOp>(randomInitRegFragmentName.getAttr(), [&] {
837 b.create<sv::VerbatimOp>("\n// Include register initializers in init "
838 "blocks unless synthesis is set");
839 emitGuard("RANDOMIZE", [&]() {
840 emitGuardedDefine("RANDOMIZE_REG_INIT", "RANDOMIZE");
841 });
842 emitGuard("SYNTHESIS", [&] {
843 emitGuardedDefine("ENABLE_INITIAL_REG_", "ENABLE_INITIAL_REG_",
844 StringRef(), "");
845 });
846 b.create<sv::VerbatimOp>("");
847 });
848 }
849}
850
851std::unique_ptr<Pass>
852circt::createLowerSeqToSVPass(const LowerSeqToSVOptions &options) {
853 return std::make_unique<SeqToSVPass>(options);
854}
assert(baseType &&"element must be base type")
static bool isLegalOp(Operation *op)
Returns true if the given op is considered as legal - i.e.
Definition DCToHW.cpp:844
static FIRRTLBaseType convertType(FIRRTLBaseType type)
Returns null type if no conversion is needed.
Definition DropConst.cpp:32
static bool isLegalType(Type ty)
Definition SeqToSV.cpp:540
static bool isLegalOp(Operation *op)
Definition SeqToSV.cpp:558
FIR memory lowering helper.
UniqueConfigs collectMemories(ArrayRef< hw::HWModuleOp > modules)
Groups memories by their kind from the whole design.
void lowerMemoriesInModule(hw::HWModuleOp module, ArrayRef< MemoryConfig > mems)
Lowers a group of memories from the same module.
hw::HWModuleGeneratedOp createMemoryModule(FirMemConfig &mem, ArrayRef< seq::FirMemOp > memOps)
Creates the generated module for a given configuration.
Lower FirRegOp to sv.reg and sv.always.
bool needsRegRandomization() const
create(data_type, value)
Definition hw.py:433
create(value)
Definition sv.py:106
Definition sv.py:68
StringRef getFragmentsAttrName()
Return the name of the fragments array attribute.
Definition EmitOps.h:32
FailureOr< seq::InitialOp > mergeInitialOps(Block *block)
Definition SeqOps.cpp:1093
mlir::ArrayAttr getSVAttributes(mlir::Operation *op)
Return all the SV attributes of an operation, or null if there are none.
void setSVAttributes(mlir::Operation *op, mlir::ArrayAttr attrs)
Set the SV attributes of an operation.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
std::unique_ptr< mlir::Pass > createLowerSeqToSVPass(const LowerSeqToSVOptions &options={})
Definition SeqToSV.cpp:852
StringRef chooseName(StringRef a, StringRef b)
Choose a good name for an item from two options.
Definition Naming.cpp:47
Definition seq.py:1
reg(value, clock, reset=None, reset_value=None, name=None, sym_name=None)
Definition seq.py:21
void runOnOperation() override
Definition SeqToSV.cpp:582
Generic pattern which replaces an operation by one of the same operation name, but with converted att...