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());
51 if (
auto mode = FirtoolOptions::toInferDomainsPassMode(opt.
getDomainMode()))
52 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferDomains({*mode}));
61 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createSpecializeOption(
64 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerSignatures());
70 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInjectDUTHierarchy());
74 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
75 mlir::createCSEPass());
77 pm.nest<firrtl::CircuitOp>().nestAny().addPass(mlir::createCSEPass());
80 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
81 firrtl::createPassiveWires());
83 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
86 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
87 firrtl::createLowerCHIRRTLPass());
91 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
92 firrtl::createLowerMatches());
95 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferWidths());
97 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createMemToRegOfVec(
101 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInferResets());
103 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createDropConst());
106 firrtl::DedupOptions opts;
108 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createDedup(opts));
112 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createLowerFIRRTLTypes(
115 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createVBToBV());
119 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
120 firrtl::createFlattenMemory());
125 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createLowerFIRRTLTypes(
130 auto &modulePM = pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>();
131 modulePM.addPass(firrtl::createExpandWhens());
132 modulePM.addPass(firrtl::createSFCCompat());
135 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createCheckCombLoops());
139 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createSpecializeLayers());
143 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createProbesToSignals());
145 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInliner());
147 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
148 firrtl::createLayerMerge());
155 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
156 firrtl::createRandomizeRegisterInit());
160 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
166 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
167 firrtl::createInferReadWrite());
170 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerMemory());
173 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createIMConstProp());
175 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createAddSeqMemPorts());
177 pm.addPass(firrtl::createCreateSiFiveMetadata(
184 pm.addNestedPass<firrtl::CircuitOp>(firrtl::createExtractInstances());
188 pm.addNestedPass<firrtl::CircuitOp>(mlir::createSymbolDCEPass());
191 pm.addPass(firrtl::createInnerSymbolDCE());
198 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
199 circt::firrtl::createEliminateWires());
200 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
202 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
203 circt::firrtl::createRegisterOptimizer());
206 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createIMConstProp());
207 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
209 pm.addPass(firrtl::createIMDeadCodeElim());
211 pm.nest<firrtl::CircuitOp>().addPass(
212 firrtl::createAnnotateInputOnlyModules());
213 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createInliner());
214 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
220 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
221 firrtl::createMergeConnections(
226 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
227 firrtl::createVectorization());
234 StringRef inputFilename) {
239 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLayerSink());
244 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerXMR());
253 pm.nest<firrtl::CircuitOp>().addPass(
256 pm.nest<firrtl::CircuitOp>().nest<firrtl::FModuleOp>().addPass(
260 if (outputFilename ==
"-")
263 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createAssignOutputDirs(
264 {outputFilename.str()}));
272 pm.addNestedPass<firrtl::CircuitOp>(
278 ? llvm::sys::path::parent_path(inputFilename)
280 pm.nest<firrtl::CircuitOp>().addPass(
281 firrtl::createBlackBoxReader({blackBoxRoot.str()}));
285 pm.nest<firrtl::CircuitOp>().addPass(
288 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerDPI());
289 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerDomains());
290 pm.nest<firrtl::CircuitOp>().addPass(firrtl::createLowerClasses());
291 pm.nest<firrtl::CircuitOp>().addPass(om::createVerifyObjectFieldsPass());
294 pm.nest<firrtl::CircuitOp>().addPass(circt::firrtl::createLint(
303 modulePM.addPass(mlir::createCSEPass());
308 pm.addPass(hw::createVerifyInnerRefNamespace());
311 pm.addPass(om::createVerifyObjectFieldsPass());
314 pm.addNestedPass<
hw::HWModuleOp>(verif::createVerifyClockedAssertLikePass());
321 pm.nestAny().addPass(verif::createStripContractsPass());
322 pm.addPass(verif::createLowerTestsPass());
327 pm.addPass(sv::createSVExtractTestCodePass(
336 FirtoolOptions::RandomKind::Reg),
342 pm.addPass(seq::createHWMemSimImplPass(
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::createHWCleanupPass(
366 pm.addPass(hw::createVerifyInnerRefNamespace());
369 pm.addPass(om::createVerifyObjectFieldsPass());
380 pm.addNestedPass<
hw::HWModuleOp>(verif::createVerifyClockedAssertLikePass());
391 if (
auto fileLoc = dyn_cast<FileLineColLoc>(loc))
392 return fileLoc.getFilename().getValue().ends_with(
".fir");
398 [](mlir::Location loc) {
return true; }));
405 pm.addPass(hw::createVerifyInnerRefNamespace());
416 std::unique_ptr<llvm::raw_ostream> os) {
426 llvm::raw_ostream &os) {
436 llvm::StringRef directory) {
446 pm.addPass(firrtl::createFinalizeIR());
447 pm.addPass(om::createFreezePathsPass());
454 llvm::raw_ostream &os) {
456 pm.addNestedPass<
hw::HWModuleOp>(circt::verif::createPrepareForFormalPass());
457 pm.addPass(circt::hw::createFlattenModules());
470struct FirtoolCmdOptions {
473 llvm::cl::desc(
"Output filename, or directory for split output"),
474 llvm::cl::value_desc(
"filename"),
479 "disable-annotation-unknown",
480 llvm::cl::desc(
"Ignore unknown annotations when parsing"),
481 llvm::cl::init(
false)};
484 "disable-annotation-classless",
485 llvm::cl::desc(
"Ignore annotations without a class when parsing"),
486 llvm::cl::init(
false)};
489 "lower-annotations-no-ref-type-ports",
491 "Create real ports instead of ref type ports when resolving "
492 "wiring problems inside the LowerAnnotations pass"),
493 llvm::cl::init(
false), llvm::cl::Hidden};
497 llvm::cl::desc(
"Convert probes to non-probe signals"),
498 llvm::cl::init(
false), llvm::cl::Hidden};
502 "preserve-aggregate",
503 llvm::cl::desc(
"Specify input file format:"),
506 "Preserve no aggregate"),
508 "Preserve only 1d vectors of ground type"),
510 "Preserve only vectors"),
512 "Preserve vectors and bundles")),
518 llvm::cl::desc(
"Specify the values which can be optimized away"),
521 "Strip all names. No name is preserved"),
523 "Names could be preserved by best-effort unlike `strip`"),
525 "Preserve values with meaningful names"),
527 "Preserve all values")),
531 "g", llvm::cl::desc(
"Enable the generation of debug information"),
532 llvm::cl::init(
false)};
536 "O", llvm::cl::desc(
"Controls how much optimization should be performed"),
539 "Compile with only necessary optimizations"),
541 "release",
"Compile with optimizations")),
545 llvm::cl::desc(
"Disable layer sink"),
550 llvm::cl::desc(
"Disable optimizations"),
555 llvm::cl::desc(
"Transform vectors of bundles to bundles of vectors"),
556 llvm::cl::init(
false)};
560 llvm::cl::desc(
"Disable deduplication of structurally identical modules"),
561 llvm::cl::init(
false)};
566 "Deduplicate FIRRTL classes, violating their nominal typing"),
567 llvm::cl::init(
true)};
570 "grand-central-companion-mode",
571 llvm::cl::desc(
"Specifies the handling of Grand Central companions"),
573 clEnumValN(firrtl::CompanionMode::Bind,
"bind",
574 "Lower companion instances to SystemVerilog binds"),
575 clEnumValN(firrtl::CompanionMode::Instantiate,
"instantiate",
576 "Instantiate companions in the design"),
577 clEnumValN(firrtl::CompanionMode::Drop,
"drop",
578 "Remove companions from the design")),
579 llvm::cl::init(firrtl::CompanionMode::Bind),
586 "Disable lowering of FIRRTL view intrinsics (delete them instead)"),
587 llvm::cl::init(
false),
591 "disable-aggressive-merge-connections",
593 "Disable aggressive merge connections (i.e. merge all field-level "
594 "connections into bulk connections)"),
595 llvm::cl::init(
false)};
599 llvm::cl::desc(
"Lower memories to have memories with masks as an "
600 "array with one memory per ground type"),
601 llvm::cl::init(
false)};
606 "Optional path to use as the root of black box annotations"),
607 llvm::cl::value_desc(
"path"),
613 llvm::cl::desc(
"Replace the seq mem for macro replacement and emit "
614 "relevant metadata"),
615 llvm::cl::init(
false)};
618 "repl-seq-mem-file", llvm::cl::desc(
"File name for seq mem metadata"),
622 "extract-test-code", llvm::cl::desc(
"Run the extract test code pass"),
623 llvm::cl::init(
false)};
626 "ignore-read-enable-mem",
627 llvm::cl::desc(
"Ignore the read enable signal, instead of "
628 "assigning X on read disable"),
629 llvm::cl::init(
false)};
633 "Disable random initialization code (may break semantics!)"),
635 clEnumValN(firtool::FirtoolOptions::RandomKind::Mem,
636 "disable-mem-randomization",
637 "Disable emission of memory randomization code"),
638 clEnumValN(firtool::FirtoolOptions::RandomKind::Reg,
639 "disable-reg-randomization",
640 "Disable emission of register randomization code"),
641 clEnumValN(firtool::FirtoolOptions::RandomKind::All,
642 "disable-all-randomization",
643 "Disable emission of all randomization code")),
644 llvm::cl::init(firtool::FirtoolOptions::RandomKind::None)};
647 "output-annotation-file",
648 llvm::cl::desc(
"Optional output annotation file"),
649 llvm::cl::CommaSeparated, llvm::cl::value_desc(
"filename")};
652 "warn-on-unprocessed-annotations",
654 "Warn about annotations that were not removed by lower-to-hw"),
655 llvm::cl::init(
false)};
659 llvm::cl::desc(
"Annotate mux pragmas for memory array access"),
660 llvm::cl::init(
false)};
663 "verification-flavor",
664 llvm::cl::desc(
"Specify a verification flavor used in LowerFIRRTLToHW"),
666 clEnumValN(firrtl::VerificationFlavor::None,
"none",
667 "Use the flavor specified by the op"),
668 clEnumValN(firrtl::VerificationFlavor::IfElseFatal,
"if-else-fatal",
669 "Use Use `if(cond) else $fatal(..)` format"),
670 clEnumValN(firrtl::VerificationFlavor::Immediate,
"immediate",
671 "Use immediate verif statements"),
672 clEnumValN(firrtl::VerificationFlavor::SVA,
"sva",
"Use SVA")),
673 llvm::cl::init(firrtl::VerificationFlavor::None)};
676 "emit-separate-always-blocks",
678 "Prevent always blocks from being merged and emit constructs into "
679 "separate always blocks whenever possible"),
680 llvm::cl::init(
false)};
683 "etc-disable-instance-extraction",
684 llvm::cl::desc(
"Disable extracting instances only that feed test code"),
685 llvm::cl::init(
false)};
688 "etc-disable-register-extraction",
689 llvm::cl::desc(
"Disable extracting registers that only feed test code"),
690 llvm::cl::init(
false)};
693 "etc-disable-module-inlining",
694 llvm::cl::desc(
"Disable inlining modules that only feed test code"),
695 llvm::cl::init(
false)};
698 "add-vivado-ram-address-conflict-synthesis-bug-workaround",
700 "Add a vivado specific SV attribute (* ram_style = "
701 "\"distributed\" *) to unpacked array registers as a workaronud "
702 "for a vivado synthesis bug that incorrectly modifies "
703 "address conflict behavivor of combinational memories"),
704 llvm::cl::init(
false)};
711 "ckg-name", llvm::cl::desc(
"Clock gate module name"),
712 llvm::cl::init(
"EICG_wrapper")};
715 "ckg-input", llvm::cl::desc(
"Clock gate input port name"),
716 llvm::cl::init(
"in")};
719 "ckg-output", llvm::cl::desc(
"Clock gate output port name"),
720 llvm::cl::init(
"out")};
723 "ckg-enable", llvm::cl::desc(
"Clock gate enable port name"),
724 llvm::cl::init(
"en")};
728 llvm::cl::desc(
"Clock gate test enable port name (optional)"),
729 llvm::cl::init(
"test_en")};
732 "export-module-hierarchy",
733 llvm::cl::desc(
"Export module and instance hierarchy as JSON"),
734 llvm::cl::init(
false)};
737 "strip-fir-debug-info",
739 "Disable source fir locator information in output Verilog"),
740 llvm::cl::init(
true)};
744 llvm::cl::desc(
"Disable source locator information in output Verilog"),
745 llvm::cl::init(
false)};
748 "fixup-eicg-wrapper",
749 llvm::cl::desc(
"Lower `EICG_wrapper` modules into clock gate intrinsics"),
750 llvm::cl::init(
false)};
753 "select-default-for-unspecified-instance-choice",
755 "Specialize instance choice to default, if no option selected"),
756 llvm::cl::init(
false)};
760 llvm::cl::desc(
"Control how symbolic values are lowered"),
761 llvm::cl::init(verif::SymbolicValueLowering::ExtModule),
762 verif::symbolicValueLoweringCLValues()};
765 "disable-wire-elimination", llvm::cl::desc(
"Disable wire elimination"),
766 llvm::cl::init(
false)};
769 "emit-all-bind-files",
770 llvm::cl::desc(
"Emit bindfiles for private modules"),
771 llvm::cl::init(
false)};
774 "inline-input-only-modules", llvm::cl::desc(
"Inline input-only modules"),
775 llvm::cl::init(
false)};
778 "domain-mode", llvm::cl::desc(
"Enable domain inference and checking"),
779 llvm::cl::init(firtool::FirtoolOptions::DomainMode::Disable),
781 clEnumValN(firtool::FirtoolOptions::DomainMode::Disable,
"disable",
782 "Disable domain checking"),
783 clEnumValN(firtool::FirtoolOptions::DomainMode::Infer,
"infer",
784 "Check domains with inference for private modules"),
785 clEnumValN(firtool::FirtoolOptions::DomainMode::Check,
"check",
786 "Check domains without inference"),
787 clEnumValN(firtool::FirtoolOptions::DomainMode::InferAll,
"infer-all",
788 "Check domains with inference for both public and private "
796 "lint-static-asserts", llvm::cl::desc(
"Lint static assertions"),
797 llvm::cl::init(
true)};
801 "lint-xmrs-in-design", llvm::cl::desc(
"Lint XMRs in the design"),
802 llvm::cl::init(
false)};
818 : outputFilename(
"-"), disableAnnotationsUnknown(false),
819 disableAnnotationsClassless(false), lowerAnnotationsNoRefTypePorts(false),
820 probesToSignals(false),
821 preserveAggregate(firrtl::PreserveAggregate::None),
822 preserveMode(firrtl::PreserveValues::None), enableDebugInfo(false),
823 buildMode(BuildModeRelease), disableLayerSink(false),
824 disableOptimization(false), vbToBV(false), noDedup(false),
825 dedupClasses(true), companionMode(firrtl::CompanionMode::Bind),
826 noViews(false), disableAggressiveMergeConnections(false),
827 lowerMemories(false), blackBoxRootPath(
""), replSeqMem(false),
828 replSeqMemFile(
""), extractTestCode(false), ignoreReadEnableMem(false),
829 disableRandom(
RandomKind::None), outputAnnotationFilename(
""),
830 enableAnnotationWarning(false), addMuxPragmas(false),
831 verificationFlavor(firrtl::VerificationFlavor::None),
832 emitSeparateAlwaysBlocks(false), etcDisableInstanceExtraction(false),
833 etcDisableRegisterExtraction(false), etcDisableModuleInlining(false),
834 addVivadoRAMAddressConflictSynthesisBugWorkaround(false),
835 ckgModuleName(
"EICG_wrapper"), ckgInputName(
"in"), ckgOutputName(
"out"),
836 ckgEnableName(
"en"), ckgTestEnableName(
"test_en"), ckgInstName(
"ckg"),
837 exportModuleHierarchy(false), stripFirDebugInfo(true),
838 stripDebugInfo(false), fixupEICGWrapper(false),
839 disableCSEinClasses(false), selectDefaultInstanceChoice(false),
840 symbolicValueLowering(
verif::SymbolicValueLowering::ExtModule),
841 disableWireElimination(false), lintStaticAsserts(true),
842 lintXmrsInDesign(true), emitAllBindFiles(false),
843 inlineInputOnlyModules(false), domainMode(
DomainMode::Disable) {
863 clOptions->disableAggressiveMergeConnections;
880 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.
std::unique_ptr< mlir::Pass > createVerifyObjectFieldsPass()
std::unique_ptr< mlir::Pass > createHWExportModuleHierarchyPass()
std::unique_ptr< mlir::Pass > createHWLegalizeModulesPass()
std::unique_ptr< mlir::Pass > createPrettifyVerilogPass()
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
std::unique_ptr< mlir::Pass > createExportSplitVerilogPass(llvm::StringRef directory="./")
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 > createLowerFIRRTLToHWPass(bool enableAnnotationWarning=false, firrtl::VerificationFlavor assertionFlavor=firrtl::VerificationFlavor::None)
This is the pass constructor.
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)