mirror of
https://github.com/YosysHQ/yosys
synced 2025-10-24 08:24:35 +00:00
Docs: adding other macro command lists
Also updates `macro_commands.py` to skip empty lines, and moves comment stripping earlier in parsing.
This commit is contained in:
parent
6c8949cacc
commit
449135a9d4
8 changed files with 111 additions and 102 deletions
27
docs/source/code_examples/macro_commands/fsm.ys
Normal file
27
docs/source/code_examples/macro_commands/fsm.ys
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
#start:It also calls opt_clean as needed:
|
||||||
|
#end:Options:
|
||||||
|
# Identify and extract FSMs:
|
||||||
|
fsm_detect
|
||||||
|
fsm_extract
|
||||||
|
|
||||||
|
# Basic optimizations:
|
||||||
|
fsm_opt
|
||||||
|
opt_clean
|
||||||
|
fsm_opt
|
||||||
|
|
||||||
|
# Expanding to nearby gate-logic (if called with -expand):
|
||||||
|
fsm_expand
|
||||||
|
opt_clean
|
||||||
|
fsm_opt
|
||||||
|
|
||||||
|
# Re-code FSM states (unless called with -norecode):
|
||||||
|
fsm_recode
|
||||||
|
|
||||||
|
# Print information about FSMs:
|
||||||
|
fsm_info
|
||||||
|
|
||||||
|
# Export FSMs in KISS2 file format (if called with -export):
|
||||||
|
fsm_export
|
||||||
|
|
||||||
|
# Map FSMs to RTL cells (unless called with -nomap):
|
||||||
|
fsm_map
|
15
docs/source/code_examples/macro_commands/memory.ys
Normal file
15
docs/source/code_examples/macro_commands/memory.ys
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
#start:passes in a useful order:
|
||||||
|
#end:This converts memories to word-wide DFFs and address decoders
|
||||||
|
opt_mem
|
||||||
|
opt_mem_priority
|
||||||
|
opt_mem_feedback
|
||||||
|
memory_bmux2rom
|
||||||
|
memory_dff
|
||||||
|
opt_clean
|
||||||
|
memory_share
|
||||||
|
opt_mem_widen
|
||||||
|
memory_memx (when called with -memx)
|
||||||
|
opt_clean
|
||||||
|
memory_collect
|
||||||
|
memory_bram -rules <bram_rules> (when called with -bram)
|
||||||
|
memory_map (skipped if called with -nomap)
|
14
docs/source/code_examples/macro_commands/proc.ys
Normal file
14
docs/source/code_examples/macro_commands/proc.ys
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
#start: passes in the most common order.
|
||||||
|
#end: This replaces the processes
|
||||||
|
proc_clean # removes empty branches and processes
|
||||||
|
proc_rmdead # removes unreachable branches
|
||||||
|
proc_prune
|
||||||
|
proc_init # special handling of “initial” blocks
|
||||||
|
proc_arst # identifies modeling of async resets
|
||||||
|
proc_rom
|
||||||
|
proc_mux # converts decision trees to multiplexer networks
|
||||||
|
proc_dlatch
|
||||||
|
proc_dff # extracts registers from processes
|
||||||
|
proc_memwr
|
||||||
|
proc_clean # this should remove all the processes, provided all went fine
|
||||||
|
opt_expr -keepdc
|
|
@ -5,79 +5,22 @@ The :cmd:ref:`fsm` command identifies, extracts, optimizes (re-encodes), and
|
||||||
re-synthesizes finite state machines. It again is a macro that calls a series of
|
re-synthesizes finite state machines. It again is a macro that calls a series of
|
||||||
other commands:
|
other commands:
|
||||||
|
|
||||||
#. :cmd:ref:`fsm_detect` identifies FSM state registers and marks them
|
.. literalinclude:: /code_examples/macro_commands/fsm.ys
|
||||||
with the ``(* fsm_encoding = "auto" *)`` attribute, if they do not have the
|
:language: yoscrypt
|
||||||
``fsm_encoding`` set already. Mark registers with ``(* fsm_encoding = "none"
|
:start-after: #end:
|
||||||
*)`` to disable FSM optimization for a register.
|
:caption: Passes called by :cmd:ref:`fsm`
|
||||||
#. :cmd:ref:`fsm_extract` replaces the entire FSM (logic and state registers)
|
|
||||||
with a ``$fsm`` cell.
|
|
||||||
#. :cmd:ref:`fsm_opt` optimizes the FSM. Called multiple times.
|
|
||||||
#. :cmd:ref:`fsm_expand` optionally merges additional auxilliary gates into the
|
|
||||||
``$fsm`` cell.
|
|
||||||
#. :cmd:ref:`fsm_recode` also optimizes the FSM.
|
|
||||||
#. :cmd:ref:`fsm_info` logs internal FSM information.
|
|
||||||
#. :cmd:ref:`fsm_export` optionally exports each FSM to KISS2 files.
|
|
||||||
#. :cmd:ref:`fsm_map` converts the (optimized) ``$fsm`` cell back to logic and
|
|
||||||
registers.
|
|
||||||
|
|
||||||
See also :doc:`/cmd/fsm`.
|
See also :doc:`/cmd/fsm`.
|
||||||
|
|
||||||
The fsm pass performs finite-state-machine (FSM) extraction and recoding. The
|
|
||||||
fsm pass simply executes the following other passes:
|
|
||||||
|
|
||||||
- Identify and extract FSMs:
|
|
||||||
|
|
||||||
- fsm_detect
|
|
||||||
- fsm_extract
|
|
||||||
|
|
||||||
- Basic optimizations:
|
|
||||||
|
|
||||||
- fsm_opt
|
|
||||||
- opt_clean
|
|
||||||
- fsm_opt
|
|
||||||
|
|
||||||
- Expanding to nearby gate-logic (if called with -expand):
|
|
||||||
|
|
||||||
- fsm_expand
|
|
||||||
- opt_clean
|
|
||||||
- fsm_opt
|
|
||||||
|
|
||||||
- Re-code FSM states (unless called with -norecode):
|
|
||||||
|
|
||||||
- fsm_recode
|
|
||||||
|
|
||||||
- Print information about FSMs:
|
|
||||||
|
|
||||||
- fsm_info
|
|
||||||
|
|
||||||
- Export FSMs in KISS2 file format (if called with -export):
|
|
||||||
|
|
||||||
- fsm_export
|
|
||||||
|
|
||||||
- Map FSMs to RTL cells (unless called with -nomap):
|
|
||||||
|
|
||||||
- fsm_map
|
|
||||||
|
|
||||||
The fsm_detect pass identifies FSM state registers and marks them using the
|
|
||||||
``\fsm_encoding = "auto"`` attribute. The fsm_extract extracts all FSMs marked
|
|
||||||
using the ``\fsm_encoding`` attribute (unless ``\fsm_encoding`` is set to
|
|
||||||
"none") and replaces the corresponding RTL cells with a ``$fsm`` cell. All other
|
|
||||||
``fsm_`` passes operate on these ``$fsm`` cells. The fsm_map call finally
|
|
||||||
replaces the ``$fsm`` cells with RTL cells.
|
|
||||||
|
|
||||||
Note that these optimizations operate on an RTL netlist. I.e. the :cmd:ref:`fsm`
|
|
||||||
pass should be executed after the proc pass has transformed all
|
|
||||||
``RTLIL::Process`` objects to RTL cells.
|
|
||||||
|
|
||||||
The algorithms used for FSM detection and extraction are influenced by a more
|
The algorithms used for FSM detection and extraction are influenced by a more
|
||||||
general reported technique :cite:p:`fsmextract`.
|
general reported technique :cite:p:`fsmextract`.
|
||||||
|
|
||||||
FSM detection
|
FSM detection
|
||||||
~~~~~~~~~~~~~
|
~~~~~~~~~~~~~
|
||||||
|
|
||||||
The fsm_detect pass identifies FSM state registers. It sets the ``\fsm_encoding
|
The :cmd:ref:`fsm_detect` pass identifies FSM state registers. It sets the
|
||||||
= "auto"`` attribute on any (multi-bit) wire that matches the following
|
``\fsm_encoding = "auto"`` attribute on any (multi-bit) wire that matches the
|
||||||
description:
|
following description:
|
||||||
|
|
||||||
- Does not already have the ``\fsm_encoding`` attribute.
|
- Does not already have the ``\fsm_encoding`` attribute.
|
||||||
- Is not an output of the containing module.
|
- Is not an output of the containing module.
|
||||||
|
@ -101,7 +44,7 @@ results.
|
||||||
FSM extraction
|
FSM extraction
|
||||||
~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~
|
||||||
|
|
||||||
The fsm_extract pass operates on all state signals marked with the
|
The :cmd:ref:`fsm_extract` pass operates on all state signals marked with the
|
||||||
(``\fsm_encoding != "none"``) attribute. For each state signal the following
|
(``\fsm_encoding != "none"``) attribute. For each state signal the following
|
||||||
information is determined:
|
information is determined:
|
||||||
|
|
||||||
|
@ -142,8 +85,8 @@ given set of result signals using a set of signal-value assignments. It can also
|
||||||
be passed a list of stop-signals that abort the ConstEval algorithm if the value
|
be passed a list of stop-signals that abort the ConstEval algorithm if the value
|
||||||
of a stop-signal is needed in order to calculate the result signals.
|
of a stop-signal is needed in order to calculate the result signals.
|
||||||
|
|
||||||
The fsm_extract pass uses the ConstEval class in the following way to create a
|
The :cmd:ref:`fsm_extract` pass uses the ConstEval class in the following way to
|
||||||
transition table. For each state:
|
create a transition table. For each state:
|
||||||
|
|
||||||
1. Create a ConstEval object for the module containing the FSM
|
1. Create a ConstEval object for the module containing the FSM
|
||||||
2. Add all control inputs to the list of stop signals
|
2. Add all control inputs to the list of stop signals
|
||||||
|
@ -163,8 +106,9 @@ drivers for the control outputs are disconnected.
|
||||||
FSM optimization
|
FSM optimization
|
||||||
~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
The fsm_opt pass performs basic optimizations on ``$fsm`` cells (not including
|
The :cmd:ref:`fsm_opt` pass performs basic optimizations on ``$fsm`` cells (not
|
||||||
state recoding). The following optimizations are performed (in this order):
|
including state recoding). The following optimizations are performed (in this
|
||||||
|
order):
|
||||||
|
|
||||||
- Unused control outputs are removed from the ``$fsm`` cell. The attribute
|
- Unused control outputs are removed from the ``$fsm`` cell. The attribute
|
||||||
``\unused_bits`` (that is usually set by the :cmd:ref:`opt_clean` pass) is
|
``\unused_bits`` (that is usually set by the :cmd:ref:`opt_clean` pass) is
|
||||||
|
@ -188,10 +132,11 @@ state recoding). The following optimizations are performed (in this order):
|
||||||
FSM recoding
|
FSM recoding
|
||||||
~~~~~~~~~~~~
|
~~~~~~~~~~~~
|
||||||
|
|
||||||
The fsm_recode pass assigns new bit pattern to the states. Usually this also
|
The :cmd:ref:`fsm_recode` pass assigns new bit pattern to the states. Usually
|
||||||
implies a change in the width of the state signal. At the moment of this writing
|
this also implies a change in the width of the state signal. At the moment of
|
||||||
only one-hot encoding with all-zero for the reset state is supported.
|
this writing only one-hot encoding with all-zero for the reset state is
|
||||||
|
supported.
|
||||||
|
|
||||||
The fsm_recode pass can also write a text file with the changes performed by it
|
The :cmd:ref:`fsm_recode` pass can also write a text file with the changes
|
||||||
that can be used when verifying designs synthesized by Yosys using Synopsys
|
performed by it that can be used when verifying designs synthesized by Yosys
|
||||||
Formality.
|
using Synopsys Formality.
|
||||||
|
|
|
@ -7,19 +7,22 @@ The :cmd:ref:`memory` command
|
||||||
In the RTL netlist, memory reads and writes are individual cells. This makes
|
In the RTL netlist, memory reads and writes are individual cells. This makes
|
||||||
consolidating the number of ports for a memory easier. The :cmd:ref:`memory`
|
consolidating the number of ports for a memory easier. The :cmd:ref:`memory`
|
||||||
pass transforms memories to an implementation. Per default that is logic for
|
pass transforms memories to an implementation. Per default that is logic for
|
||||||
address decoders and registers. It also is a macro command that the other common
|
address decoders and registers. It also is a macro command that calls the other
|
||||||
``memory_*`` commands in a sensible order:
|
common ``memory_*`` passes in a sensible order:
|
||||||
|
|
||||||
.. todo:: fill out missing :cmd:ref:`memory` subcommands descriptions
|
.. literalinclude:: /code_examples/macro_commands/memory.ys
|
||||||
|
:language: yoscrypt
|
||||||
|
:start-after: #end:
|
||||||
|
:caption: Passes called by :cmd:ref:`memory`
|
||||||
|
|
||||||
#. :cmd:ref:`memory_bmux2rom`
|
.. todo:: Make ``memory_*`` notes less quick
|
||||||
#. :cmd:ref:`memory_dff` merges registers into the memory read- and write cells.
|
|
||||||
#. :cmd:ref:`memory_share`
|
Some quick notes:
|
||||||
#. :cmd:ref:`memory_memx`
|
|
||||||
#. :cmd:ref:`memory_collect` collects all read and write cells for a memory and
|
- :cmd:ref:`memory_dff` merges registers into the memory read- and write cells.
|
||||||
|
- :cmd:ref:`memory_collect` collects all read and write cells for a memory and
|
||||||
transforms them into one multi-port memory cell.
|
transforms them into one multi-port memory cell.
|
||||||
#. :cmd:ref:`memory_bram`
|
- :cmd:ref:`memory_map` takes the multi-port memory cell and transforms it to
|
||||||
#. :cmd:ref:`memory_map` takes the multi-port memory cell and transforms it to
|
|
||||||
address decoder logic and registers.
|
address decoder logic and registers.
|
||||||
|
|
||||||
For more information about :cmd:ref:`memory`, such as disabling certain sub
|
For more information about :cmd:ref:`memory`, such as disabling certain sub
|
||||||
|
|
|
@ -1,26 +1,21 @@
|
||||||
Converting process blocks
|
Converting process blocks
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
.. role:: yoscrypt(code)
|
||||||
|
:language: yoscrypt
|
||||||
|
|
||||||
The Verilog frontend converts ``always``-blocks to RTL netlists for the
|
The Verilog frontend converts ``always``-blocks to RTL netlists for the
|
||||||
expressions and "processess" for the control- and memory elements. The
|
expressions and "processess" for the control- and memory elements. The
|
||||||
:cmd:ref:`proc` command then transforms these "processess" to netlists of RTL
|
:cmd:ref:`proc` command then transforms these "processess" to netlists of RTL
|
||||||
multiplexer and register cells. It also is a macro command that calls the other
|
multiplexer and register cells. It also is a macro command that calls the other
|
||||||
``proc_*`` commands in a sensible order:
|
``proc_*`` commands in a sensible order:
|
||||||
|
|
||||||
#. :cmd:ref:`proc_clean` removes empty branches and processes.
|
.. literalinclude:: /code_examples/macro_commands/proc.ys
|
||||||
#. :cmd:ref:`proc_rmdead` removes unreachable branches.
|
:language: yoscrypt
|
||||||
#. :cmd:ref:`proc_prune`
|
:start-after: #end:
|
||||||
#. :cmd:ref:`proc_init` special handling of "initial" blocks.
|
:caption: Passes called by :cmd:ref:`proc`
|
||||||
#. :cmd:ref:`proc_arst` identifies modeling of async resets.
|
|
||||||
#. :cmd:ref:`proc_rom`
|
|
||||||
#. :cmd:ref:`proc_mux` converts decision trees to multiplexer networks.
|
|
||||||
#. :cmd:ref:`proc_dlatch`
|
|
||||||
#. :cmd:ref:`proc_dff` extracts registers from processes.
|
|
||||||
#. :cmd:ref:`proc_memwr`
|
|
||||||
#. :cmd:ref:`proc_clean` this should remove all the processes, provided all went
|
|
||||||
fine.
|
|
||||||
|
|
||||||
After all the ``proc_*`` commands, :yoscrypt:`opt_expr` is called. This can be
|
After all the ``proc_*`` commands, :cmd:ref:`opt_expr` is called. This can be
|
||||||
disabled by calling :yoscrypt:`proc -noopt`. For more information about
|
disabled by calling :yoscrypt:`proc -noopt`. For more information about
|
||||||
:cmd:ref:`proc`, such as disabling certain sub commands, see :doc:`/cmd/proc`.
|
:cmd:ref:`proc`, such as disabling certain sub commands, see :doc:`/cmd/proc`.
|
||||||
|
|
||||||
|
|
|
@ -43,4 +43,5 @@ The following commands are executed by the :cmd:ref:`prep` command:
|
||||||
:end-before: .. raw:: latex
|
:end-before: .. raw:: latex
|
||||||
:dedent:
|
:dedent:
|
||||||
|
|
||||||
The following sections will get more into what each of these commands do.
|
:doc:`/getting_started/example_synth` covers most of these commands and what
|
||||||
|
they do.
|
||||||
|
|
|
@ -35,7 +35,12 @@ for macro in MACRO_SOURCE.glob("*.ys"):
|
||||||
# immediately after.
|
# immediately after.
|
||||||
start = f.readline()
|
start = f.readline()
|
||||||
end = f.readline()
|
end = f.readline()
|
||||||
expected_content = f.readlines()
|
file_content = f.readlines()
|
||||||
|
expected_content = []
|
||||||
|
for line in file_content:
|
||||||
|
line = line.split("#")[0].strip()
|
||||||
|
if line:
|
||||||
|
expected_content.append(line)
|
||||||
# parse {command.ys}
|
# parse {command.ys}
|
||||||
if "#start:" not in start or "#end:" not in end:
|
if "#start:" not in start or "#end:" not in end:
|
||||||
logging.error(f"Missing start and/or end string in {relative_path}, see {THIS_FILE}")
|
logging.error(f"Missing start and/or end string in {relative_path}, see {THIS_FILE}")
|
||||||
|
@ -49,7 +54,11 @@ for macro in MACRO_SOURCE.glob("*.ys"):
|
||||||
logging.error(f"Couldn't find {start!r} and/or {end!r} in `yosys -h {command}` output")
|
logging.error(f"Couldn't find {start!r} and/or {end!r} in `yosys -h {command}` output")
|
||||||
raise_error = True
|
raise_error = True
|
||||||
continue
|
continue
|
||||||
actual_content = match.group(1).strip().splitlines()
|
match_content = match.group(1).strip().splitlines()
|
||||||
|
actual_content = []
|
||||||
|
for line in match_content:
|
||||||
|
if line:
|
||||||
|
actual_content.append(line)
|
||||||
# iterate over and compare expected v actual
|
# iterate over and compare expected v actual
|
||||||
for (expected, actual) in zip(expected_content, actual_content):
|
for (expected, actual) in zip(expected_content, actual_content):
|
||||||
expected = expected.strip()
|
expected = expected.strip()
|
||||||
|
@ -59,7 +68,7 @@ for macro in MACRO_SOURCE.glob("*.ys"):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# rip apart formatting to match line parts
|
# rip apart formatting to match line parts
|
||||||
pattern = r"(?P<cmd>\S+)(?P<pass> \[.*\])?(?P<opt>.*?)(?P<cond> \(.*\))?(?P<comment>\s+#.*)?"
|
pattern = r"(?P<cmd>\S+)(?P<pass> \[.*\])?(?P<opt>.*?)(?P<cond>\s+\(.*\))?"
|
||||||
try:
|
try:
|
||||||
expected_dict = re.fullmatch(pattern, expected).groupdict()
|
expected_dict = re.fullmatch(pattern, expected).groupdict()
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue