mirror of
https://github.com/YosysHQ/yosys
synced 2025-04-06 17:44:09 +00:00
Merge pull request #4884 from YosysHQ/docs-preview-functional_tutorial
Docs: More on FunctionalIR
This commit is contained in:
commit
c74df780b7
14
Makefile
14
Makefile
|
@ -1010,6 +1010,18 @@ docs/source/cell/word_add.rst: $(TARGETS) $(EXTRA_TARGETS)
|
||||||
docs/source/generated/cells.json: docs/source/generated $(TARGETS) $(EXTRA_TARGETS)
|
docs/source/generated/cells.json: docs/source/generated $(TARGETS) $(EXTRA_TARGETS)
|
||||||
$(Q) ./$(PROGRAM_PREFIX)yosys -p 'help -dump-cells-json $@'
|
$(Q) ./$(PROGRAM_PREFIX)yosys -p 'help -dump-cells-json $@'
|
||||||
|
|
||||||
|
docs/source/generated/%.cc: backends/%.cc
|
||||||
|
$(Q) mkdir -p $(@D)
|
||||||
|
$(Q) cp $< $@
|
||||||
|
|
||||||
|
# diff returns exit code 1 if the files are different, but it's not an error
|
||||||
|
docs/source/generated/functional/rosette.diff: backends/functional/smtlib.cc backends/functional/smtlib_rosette.cc
|
||||||
|
$(Q) mkdir -p $(@D)
|
||||||
|
$(Q) diff -U 20 $^ > $@ || exit 0
|
||||||
|
|
||||||
|
PHONY: docs/gen/functional_ir
|
||||||
|
docs/gen/functional_ir: docs/source/generated/functional/smtlib.cc docs/source/generated/functional/rosette.diff
|
||||||
|
|
||||||
PHONY: docs/gen docs/usage docs/reqs
|
PHONY: docs/gen docs/usage docs/reqs
|
||||||
docs/gen: $(TARGETS)
|
docs/gen: $(TARGETS)
|
||||||
$(Q) $(MAKE) -C docs gen
|
$(Q) $(MAKE) -C docs gen
|
||||||
|
@ -1045,7 +1057,7 @@ docs/reqs:
|
||||||
$(Q) $(MAKE) -C docs reqs
|
$(Q) $(MAKE) -C docs reqs
|
||||||
|
|
||||||
.PHONY: docs/prep
|
.PHONY: docs/prep
|
||||||
docs/prep: docs/source/cmd/abc.rst docs/source/generated/cells.json docs/gen docs/usage
|
docs/prep: docs/source/cmd/abc.rst docs/source/generated/cells.json docs/gen docs/usage docs/gen/functional_ir
|
||||||
|
|
||||||
DOC_TARGET ?= html
|
DOC_TARGET ?= html
|
||||||
docs: docs/prep
|
docs: docs/prep
|
||||||
|
|
|
@ -210,14 +210,8 @@ struct SmtrModule {
|
||||||
state_struct.insert(state->name, state->sort);
|
state_struct.insert(state->name, state->sort);
|
||||||
}
|
}
|
||||||
|
|
||||||
void write(std::ostream &out)
|
void write_eval(SExprWriter &w)
|
||||||
{
|
{
|
||||||
SExprWriter w(out);
|
|
||||||
|
|
||||||
input_struct.write_definition(w);
|
|
||||||
output_struct.write_definition(w);
|
|
||||||
state_struct.write_definition(w);
|
|
||||||
|
|
||||||
w.push();
|
w.push();
|
||||||
w.open(list("define", list(name, "inputs", "state")));
|
w.open(list("define", list(name, "inputs", "state")));
|
||||||
auto inlined = [&](Functional::Node n) {
|
auto inlined = [&](Functional::Node n) {
|
||||||
|
@ -240,7 +234,10 @@ struct SmtrModule {
|
||||||
output_struct.write_value(w, [&](IdString name) { return node_to_sexpr(ir.output(name).value()); });
|
output_struct.write_value(w, [&](IdString name) { return node_to_sexpr(ir.output(name).value()); });
|
||||||
state_struct.write_value(w, [&](IdString name) { return node_to_sexpr(ir.state(name).next_value()); });
|
state_struct.write_value(w, [&](IdString name) { return node_to_sexpr(ir.state(name).next_value()); });
|
||||||
w.pop();
|
w.pop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void write_initial(SExprWriter &w)
|
||||||
|
{
|
||||||
w.push();
|
w.push();
|
||||||
auto initial = name + "_initial";
|
auto initial = name + "_initial";
|
||||||
w.open(list("define", initial));
|
w.open(list("define", initial));
|
||||||
|
@ -259,6 +256,18 @@ struct SmtrModule {
|
||||||
}
|
}
|
||||||
w.pop();
|
w.pop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void write(std::ostream &out)
|
||||||
|
{
|
||||||
|
SExprWriter w(out);
|
||||||
|
|
||||||
|
input_struct.write_definition(w);
|
||||||
|
output_struct.write_definition(w);
|
||||||
|
state_struct.write_definition(w);
|
||||||
|
|
||||||
|
write_eval(w);
|
||||||
|
write_initial(w);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct FunctionalSmtrBackend : public Backend {
|
struct FunctionalSmtrBackend : public Backend {
|
||||||
|
@ -267,7 +276,7 @@ struct FunctionalSmtrBackend : public Backend {
|
||||||
void help() override {
|
void help() override {
|
||||||
// |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
|
// |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
|
||||||
log("\n");
|
log("\n");
|
||||||
log(" write_functional_rosette [options] [selection] [filename]\n");
|
log(" write_functional_rosette [options] [filename]\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
log("Functional Rosette Backend.\n");
|
log("Functional Rosette Backend.\n");
|
||||||
log("\n");
|
log("\n");
|
||||||
|
|
44
docs/source/code_examples/functional/dummy.cc
Normal file
44
docs/source/code_examples/functional/dummy.cc
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
#include "kernel/functional.h"
|
||||||
|
#include "kernel/yosys.h"
|
||||||
|
|
||||||
|
USING_YOSYS_NAMESPACE
|
||||||
|
PRIVATE_NAMESPACE_BEGIN
|
||||||
|
|
||||||
|
struct FunctionalDummyBackend : public Backend {
|
||||||
|
FunctionalDummyBackend() : Backend("functional_dummy", "dump generated Functional IR") {}
|
||||||
|
void execute(std::ostream *&f, std::string filename, std::vector<std::string> args, RTLIL::Design *design) override
|
||||||
|
{
|
||||||
|
// backend pass boiler plate
|
||||||
|
log_header(design, "Executing dummy functional backend.\n");
|
||||||
|
|
||||||
|
size_t argidx = 1;
|
||||||
|
extra_args(f, filename, args, argidx, design);
|
||||||
|
|
||||||
|
for (auto module : design->selected_modules())
|
||||||
|
{
|
||||||
|
log("Processing module `%s`.\n", module->name.c_str());
|
||||||
|
|
||||||
|
// convert module to FunctionalIR
|
||||||
|
auto ir = Functional::IR::from_module(module);
|
||||||
|
*f << "module " << module->name.c_str() << "\n";
|
||||||
|
|
||||||
|
// write node functions
|
||||||
|
for (auto node : ir)
|
||||||
|
*f << " assign " << id2cstr(node.name())
|
||||||
|
<< " = " << node.to_string() << "\n";
|
||||||
|
*f << "\n";
|
||||||
|
|
||||||
|
// write outputs and next state
|
||||||
|
for (auto output : ir.outputs())
|
||||||
|
*f << " " << id2cstr(output->kind)
|
||||||
|
<< " " << id2cstr(output->name)
|
||||||
|
<< " = " << id2cstr(output->value().name()) << "\n";
|
||||||
|
for (auto state : ir.states())
|
||||||
|
*f << " " << id2cstr(state->kind)
|
||||||
|
<< " " << id2cstr(state->name)
|
||||||
|
<< " = " << id2cstr(state->next_value().name()) << "\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} FunctionalDummyBackend;
|
||||||
|
|
||||||
|
PRIVATE_NAMESPACE_END
|
|
@ -1,94 +1,524 @@
|
||||||
Writing a new backend using FunctionalIR
|
Writing a new backend using FunctionalIR
|
||||||
===========================================
|
========================================
|
||||||
|
|
||||||
To simplify the writing of backends for functional languages or similar targets, Yosys provides an alternative intermediate representation called FunctionalIR which maps more directly on those targets.
|
What is FunctionalIR
|
||||||
|
--------------------
|
||||||
|
|
||||||
FunctionalIR represents the design as a function ``(inputs, current_state) -> (outputs, next_state)``.
|
To simplify the writing of backends for functional languages or similar targets,
|
||||||
This function is broken down into a series of assignments to variables.
|
Yosys provides an alternative intermediate representation called FunctionalIR
|
||||||
Each assignment is a simple operation, such as an addition.
|
which maps more directly on those targets.
|
||||||
Complex operations are broken up into multiple steps.
|
|
||||||
For example, an RTLIL addition will be translated into a sign/zero extension of the inputs, followed by an addition.
|
|
||||||
|
|
||||||
Like SSA form, each variable is assigned to exactly once.
|
FunctionalIR represents the design as a function ``(inputs, current_state) ->
|
||||||
We can thus treat variables and assignments as equivalent and, since this is a graph-like representation, those variables are also called "nodes".
|
(outputs, next_state)``. This function is broken down into a series of
|
||||||
Unlike RTLIL's cells and wires representation, this representation is strictly ordered (topologically sorted) with definitions preceding their use.
|
assignments to variables. Each assignment is a simple operation, such as an
|
||||||
|
addition. Complex operations are broken up into multiple steps. For example, an
|
||||||
|
RTLIL addition will be translated into a sign/zero extension of the inputs,
|
||||||
|
followed by an addition.
|
||||||
|
|
||||||
Every node has a "sort" (the FunctionalIR term for what might otherwise be called a "type"). The sorts available are
|
Like SSA form, each variable is assigned to exactly once. We can thus treat
|
||||||
|
variables and assignments as equivalent and, since this is a graph-like
|
||||||
|
representation, those variables are also called "nodes". Unlike RTLIL's cells
|
||||||
|
and wires representation, this representation is strictly ordered (topologically
|
||||||
|
sorted) with definitions preceding their use.
|
||||||
|
|
||||||
|
Every node has a "sort" (the FunctionalIR term for what might otherwise be
|
||||||
|
called a "type"). The sorts available are
|
||||||
|
|
||||||
- ``bit[n]`` for an ``n``-bit bitvector, and
|
- ``bit[n]`` for an ``n``-bit bitvector, and
|
||||||
- ``memory[n,m]`` for an immutable array of ``2**n`` values of sort ``bit[m]``.
|
- ``memory[n,m]`` for an immutable array of ``2**n`` values of sort ``bit[m]``.
|
||||||
|
|
||||||
In terms of actual code, Yosys provides a class ``Functional::IR`` that represents a design in FunctionalIR.
|
In terms of actual code, Yosys provides a class ``Functional::IR`` that
|
||||||
``Functional::IR::from_module`` generates an instance from an RTLIL module.
|
represents a design in FunctionalIR. ``Functional::IR::from_module`` generates
|
||||||
The entire design is stored as a whole in an internal data structure.
|
an instance from an RTLIL module. The entire design is stored as a whole in an
|
||||||
To access the design, the ``Functional::Node`` class provides a reference to a particular node in the design.
|
internal data structure. To access the design, the ``Functional::Node`` class
|
||||||
The ``Functional::IR`` class supports the syntax ``for(auto node : ir)`` to iterate over every node.
|
provides a reference to a particular node in the design. The ``Functional::IR``
|
||||||
|
class supports the syntax ``for(auto node : ir)`` to iterate over every node.
|
||||||
|
|
||||||
``Functional::IR`` also keeps track of inputs, outputs and states.
|
``Functional::IR`` also keeps track of inputs, outputs and states. By a "state"
|
||||||
By a "state" we mean a pair of a "current state" input and a "next state" output.
|
we mean a pair of a "current state" input and a "next state" output. One such
|
||||||
One such pair is created for every register and for every memory.
|
pair is created for every register and for every memory. Every input, output and
|
||||||
Every input, output and state has a name (equal to their name in RTLIL), a sort and a kind.
|
state has a name (equal to their name in RTLIL), a sort and a kind. The kind
|
||||||
The kind field usually remains as the default value ``$input``, ``$output`` or ``$state``, however some RTLIL cells such as ``$assert`` or ``$anyseq`` generate auxiliary inputs/outputs/states that are given a different kind to distinguish them from ordinary RTLIL inputs/outputs/states.
|
field usually remains as the default value ``$input``, ``$output`` or
|
||||||
|
``$state``, however some RTLIL cells such as ``$assert`` or ``$anyseq`` generate
|
||||||
|
auxiliary inputs/outputs/states that are given a different kind to distinguish
|
||||||
|
them from ordinary RTLIL inputs/outputs/states.
|
||||||
|
|
||||||
- To access an individual input/output/state, use ``ir.input(name, kind)``, ``ir.output(name, kind)`` or ``ir.state(name, kind)``. ``kind`` defaults to the default kind.
|
- To access an individual input/output/state, use ``ir.input(name, kind)``,
|
||||||
- To iterate over all inputs/outputs/states of a certain kind, methods ``ir.inputs``, ``ir.outputs``, ``ir.states`` are provided. Their argument defaults to the default kinds mentioned.
|
``ir.output(name, kind)`` or ``ir.state(name, kind)``. ``kind`` defaults to
|
||||||
- To iterate over inputs/outputs/states of any kind, use ``ir.all_inputs``, ``ir.all_outputs`` and ``ir.all_states``.
|
the default kind.
|
||||||
- Outputs have a node that indicate the value of the output, this can be retrieved via ``output.value()``.
|
- To iterate over all inputs/outputs/states of a certain kind, methods
|
||||||
- States have a node that indicate the next value of the state, this can be retrieved via ``state.next_value()``.
|
``ir.inputs``, ``ir.outputs``, ``ir.states`` are provided. Their argument
|
||||||
They also have an initial value that is accessed as either ``state.initial_value_signal()`` or ``state.initial_value_memory()``, depending on their sort.
|
defaults to the default kinds mentioned.
|
||||||
|
- To iterate over inputs/outputs/states of any kind, use ``ir.all_inputs``,
|
||||||
|
``ir.all_outputs`` and ``ir.all_states``.
|
||||||
|
- Outputs have a node that indicate the value of the output, this can be
|
||||||
|
retrieved via ``output.value()``.
|
||||||
|
- States have a node that indicate the next value of the state, this can be
|
||||||
|
retrieved via ``state.next_value()``. They also have an initial value that is
|
||||||
|
accessed as either ``state.initial_value_signal()`` or
|
||||||
|
``state.initial_value_memory()``, depending on their sort.
|
||||||
|
|
||||||
Each node has a "function", which defines its operation (for a complete list of functions and a specification of their operation, see ``functional.h``).
|
Each node has a "function", which defines its operation (for a complete list of
|
||||||
Functions are represented as an enum ``Functional::Fn`` and the function field can be accessed as ``node.fn()``.
|
functions and a specification of their operation, see ``functional.h``).
|
||||||
Since the most common operation is a switch over the function that also accesses the arguments, the ``Node`` class provides a method ``visit`` that implements the visitor pattern.
|
Functions are represented as an enum ``Functional::Fn`` and the function field
|
||||||
For example, for an addition node ``node`` with arguments ``n1`` and ``n2``, ``node.visit(visitor)`` would call ``visitor.add(node, n1, n2)``.
|
can be accessed as ``node.fn()``. Since the most common operation is a switch
|
||||||
Thus typically one would implement a class with a method for every function.
|
over the function that also accesses the arguments, the ``Node`` class provides
|
||||||
Visitors should inherit from either ``Functional::AbstractVisitor<ReturnType>`` or ``Functional::DefaultVisitor<ReturnType>``.
|
a method ``visit`` that implements the visitor pattern. For example, for an
|
||||||
The former will produce a compiler error if a case is unhandled, the latter will call ``default_handler(node)`` instead.
|
addition node ``node`` with arguments ``n1`` and ``n2``, ``node.visit(visitor)``
|
||||||
Visitor methods should be marked as ``override`` to provide compiler errors if the arguments are wrong.
|
would call ``visitor.add(node, n1, n2)``. Thus typically one would implement a
|
||||||
|
class with a method for every function. Visitors should inherit from either
|
||||||
|
``Functional::AbstractVisitor<ReturnType>`` or
|
||||||
|
``Functional::DefaultVisitor<ReturnType>``. The former will produce a compiler
|
||||||
|
error if a case is unhandled, the latter will call ``default_handler(node)``
|
||||||
|
instead. Visitor methods should be marked as ``override`` to provide compiler
|
||||||
|
errors if the arguments are wrong.
|
||||||
|
|
||||||
Utility classes
|
Utility classes
|
||||||
-----------------
|
~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
``functional.h`` also provides utility classes that are independent of the main FunctionalIR representation but are likely to be useful for backends.
|
``functional.h`` also provides utility classes that are independent of the main
|
||||||
|
FunctionalIR representation but are likely to be useful for backends.
|
||||||
|
|
||||||
``Functional::Writer`` provides a simple formatting class that wraps a ``std::ostream`` and provides the following methods:
|
``Functional::Writer`` provides a simple formatting class that wraps a
|
||||||
|
``std::ostream`` and provides the following methods:
|
||||||
|
|
||||||
- ``writer << value`` wraps ``os << value``.
|
- ``writer << value`` wraps ``os << value``.
|
||||||
- ``writer.print(fmt, value0, value1, value2, ...)`` replaces ``{0}``, ``{1}``, ``{2}``, etc in the string ``fmt`` with ``value0``, ``value1``, ``value2``, resp.
|
- ``writer.print(fmt, value0, value1, value2, ...)`` replaces ``{0}``, ``{1}``,
|
||||||
Each value is formatted using ``os << value``.
|
``{2}``, etc in the string ``fmt`` with ``value0``, ``value1``, ``value2``,
|
||||||
It is also possible to write ``{}`` to refer to one past the last index, i.e. ``{1} {} {} {7} {}`` is equivalent to ``{1} {2} {3} {7} {8}``.
|
resp. Each value is formatted using ``os << value``. It is also possible to
|
||||||
- ``writer.print_with(fn, fmt, value0, value1, value2, ...)`` functions much the same as ``print`` but it uses ``os << fn(value)`` to print each value and falls back to ``os << value`` if ``fn(value)`` is not legal.
|
write ``{}`` to refer to one past the last index, i.e. ``{1} {} {} {7} {}`` is
|
||||||
|
equivalent to ``{1} {2} {3} {7} {8}``.
|
||||||
|
- ``writer.print_with(fn, fmt, value0, value1, value2, ...)`` functions much the
|
||||||
|
same as ``print`` but it uses ``os << fn(value)`` to print each value and
|
||||||
|
falls back to ``os << value`` if ``fn(value)`` is not legal.
|
||||||
|
|
||||||
``Functional::Scope`` keeps track of variable names in a target language.
|
``Functional::Scope`` keeps track of variable names in a target language. It is
|
||||||
It is used to translate between different sets of legal characters and to avoid accidentally re-defining identifiers.
|
used to translate between different sets of legal characters and to avoid
|
||||||
Users should derive a class from ``Scope`` and supply the following:
|
accidentally re-defining identifiers. Users should derive a class from ``Scope``
|
||||||
|
and supply the following:
|
||||||
|
|
||||||
- ``Scope<Id>`` takes a template argument that specifies a type that's used to uniquely distinguish variables.
|
- ``Scope<Id>`` takes a template argument that specifies a type that's used to
|
||||||
Typically this would be ``int`` (if variables are used for ``Functional::IR`` nodes) or ``IdString``.
|
uniquely distinguish variables. Typically this would be ``int`` (if variables
|
||||||
- The derived class should provide a constructor that calls ``reserve`` for every reserved word in the target language.
|
are used for ``Functional::IR`` nodes) or ``IdString``.
|
||||||
- A method ``bool is_legal_character(char c, int index)`` has to be provided that returns ``true`` iff ``c`` is legal in an identifier at position ``index``.
|
- The derived class should provide a constructor that calls ``reserve`` for
|
||||||
|
every reserved word in the target language.
|
||||||
|
- A method ``bool is_character_legal(char c, int index)`` has to be provided
|
||||||
|
that returns ``true`` iff ``c`` is legal in an identifier at position
|
||||||
|
``index``.
|
||||||
|
|
||||||
Given an instance ``scope`` of the derived class, the following methods are then available:
|
Given an instance ``scope`` of the derived class, the following methods are then
|
||||||
|
available:
|
||||||
|
|
||||||
- ``scope.reserve(std::string name)`` marks the given name as being in-use
|
- ``scope.reserve(std::string name)`` marks the given name as being in-use
|
||||||
- ``scope.unique_name(IdString suggestion)`` generates a previously unused name and attempts to make it similar to ``suggestion``.
|
- ``scope.unique_name(IdString suggestion)`` generates a previously unused name
|
||||||
- ``scope(Id id, IdString suggestion)`` functions similar to ``unique_name``, except that multiple calls with the same ``id`` are guaranteed to retrieve the same name (independent of ``suggestion``).
|
and attempts to make it similar to ``suggestion``.
|
||||||
|
- ``scope(Id id, IdString suggestion)`` functions similar to ``unique_name``,
|
||||||
|
except that multiple calls with the same ``id`` are guaranteed to retrieve the
|
||||||
|
same name (independent of ``suggestion``).
|
||||||
|
|
||||||
``sexpr.h`` provides classes that represent and pretty-print s-expressions.
|
``sexpr.h`` provides classes that represent and pretty-print s-expressions.
|
||||||
S-expressions can be constructed with ``SExpr::list``, for example ``SExpr expr = SExpr::list("add", "x", SExpr::list("mul", "y", "z"))`` represents ``(add x (mul y z))``
|
S-expressions can be constructed with ``SExpr::list``, for example ``SExpr expr
|
||||||
(by adding ``using SExprUtil::list`` to the top of the file, ``list`` can be used as shorthand for ``SExpr::list``).
|
= SExpr::list("add", "x", SExpr::list("mul", "y", "z"))`` represents ``(add x
|
||||||
For prettyprinting, ``SExprWriter`` wraps an ``std::ostream`` and provides the following methods:
|
(mul y z))`` (by adding ``using SExprUtil::list`` to the top of the file,
|
||||||
|
``list`` can be used as shorthand for ``SExpr::list``). For prettyprinting,
|
||||||
|
``SExprWriter`` wraps an ``std::ostream`` and provides the following methods:
|
||||||
|
|
||||||
- ``writer << sexpr`` writes the provided expression to the output, breaking long lines and adding appropriate indentation.
|
- ``writer << sexpr`` writes the provided expression to the output, breaking
|
||||||
- ``writer.open(sexpr)`` is similar to ``writer << sexpr`` but will omit the last closing parenthesis.
|
long lines and adding appropriate indentation.
|
||||||
Further arguments can then be added separately with ``<<`` or ``open``.
|
- ``writer.open(sexpr)`` is similar to ``writer << sexpr`` but will omit the
|
||||||
This allows for printing large s-expressions without needing to construct the whole expression in memory first.
|
last closing parenthesis. Further arguments can then be added separately with
|
||||||
- ``writer.open(sexpr, false)`` is similar to ``writer.open(sexpr)`` but further arguments will not be indented.
|
``<<`` or ``open``. This allows for printing large s-expressions without
|
||||||
This is used to avoid unlimited indentation on structures with unlimited nesting.
|
needing to construct the whole expression in memory first.
|
||||||
|
- ``writer.open(sexpr, false)`` is similar to ``writer.open(sexpr)`` but further
|
||||||
|
arguments will not be indented. This is used to avoid unlimited indentation on
|
||||||
|
structures with unlimited nesting.
|
||||||
- ``writer.close(n = 1)`` closes the last ``n`` open s-expressions.
|
- ``writer.close(n = 1)`` closes the last ``n`` open s-expressions.
|
||||||
- ``writer.push()`` and ``writer.pop()`` are used to automatically close s-expressions.
|
- ``writer.push()`` and ``writer.pop()`` are used to automatically close
|
||||||
``writer.pop()`` closes all s-expressions opened since the last call to ``writer.push()``.
|
s-expressions. ``writer.pop()`` closes all s-expressions opened since the last
|
||||||
|
call to ``writer.push()``.
|
||||||
- ``writer.comment(string)`` writes a comment on a separate-line.
|
- ``writer.comment(string)`` writes a comment on a separate-line.
|
||||||
``writer.comment(string, true)`` appends a comment to the last printed s-expression.
|
``writer.comment(string, true)`` appends a comment to the last printed
|
||||||
- ``writer.flush()`` flushes any buffering and should be called before any direct access to the underlying ``std::ostream``. It does not close unclosed parentheses.
|
s-expression.
|
||||||
|
- ``writer.flush()`` flushes any buffering and should be called before any
|
||||||
|
direct access to the underlying ``std::ostream``. It does not close unclosed
|
||||||
|
parentheses.
|
||||||
- The destructor calls ``flush`` but also closes all unclosed parentheses.
|
- The destructor calls ``flush`` but also closes all unclosed parentheses.
|
||||||
|
|
||||||
|
.. _minimal backend:
|
||||||
|
|
||||||
|
Example: A minimal functional backend
|
||||||
|
-------------------------------------
|
||||||
|
|
||||||
|
At its most basic, there are three steps we need to accomplish for a minimal
|
||||||
|
functional backend. First, we need to convert our design into FunctionalIR.
|
||||||
|
This is most easily done by calling the ``Functional::IR::from_module()`` static
|
||||||
|
method with our top-level module, or iterating over and converting each of the
|
||||||
|
modules in our design. Second, we need to handle each of the
|
||||||
|
``Functional::Node``\ s in our design. Iterating over the ``Functional::IR``
|
||||||
|
includes reading the module inputs and current state, but not writing the
|
||||||
|
results. So our final step is to handle the outputs and next state.
|
||||||
|
|
||||||
|
In order to add an output command to Yosys, we implement the ``Yosys::Backend``
|
||||||
|
class and provide an instance of it:
|
||||||
|
|
||||||
|
.. literalinclude:: /code_examples/functional/dummy.cc
|
||||||
|
:language: c++
|
||||||
|
:caption: Example source code for a minimal functional backend, ``dummy.cc``
|
||||||
|
|
||||||
|
Because we are using the ``Backend`` class, our ``"functional_dummy"`` is
|
||||||
|
registered as the ``write_functional_dummy`` command. The ``execute`` method is
|
||||||
|
the part that runs when the user calls the command, handling any options,
|
||||||
|
preparing the output file for writing, and iterating over selected modules in
|
||||||
|
the design. Since we don't have any options here, we set ``argidx = 1`` and
|
||||||
|
call the ``extra_args()`` method. This method will read the command arguments,
|
||||||
|
raising an error if there are any unexpected ones. It will also assign the
|
||||||
|
pointer ``f`` to the output file, or stdout if none is given.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
|
||||||
|
For more on adding new commands to Yosys and how they work, refer to
|
||||||
|
:doc:`/yosys_internals/extending_yosys/extensions`.
|
||||||
|
|
||||||
|
For this minimal example all we are doing is printing out each node. The
|
||||||
|
``node.name()`` method returns an ``RTLIL::IdString``, which we convert for
|
||||||
|
printing with ``id2cstr()``. Then, to print the function of the node, we use
|
||||||
|
``node.to_string()`` which gives us a string of the form ``function(args)``. The
|
||||||
|
``function`` part is the result of ``Functional::IR::fn_to_string(node.fn())``;
|
||||||
|
while ``args`` is the zero or more arguments passed to the function, most
|
||||||
|
commonly the name of another node. Behind the scenes, the ``node.to_string()``
|
||||||
|
method actually wraps ``node.visit(visitor)`` with a private visitor whose
|
||||||
|
return type is ``std::string``.
|
||||||
|
|
||||||
|
Finally we iterate over the module's outputs and states, using
|
||||||
|
``Functional::IROutput::value()`` and ``Functional::IRState::next_value()``
|
||||||
|
respectively in order to get the results of the transfer function.
|
||||||
|
|
||||||
|
Example: Adapting SMT-LIB backend for Rosette
|
||||||
|
---------------------------------------------
|
||||||
|
|
||||||
|
This section will introduce the SMT-LIB functional backend
|
||||||
|
(`write_functional_smt2`) and what changes are needed to work with another
|
||||||
|
s-expression target, `Rosette`_ (`write_functional_rosette`).
|
||||||
|
|
||||||
|
.. _Rosette: http://emina.github.io/rosette/
|
||||||
|
|
||||||
|
Overview
|
||||||
|
~~~~~~~~
|
||||||
|
|
||||||
|
Rosette is a solver-aided programming language that extends `Racket`_ with
|
||||||
|
language constructs for program synthesis, verification, and more. To verify
|
||||||
|
or synthesize code, Rosette compiles it to logical constraints solved with
|
||||||
|
off-the-shelf `SMT`_ solvers.
|
||||||
|
|
||||||
|
-- https://emina.github.io/rosette/
|
||||||
|
|
||||||
|
.. _Racket: http://racket-lang.org/
|
||||||
|
.. _SMT: http://smtlib.cs.uiowa.edu/
|
||||||
|
|
||||||
|
Rosette, being backed by SMT solvers and written with s-expressions, uses code
|
||||||
|
very similar to the `write_functional_smt2` output. As a result, the SMT-LIB
|
||||||
|
functional backend can be used as a starting point for implementing a Rosette
|
||||||
|
backend.
|
||||||
|
|
||||||
|
Full code listings for the initial SMT-LIB backend and the converted Rosette
|
||||||
|
backend are included in the Yosys source repository under
|
||||||
|
:file:`backends/functional` as ``smtlib.cc`` and ``smtlib_rosette.cc``
|
||||||
|
respectively. Note that the Rosette language is an extension of the Racket
|
||||||
|
language; this guide tends to refer to Racket when talking about the underlying
|
||||||
|
semantics/syntax of the language.
|
||||||
|
|
||||||
|
The major changes from the SMT-LIB backend are as follows:
|
||||||
|
|
||||||
|
- all of the ``Smt`` prefixes in names are replaced with ``Smtr`` to mean
|
||||||
|
``smtlib_rosette``;
|
||||||
|
- syntax is adjusted for Racket;
|
||||||
|
- data structures for input/output/state are changed from using
|
||||||
|
``declare-datatype`` with statically typed fields, to using ``struct`` with no
|
||||||
|
static typing;
|
||||||
|
- the transfer function also loses its static typing;
|
||||||
|
- sign/zero extension in Rosette use the output width instead of the number of
|
||||||
|
extra bits, gaining static typing;
|
||||||
|
- the single scope is traded for a global scope with local scope for each
|
||||||
|
struct;
|
||||||
|
- initial state is provided as a constant value instead of a set of assertions;
|
||||||
|
- and the ``-provides`` option is introduced to more easily use generated code
|
||||||
|
within Rosette based applications.
|
||||||
|
|
||||||
|
Scope
|
||||||
|
~~~~~
|
||||||
|
|
||||||
|
Our first addition to the `minimal backend`_ above is that for both SMT-LIB and
|
||||||
|
Rosette backends, we are now targetting real languages which bring with them
|
||||||
|
their own sets of constraints with what we can use as identifiers. This is
|
||||||
|
where the ``Functional::Scope`` class described above comes in; by using this
|
||||||
|
class we can safely rename our identifiers in the generated output without
|
||||||
|
worrying about collisions or illegal names/characters.
|
||||||
|
|
||||||
|
In the SMT-LIB version, the ``SmtScope`` class implements ``Scope<int>``;
|
||||||
|
provides a constructor that iterates over a list of reserved keywords, calling
|
||||||
|
``reserve`` on each; and defines the ``is_character_legal`` method to reject any
|
||||||
|
characters which are not allowed in SMT-LIB variable names to then be replaced
|
||||||
|
with underscores in the output. To use this scope we create an instance of it,
|
||||||
|
and call the ``Scope::unique_name()`` method to generate a unique and legal name
|
||||||
|
for each of our identifiers.
|
||||||
|
|
||||||
|
In the Rosette version we update the list of legal ascii characters in the
|
||||||
|
``is_character_legal`` method to only those allowed in Racket variable names.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Scope`` class
|
||||||
|
:start-at: -struct SmtScope : public Functional::Scope<int> {
|
||||||
|
:end-at: };
|
||||||
|
|
||||||
|
For the reserved keywords we trade the SMT-LIB specification for Racket to
|
||||||
|
prevent parts of our design from accidentally being treated as Racket code. We
|
||||||
|
also no longer need to reserve ``pair``, ``first``, and ``second``. In
|
||||||
|
`write_functional_smt2` these are used for combining the ``(inputs,
|
||||||
|
current_state)`` and ``(outputs, next_state)`` into a single variable. Racket
|
||||||
|
provides this functionality natively with ``cons``, which we will see later.
|
||||||
|
|
||||||
|
.. inlined diff for skipping the actual lists
|
||||||
|
.. code-block:: diff
|
||||||
|
:caption: diff of ``reserved_keywords`` list
|
||||||
|
|
||||||
|
const char *reserved_keywords[] = {
|
||||||
|
- // reserved keywords from the smtlib spec
|
||||||
|
- ...
|
||||||
|
+ // reserved keywords from the racket spec
|
||||||
|
+ ...
|
||||||
|
|
||||||
|
// reserved for our own purposes
|
||||||
|
- "pair", "Pair", "first", "second",
|
||||||
|
- "inputs", "state",
|
||||||
|
+ "inputs", "state", "name",
|
||||||
|
nullptr
|
||||||
|
};
|
||||||
|
|
||||||
|
.. note:: We skip over the actual list of reserved keywords from both the smtlib
|
||||||
|
and racket specifications to save on space in this document.
|
||||||
|
|
||||||
|
Sort
|
||||||
|
~~~~
|
||||||
|
|
||||||
|
Next up in `write_functional_smt2` we see the ``Sort`` class. This is a wrapper
|
||||||
|
for the ``Functional::Sort`` class, providing the additional functionality of
|
||||||
|
mapping variable declarations to s-expressions with the ``to_sexpr()`` method.
|
||||||
|
The main change from ``SmtSort`` to ``SmtrSort`` is a syntactical one with
|
||||||
|
signals represented as ``bitvector``\ s, and memories as ``list``\ s of signals.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Sort`` wrapper
|
||||||
|
:start-at: SExpr to_sexpr() const {
|
||||||
|
:end-before: };
|
||||||
|
|
||||||
|
Struct
|
||||||
|
~~~~~~
|
||||||
|
|
||||||
|
As we saw in the `minimal backend`_ above, the ``Functional::IR`` class tracks
|
||||||
|
the set of inputs, the set of outputs, and the set of "state" variables. The
|
||||||
|
SMT-LIB backend maps each of these sets into its own ``SmtStruct``, with each
|
||||||
|
variable getting a corresponding field in the struct and a specified `Sort`_.
|
||||||
|
`write_functional_smt2` then defines each of these structs as a new
|
||||||
|
``datatype``, with each element being strongly-typed.
|
||||||
|
|
||||||
|
In Rosette, rather than defining new datatypes for our structs, we use the
|
||||||
|
native ``struct``. We also only declare each field by name because Racket
|
||||||
|
provides less static typing. For ease of use, we provide the expected type for
|
||||||
|
each field as comments.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``write_definition`` method
|
||||||
|
:start-at: void write_definition
|
||||||
|
:end-before: template<typename Fn> void write_value
|
||||||
|
|
||||||
|
Each field is added to the ``SmtStruct`` with the ``insert`` method, which also
|
||||||
|
reserves a unique name (or accessor) within the `Scope`_. These accessors
|
||||||
|
combine the struct name and field name and are globally unique, being used in
|
||||||
|
the ``access`` method for reading values from the input/current state.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/smtlib.cc
|
||||||
|
:language: c++
|
||||||
|
:caption: ``Struct::access()`` method
|
||||||
|
:start-at: SExpr access(
|
||||||
|
:end-before: };
|
||||||
|
|
||||||
|
In Rosette, struct fields are accessed as ``<struct_name>-<field_name>`` so
|
||||||
|
including the struct name in the field name would be redundant. For
|
||||||
|
`write_functional_rosette` we instead choose to make field names unique only
|
||||||
|
within the struct, while accessors are unique across the whole module. We thus
|
||||||
|
modify the class constructor and ``insert`` method to support this; providing
|
||||||
|
one scope that is local to the struct (``local_scope``) and one which is shared
|
||||||
|
across the whole module (``global_scope``), leaving the ``access`` method
|
||||||
|
unchanged.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of struct constructor
|
||||||
|
:start-at: SmtStruct(std::string name, SmtScope &scope)
|
||||||
|
:end-before: void write_definition
|
||||||
|
|
||||||
|
Finally, ``SmtStruct`` also provides a ``write_value`` template method which
|
||||||
|
calls a provided function on each element in the struct. This is used later for
|
||||||
|
assigning values to the output/next state pair. The only change here is to
|
||||||
|
remove the check for zero-argument constructors since this is not necessary with
|
||||||
|
Rosette ``struct``\ s.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``write_value`` method
|
||||||
|
:start-at: template<typename Fn> void write_value
|
||||||
|
:end-before: SExpr access
|
||||||
|
|
||||||
|
PrintVisitor
|
||||||
|
~~~~~~~~~~~~
|
||||||
|
|
||||||
|
Remember in the `minimal backend`_ we converted nodes into strings for writing
|
||||||
|
using the ``node.to_string()`` method, which wrapped ``node.visit()`` with a
|
||||||
|
private visitor. We now want a custom visitor which can convert nodes into
|
||||||
|
s-expressions. This is where the ``PrintVisitor`` comes in, implementing the
|
||||||
|
abstract ``Functional::AbstractVisitor`` class with a return type of ``SExpr``.
|
||||||
|
For most functions, the Rosette output is very similar to the corresponding
|
||||||
|
SMT-LIB function with minor adjustments for syntax.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: portion of ``Functional::AbstractVisitor`` implementation diff showing similarities
|
||||||
|
:start-at: SExpr logical_shift_left
|
||||||
|
:end-at: "list-set-bv"
|
||||||
|
|
||||||
|
However there are some differences in the two formats with regards to how
|
||||||
|
booleans are handled, with Rosette providing built-in functions for conversion.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: portion of ``Functional::AbstractVisitor`` implementation diff showing differences
|
||||||
|
:start-at: SExpr from_bool
|
||||||
|
:end-before: SExpr extract
|
||||||
|
|
||||||
|
Of note here is the rare instance of the Rosette implementation *gaining* static
|
||||||
|
typing rather than losing it. Where SMT_LIB calls zero/sign extension with the
|
||||||
|
number of extra bits needed (given by ``out_width - a.width()``), Rosette
|
||||||
|
instead specifies the type of the output (given by ``list("bitvector",
|
||||||
|
out_width)``).
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: zero/sign extension implementation diff
|
||||||
|
:start-after: SExpr buf(
|
||||||
|
:end-before: SExpr concat(
|
||||||
|
:lines: 2-3, 5-6
|
||||||
|
|
||||||
|
.. note:: Be sure to check the source code for the full list of differences here.
|
||||||
|
|
||||||
|
Module
|
||||||
|
~~~~~~
|
||||||
|
|
||||||
|
With most of the supporting classes out of the way, we now reach our three main
|
||||||
|
steps from the `minimal backend`_. These are all handled by the ``SmtModule``
|
||||||
|
class, with the mapping from RTLIL module to FunctionalIR happening in the
|
||||||
|
constructor. Each of the three ``SmtStruct``\ s; inputs, outputs, and state;
|
||||||
|
are also created in the constructor, with each value in the corresponding lists
|
||||||
|
in the IR being ``insert``\ ed.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/smtlib.cc
|
||||||
|
:language: c++
|
||||||
|
:caption: ``SmtModule`` constructor
|
||||||
|
:start-at: SmtModule(Module
|
||||||
|
:end-at: }
|
||||||
|
|
||||||
|
Since Racket uses the ``-`` to access struct fields, the ``SmtrModule`` instead
|
||||||
|
uses an underscore for the name of the initial state.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Module`` constructor
|
||||||
|
:start-at: scope.reserve(name
|
||||||
|
:end-before: for (auto input
|
||||||
|
|
||||||
|
The ``write`` method is then responsible for writing the FunctionalIR to the
|
||||||
|
output file, formatted for the corresponding backend. ``SmtModule::write()``
|
||||||
|
breaks the output file down into four parts: defining the three structs,
|
||||||
|
declaring the ``pair`` datatype, defining the transfer function ``(inputs,
|
||||||
|
current_state) -> (outputs, next_state)`` with ``write_eval``, and declaring the
|
||||||
|
initial state with ``write_initial``. The only change for the ``SmtrModule`` is
|
||||||
|
that the ``pair`` declaration isn't needed.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Module::write()`` method
|
||||||
|
:start-at: void write(std::ostream &out)
|
||||||
|
:end-at: }
|
||||||
|
|
||||||
|
The ``write_eval`` method is where the FunctionalIR nodes, outputs, and next
|
||||||
|
state are handled. Just as with the `minimal backend`_, we iterate over the
|
||||||
|
nodes with ``for(auto n : ir)``, and then use the ``Struct::write_value()``
|
||||||
|
method for the ``output_struct`` and ``state_struct`` to iterate over the
|
||||||
|
outputs and next state respectively.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/smtlib.cc
|
||||||
|
:language: c++
|
||||||
|
:caption: iterating over FunctionalIR nodes in ``SmtModule::write_eval()``
|
||||||
|
:start-at: for(auto n : ir)
|
||||||
|
:end-at: }
|
||||||
|
|
||||||
|
The main differences between our two backends here are syntactical. First we
|
||||||
|
change the ``define-fun`` for the Racket style ``define`` which drops the
|
||||||
|
explicitly typed inputs/outputs. And then we change the final result from a
|
||||||
|
``pair`` to the native ``cons`` which acts in much the same way, returning both
|
||||||
|
the ``outputs`` and the ``next_state`` in a single variable.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Module::write_eval()`` transfer function declaration
|
||||||
|
:start-at: w.open(list("define-fun"
|
||||||
|
:end-at: w.open(list("define"
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of output/next state handling ``Module::write_eval()``
|
||||||
|
:start-at: w.open(list("pair"
|
||||||
|
:end-at: w.pop();
|
||||||
|
|
||||||
|
For the ``write_initial`` method, the SMT-LIB backend uses ``declare-const`` and
|
||||||
|
``assert``\ s which must always hold true. For Rosette we instead define the
|
||||||
|
initial state as any other variable that can be used by external code. This
|
||||||
|
variable, ``[name]_initial``, can then be used in the ``[name]`` function call;
|
||||||
|
allowing the Rosette code to be used in the generation of the ``next_state``,
|
||||||
|
whereas the SMT-LIB code can only verify that a given ``next_state`` is correct.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: diff of ``Module::write_initial()`` method
|
||||||
|
:start-at: void write_initial
|
||||||
|
:end-before: void write
|
||||||
|
|
||||||
|
Backend
|
||||||
|
~~~~~~~
|
||||||
|
|
||||||
|
The final part is the ``Backend`` itself, with much of the same boiler plate as
|
||||||
|
the `minimal backend`_. The main difference is that we use the `Module`_ to
|
||||||
|
perform the actual processing.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/smtlib.cc
|
||||||
|
:language: c++
|
||||||
|
:caption: The ``FunctionalSmtBackend``
|
||||||
|
:start-at: struct FunctionalSmtBackend
|
||||||
|
:end-at: } FunctionalSmtBackend;
|
||||||
|
|
||||||
|
There are two additions here for Rosette. The first is that the output file
|
||||||
|
needs to start with the ``#lang`` definition which tells the
|
||||||
|
compiler/interpreter that we want to use the Rosette language module. The
|
||||||
|
second is that the `write_functional_rosette` command takes an optional
|
||||||
|
argument, ``-provides``. If this argument is given, then the output file gets
|
||||||
|
an additional line declaring that everything in the file should be exported for
|
||||||
|
use; allowing the file to be treated as a Racket package with structs and
|
||||||
|
mapping function available for use externally.
|
||||||
|
|
||||||
|
.. literalinclude:: /generated/functional/rosette.diff
|
||||||
|
:language: diff
|
||||||
|
:caption: relevant portion of diff of ``Backend::execute()`` method
|
||||||
|
:start-at: lang rosette/safe
|
||||||
|
:end-before: for (auto module
|
||||||
|
|
Loading…
Reference in a new issue