20#include "mlir/Transforms/Passes.h"
21#include "llvm/Support/FileSystem.h"
22#include "llvm/Support/Path.h"
29 pm.nest<firrtl::CircuitOp>().addPass(
30 firrtl::createCheckRecursiveInstantiation());
31 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createCheckLayers());
33 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerOpenAggs());
35 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createResolvePaths());
37 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerFIRRTLAnnotations(
43 pm.nest<firrtl::CircuitOp>().addNestedPass<firrtl::FModuleOp>(
44 firrtl::createMaterializeDebugInfo());
46 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerIntmodules(
48 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
49 firrtl::createLowerIntrinsics());
56 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerSignatures());
62 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInjectDUTHierarchy());
66 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
67 mlir::createCSEPass());
69 pm.nest<firrtl::CircuitOp>().nestAny().addPass(mlir::createCSEPass());
72 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
73 firrtl::createPassiveWires());
75 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
78 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
79 firrtl::createLowerCHIRRTLPass());
83 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
84 firrtl::createLowerMatches());
87 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferWidths());
89 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createMemToRegOfVec(
92 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferResets());
95 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createSpecializeOption(
99 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createDropConst());
102 firrtl::DedupOptions opts;
104 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createDedup(opts));
108 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createLowerFIRRTLTypes(
111 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createVBToBV());
115 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
116 firrtl::createFlattenMemory());
121 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createLowerFIRRTLTypes(
126 auto &modulePM = pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>();
127 modulePM.addPass(firrtl::createExpandWhens());
128 modulePM.addPass(firrtl::createSFCCompat());
136 if (
auto mode = FirtoolOptions::toInferDomainsPassMode(opt.
getDomainMode()))
137 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferDomains({*mode}));
139 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createCheckCombLoops());
143 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createSpecializeLayers());
147 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createProbesToSignals());
149 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInliner());
151 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
152 firrtl::createLayerMerge());
159 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
160 firrtl::createRandomizeRegisterInit());
164 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
170 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
171 firrtl::createInferReadWrite());
174 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerMemory());
177 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createIMConstProp());
179 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createAddSeqMemPorts());
181 pm.addPass(firrtl::createCreateSiFiveMetadata(
188 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createExtractInstances());
192 pm.addNestedPass<firrtl::CircuitOp>(mlir::createSymbolDCEPass());
195 pm.addPass(firrtl::createInnerSymbolDCE());
202 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
203 circt::firrtl::createEliminateWires());
204 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
206 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
207 circt::firrtl::createRegisterOptimizer());
210 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createIMConstProp());
211 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
213 pm.addPass(firrtl::createIMDeadCodeElim());
215 pm.nest<firrtl::CircuitOp>().addPass(
216 firrtl::createAnnotateInputOnlyModules());
217 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInliner());
218 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
224 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
225 firrtl::createMergeConnections(
230 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
231 firrtl::createVectorization());
238 StringRef inputFilename) {
241 pm.nest<firrtl::CircuitOp>().addPass(
242 firrtl::createPopulateInstanceChoiceSymbols());
248 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLayerSink());
253 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerXMR());
262 pm.nest<firrtl::CircuitOp>().addPass(
265 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
269 if (outputFilename ==
"-")
272 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createAssignOutputDirs(
273 {outputFilename.str()}));
281 pm.addNestedPass<firrtl::CircuitOp>(
287 ? llvm::sys::path::parent_path(inputFilename)
289 pm.nest<firrtl::CircuitOp>().addPass(
290 firrtl::createBlackBoxReader({blackBoxRoot.str()}));
294 pm.nest<firrtl::CircuitOp>().addPass(
297 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerDPI());
298 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerDomains());
299 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerClasses());
302 pm.nest<firrtl::CircuitOp>().addPass(circt::firrtl::createLint(
312 modulePM.addPass(mlir::createCSEPass());
317 pm.addPass(hw::createVerifyInnerRefNamespace());
320 pm.addNestedPass<
hw::HWModuleOp>(verif::createVerifyClockedAssertLikePass());
327 pm.nestAny().addPass(verif::createStripContractsPass());
328 pm.addPass(verif::createLowerTestsPass());
336 FirtoolOptions::RandomKind::Reg),
342 pm.addPass(seq::createHWMemSimImpl(
344 FirtoolOptions::RandomKind::Mem),
349 ? seq::ReadEnableMode::Ignore
350 : seq::ReadEnableMode::Undefined,
358 modulePM.addPass(mlir::createCSEPass());
360 modulePM.addPass(mlir::createCSEPass());
361 modulePM.addPass(sv::createHWCleanup(
366 pm.addPass(hw::createVerifyInnerRefNamespace());
377 pm.addNestedPass<
hw::HWModuleOp>(verif::createVerifyClockedAssertLikePass());
388 if (
auto fileLoc = dyn_cast<FileLineColLoc>(loc))
389 return fileLoc.getFilename().getValue().ends_with(
".fir");
395 [](mlir::Location loc) {
return true; }));
399 pm.addPass(sv::createHWExportModuleHierarchy());
402 pm.addPass(hw::createVerifyInnerRefNamespace());
410 std::unique_ptr<llvm::raw_ostream> os) {
420 llvm::raw_ostream &os) {
430 llvm::StringRef directory) {
440 pm.addPass(firrtl::createFinalizeIR());
441 pm.addPass(om::createFreezePathsPass());
442 om::ElaborateObjectOptions options;
443 options.allPublicClasses =
true;
444 options.allowUnevaluated =
true;
445 pm.addPass(om::createElaborateObject(options));
455 llvm::raw_ostream &os) {
460 mpm.addPass(circt::seq::createLowerSeqShiftReg());
462 mpm.addPass(circt::seq::createLowerSeqCompRegCE());
464 mpm.addPass(circt::verif::createPrepareForFormalPass());
465 pm.addPass(circt::hw::createFlattenModules());
478class FirtoolCmdOptions {
482 llvm::cl::desc(
"Output filename, or directory for split output"),
483 llvm::cl::value_desc(
"filename"),
488 "disable-annotation-unknown",
489 llvm::cl::desc(
"Ignore unknown annotations when parsing"),
490 llvm::cl::init(
false)};
493 "disable-annotation-classless",
494 llvm::cl::desc(
"Ignore annotations without a class when parsing"),
495 llvm::cl::init(
false)};
498 "lower-annotations-no-ref-type-ports",
500 "Create real ports instead of ref type ports when resolving "
501 "wiring problems inside the LowerAnnotations pass"),
502 llvm::cl::init(
false), llvm::cl::Hidden};
506 llvm::cl::desc(
"Convert probes to non-probe signals"),
507 llvm::cl::init(
false), llvm::cl::Hidden};
511 "preserve-aggregate",
512 llvm::cl::desc(
"Specify input file format:"),
515 "Preserve no aggregate"),
517 "Preserve only 1d vectors of ground type"),
519 "Preserve only vectors"),
521 "Preserve vectors and bundles")),
527 llvm::cl::desc(
"Specify the values which can be optimized away"),
530 "Strip all names. No name is preserved"),
532 "Names could be preserved by best-effort unlike `strip`"),
534 "Preserve values with meaningful names"),
536 "Preserve all values")),
540 "g", llvm::cl::desc(
"Enable the generation of debug information"),
541 llvm::cl::init(
false)};
545 "O", llvm::cl::desc(
"Controls how much optimization should be performed"),
548 "Compile with only necessary optimizations"),
550 "release",
"Compile with optimizations")),
554 llvm::cl::desc(
"Disable layer sink"),
559 llvm::cl::desc(
"Disable optimizations"),
564 llvm::cl::desc(
"Transform vectors of bundles to bundles of vectors"),
565 llvm::cl::init(
false)};
569 llvm::cl::desc(
"Disable deduplication of structurally identical modules"),
570 llvm::cl::init(
false)};
575 "Deduplicate FIRRTL classes, violating their nominal typing"),
576 llvm::cl::init(
true)};
579 "grand-central-companion-mode",
580 llvm::cl::desc(
"Specifies the handling of Grand Central companions"),
582 clEnumValN(firrtl::CompanionMode::Bind,
"bind",
583 "Lower companion instances to SystemVerilog binds"),
584 clEnumValN(firrtl::CompanionMode::Instantiate,
"instantiate",
585 "Instantiate companions in the design"),
586 clEnumValN(firrtl::CompanionMode::Drop,
"drop",
587 "Remove companions from the design")),
588 llvm::cl::init(firrtl::CompanionMode::Bind),
595 "Disable lowering of FIRRTL view intrinsics (delete them instead)"),
596 llvm::cl::init(
false),
600 "disable-aggressive-merge-connections",
602 "Disable aggressive merge connections (i.e. merge all field-level "
603 "connections into bulk connections)"),
604 llvm::cl::init(
false)};
608 llvm::cl::desc(
"Lower memories to have memories with masks as an "
609 "array with one memory per ground type"),
610 llvm::cl::init(
false)};
615 "Optional path to use as the root of black box annotations"),
616 llvm::cl::value_desc(
"path"),
622 llvm::cl::desc(
"Replace the seq mem for macro replacement and emit "
623 "relevant metadata"),
624 llvm::cl::init(
false)};
627 "repl-seq-mem-file", llvm::cl::desc(
"File name for seq mem metadata"),
631 "ignore-read-enable-mem",
632 llvm::cl::desc(
"Ignore the read enable signal, instead of "
633 "assigning X on read disable"),
634 llvm::cl::init(
false)};
637 firtool::FirtoolOptions::RandomKind::None;
644 llvm::cl::OptionCategory randomizationCategory{
645 "Disable random initialization code (may break semantics!)"};
648 "disable-mem-randomization",
649 llvm::cl::desc(
"Disable emission of memory randomization code"),
650 llvm::cl::cat(randomizationCategory), llvm::cl::ValueDisallowed,
651 llvm::cl::callback([&](
const bool &) {
653 disableRandomValue, firtool::FirtoolOptions::RandomKind::Mem);
657 "disable-reg-randomization",
658 llvm::cl::desc(
"Disable emission of register randomization code"),
659 llvm::cl::cat(randomizationCategory), llvm::cl::ValueDisallowed,
660 llvm::cl::callback([&](
const bool &) {
662 disableRandomValue, firtool::FirtoolOptions::RandomKind::Reg);
666 "disable-all-randomization",
667 llvm::cl::desc(
"Disable emission of all randomization code"),
668 llvm::cl::cat(randomizationCategory), llvm::cl::ValueDisallowed,
669 llvm::cl::callback([&](
const bool &) {
670 disableRandomValue = firtool::FirtoolOptions::RandomKind::All;
675 "output-annotation-file",
676 llvm::cl::desc(
"Optional output annotation file"),
677 llvm::cl::CommaSeparated, llvm::cl::value_desc(
"filename")};
680 "warn-on-unprocessed-annotations",
682 "Warn about annotations that were not removed by lower-to-hw"),
683 llvm::cl::init(
false)};
687 llvm::cl::desc(
"Prefer core dialects over direct SV lowering for FIRRTL "
688 "verification and printf operations"),
689 llvm::cl::init(
false)};
693 llvm::cl::desc(
"Annotate mux pragmas for memory array access"),
694 llvm::cl::init(
false)};
697 "verification-flavor",
698 llvm::cl::desc(
"Specify a verification flavor used in LowerFIRRTLToHW"),
700 clEnumValN(firrtl::VerificationFlavor::None,
"none",
701 "Use the flavor specified by the op"),
702 clEnumValN(firrtl::VerificationFlavor::IfElseFatal,
"if-else-fatal",
703 "Use Use `if(cond) else $fatal(..)` format"),
704 clEnumValN(firrtl::VerificationFlavor::Immediate,
"immediate",
705 "Use immediate verif statements"),
706 clEnumValN(firrtl::VerificationFlavor::SVA,
"sva",
"Use SVA")),
707 llvm::cl::init(firrtl::VerificationFlavor::None)};
710 "emit-separate-always-blocks",
712 "Prevent always blocks from being merged and emit constructs into "
713 "separate always blocks whenever possible"),
714 llvm::cl::init(
false)};
717 "add-vivado-ram-address-conflict-synthesis-bug-workaround",
719 "Add a vivado specific SV attribute (* ram_style = "
720 "\"distributed\" *) to unpacked array registers as a workaronud "
721 "for a vivado synthesis bug that incorrectly modifies "
722 "address conflict behavivor of combinational memories"),
723 llvm::cl::init(
false)};
730 "ckg-name", llvm::cl::desc(
"Clock gate module name"),
731 llvm::cl::init(
"EICG_wrapper")};
734 "ckg-input", llvm::cl::desc(
"Clock gate input port name"),
735 llvm::cl::init(
"in")};
738 "ckg-output", llvm::cl::desc(
"Clock gate output port name"),
739 llvm::cl::init(
"out")};
742 "ckg-enable", llvm::cl::desc(
"Clock gate enable port name"),
743 llvm::cl::init(
"en")};
747 llvm::cl::desc(
"Clock gate test enable port name (optional)"),
748 llvm::cl::init(
"test_en")};
751 "export-module-hierarchy",
752 llvm::cl::desc(
"Export module and instance hierarchy as JSON"),
753 llvm::cl::init(
false)};
756 "strip-fir-debug-info",
758 "Disable source fir locator information in output Verilog"),
759 llvm::cl::init(
true)};
763 llvm::cl::desc(
"Disable source locator information in output Verilog"),
764 llvm::cl::init(
false)};
767 "fixup-eicg-wrapper",
768 llvm::cl::desc(
"Lower `EICG_wrapper` modules into clock gate intrinsics"),
769 llvm::cl::init(
false)};
772 "select-default-for-unspecified-instance-choice",
774 "Specialize instance choice to default, if no option selected"),
775 llvm::cl::init(
false)};
779 llvm::cl::desc(
"Control how symbolic values are lowered"),
780 llvm::cl::init(verif::SymbolicValueLowering::ExtModule),
781 verif::symbolicValueLoweringCLValues()};
784 "disable-wire-elimination", llvm::cl::desc(
"Disable wire elimination"),
785 llvm::cl::init(
false)};
788 "emit-all-bind-files",
789 llvm::cl::desc(
"Emit bindfiles for private modules"),
790 llvm::cl::init(
false)};
793 "inline-input-only-modules", llvm::cl::desc(
"Inline input-only modules"),
794 llvm::cl::init(
false)};
797 "domain-mode", llvm::cl::desc(
"Enable domain inference and checking"),
798 llvm::cl::init(firtool::FirtoolOptions::DomainMode::Strip),
800 clEnumValN(firtool::FirtoolOptions::DomainMode::Check,
"check",
801 "Check domains without inference"),
802 clEnumValN(firtool::FirtoolOptions::DomainMode::Disable,
"disable",
803 "Disable domain checking"),
804 clEnumValN(firtool::FirtoolOptions::DomainMode::Infer,
"infer",
805 "Check domains with inference for private modules"),
806 clEnumValN(firtool::FirtoolOptions::DomainMode::InferAll,
"infer-all",
807 "Check domains with inference for both public and private "
809 clEnumValN(firtool::FirtoolOptions::DomainMode::Strip,
"strip",
810 "Erase all domain information"))};
817 "lint-static-asserts", llvm::cl::desc(
"Lint static assertions"),
818 llvm::cl::init(
true)};
822 "lint-xmrs-in-design", llvm::cl::desc(
"Lint XMRs in the design"),
823 llvm::cl::init(
false)};
839 : outputFilename(
"-"), disableAnnotationsUnknown(false),
840 disableAnnotationsClassless(false), lowerAnnotationsNoRefTypePorts(false),
841 probesToSignals(false),
842 preserveAggregate(firrtl::PreserveAggregate::None),
843 preserveMode(firrtl::PreserveValues::None), enableDebugInfo(false),
844 buildMode(BuildModeRelease), disableLayerSink(false),
845 disableOptimization(false), vbToBV(false), noDedup(false),
846 dedupClasses(true), companionMode(firrtl::CompanionMode::Bind),
847 noViews(false), disableAggressiveMergeConnections(false),
848 lowerMemories(false), blackBoxRootPath(
""), replSeqMem(false),
849 replSeqMemFile(
""), ignoreReadEnableMem(false),
850 disableRandom(
RandomKind::None), outputAnnotationFilename(
""),
851 enableAnnotationWarning(false), lowerToCore(false), addMuxPragmas(false),
852 verificationFlavor(firrtl::VerificationFlavor::None),
853 emitSeparateAlwaysBlocks(false),
854 addVivadoRAMAddressConflictSynthesisBugWorkaround(false),
855 ckgModuleName(
"EICG_wrapper"), ckgInputName(
"in"), ckgOutputName(
"out"),
856 ckgEnableName(
"en"), ckgTestEnableName(
"test_en"), ckgInstName(
"ckg"),
857 exportModuleHierarchy(false), stripFirDebugInfo(true),
858 stripDebugInfo(false), fixupEICGWrapper(false),
859 disableCSEinClasses(false), selectDefaultInstanceChoice(false),
860 symbolicValueLowering(
verif::SymbolicValueLowering::ExtModule),
861 disableWireElimination(false), lintStaticAsserts(true),
862 lintXmrsInDesign(true), emitAllBindFiles(false),
863 inlineInputOnlyModules(false), domainMode(
DomainMode::Disable) {
883 clOptions->disableAggressiveMergeConnections;
897 clOptions->addVivadoRAMAddressConflictSynthesisBugWorkaround;
@ All
Preserve all aggregate values.
@ OneDimVec
Preserve only 1d vectors of ground type (e.g. UInt<2>[3]).
@ Vec
Preserve only vectors (e.g. UInt<2>[3][3]).
@ None
Don't preserve aggregate at all.
@ None
Don't explicitly preserve any named values.
@ Strip
Strip all names. No name on declaration is preserved.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
std::unique_ptr< mlir::Pass > createExportSplitVerilogPass(llvm::StringRef directory="./")
std::unique_ptr< mlir::Pass > createLowerFIRRTLToHWPass(bool enableAnnotationWarning=false, firrtl::VerificationFlavor assertionFlavor=firrtl::VerificationFlavor::None, bool lowerToCore=false)
This is the pass constructor.
std::unique_ptr< OperationPass< hw::HWModuleOp > > createLowerVerifToSVPass()
Create the Verif to SV conversion pass.
std::unique_ptr< mlir::Pass > createLowerSeqToSVPass(const LowerSeqToSVOptions &options={})
std::unique_ptr< mlir::Pass > createLowerLTLToCorePass()
std::unique_ptr< mlir::Pass > createLowerSimToSVPass()
std::unique_ptr< Pass > createSimpleCanonicalizerPass()
Create a simple canonicalizer pass.
std::unique_ptr< mlir::Pass > createConvertHWToBTOR2Pass()
std::unique_ptr< mlir::Pass > createExportVerilogPass()
std::unique_ptr< mlir::Pass > createStripDebugInfoWithPredPass(const std::function< bool(mlir::Location)> &pred)
Creates a pass to strip debug information from a function.
LogicalResult populatePrepareForExportVerilog(mlir::PassManager &pm, const firtool::FirtoolOptions &opt)