From cc0070fde2aa538b3a84fa99fce6acd90c7239d9 Mon Sep 17 00:00:00 2001 From: Gus Smith Date: Sun, 18 Feb 2024 15:53:01 -0800 Subject: [PATCH] Convert Yosys fork into Yosys plugin plus other cleanup (#4) * Add old churchroad content * Lots of deleting and moving * Update workflow * Cut down Dockerfile * Fix command * Run workflow on GitHub runners * Move lakeroad-egglog files to top level * Fix test script; dump README * Rename project * Fix breakages after egglog updates * Add files * Build Yosys in Dockerfile * Add ARG * Build Yosys plugin * Fix path * Run plugin tests * Update Yosys * Delete file * Remove Yosys dependency; move test * Update PATH * Add readme * Make `lit` binary * Point to llvm-config * Try to enable caching with cargo build * Change how format check works * Title * Fix fmt.sh * Install lit via pip * Undo use of cargo chef for now Going to see if just ADDing files more intelligently clears up caching issue. * Add .dockerignore * Add files more intelligently * Take into account the weird behavior of ADD * Fix weird issue with trying to run " bash" binary * Expect a CHURCHROAD_DIR env var to be set * Load plugin in tests * Add yosys files to test files * Add env template * ignore file * Fix * Improve testing of the Yosys plugin Specifically, add an overridable YOSYS environment variable. * Add YOSYS env var to .env.template * Fix test * Whoops, wasn't FileChecking * Add TODO * Fix command * Delete comments * Fix command * Newline * Delete .gitmodules * ADD all files in the end, plus git dir, for fmt reasons * Format .cfg file * Format * Status instead of diff, shows more info for debug * ignore file * Remove other .gitignore * Update actions * Rename and reword workflow file * Update .env.template with documentation --- .dockerignore | 18 + .env.template | 18 + ...unners.yml => built-test-check-format.yml} | 20 +- .gitignore | 4 + .gitmodules | 3 - Dockerfile | 63 +- README.md | 12 + dependencies.sh | 22 + fmt.sh | 9 +- requirements.txt | 2 + run-tests.sh | 4 +- yosys | 1 - yosys-plugin/Makefile | 7 + yosys-plugin/README.md | 30 + yosys-plugin/churchroad.cc | 1891 +++++++++++++++++ yosys-plugin/run_tests.sh | 8 + yosys-plugin/tests/example.ys | 25 + yosys-plugin/tests/lit.cfg | 60 + yosys-plugin/tests/permuter.sv | 111 + yosys-plugin/tests/simple.sv | 20 + 20 files changed, 2303 insertions(+), 25 deletions(-) create mode 100644 .dockerignore create mode 100644 .env.template rename .github/workflows/{run-checks-github-runners.yml => built-test-check-format.yml} (87%) delete mode 100644 .gitmodules create mode 100644 dependencies.sh create mode 100644 requirements.txt delete mode 160000 yosys create mode 100644 yosys-plugin/Makefile create mode 100644 yosys-plugin/README.md create mode 100644 yosys-plugin/churchroad.cc create mode 100755 yosys-plugin/run_tests.sh create mode 100644 yosys-plugin/tests/example.ys create mode 100644 yosys-plugin/tests/lit.cfg create mode 100644 yosys-plugin/tests/permuter.sv create mode 100644 yosys-plugin/tests/simple.sv diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 0000000..ecd123f --- /dev/null +++ b/.dockerignore @@ -0,0 +1,18 @@ +*.dep +*.zo + +.vscode/* +.lit_test_times.txt +**/Output + +**/.DS_Store +**/.lit_test_times.txt +**/Output/ +**/__pycache__ + +target +Cargo.lock +.env + +**/*.so* +**/*.d diff --git a/.env.template b/.env.template new file mode 100644 index 0000000..95e8220 --- /dev/null +++ b/.env.template @@ -0,0 +1,18 @@ +#!/bin/bash + +# User should set these variables. +# +# Optional environment variables which you opt not to set should be +# deleted/commented out. +# +# OPTIONAL: Path to llvm-config binary. Only needed if LLVM's `llvm-config` and +# `FileCheck` binaries are not in the PATH. +export LLVM_CONFIG="/path/to/llvm-config" +# OPTIONAL: Path to Yosys binary. Only needed if the `yosys` binary is not +# available on your path, or if you want to use a specific version of Yosys. +export YOSYS="/path/to/yosys" + + +# Do not modify below this line. +SCRIPTPATH="$( cd -- "$(dirname "$0")" >/dev/null 2>&1 ; pwd -P )" +export CHURCHROAD_DIR=$SCRIPTPATH diff --git a/.github/workflows/run-checks-github-runners.yml b/.github/workflows/built-test-check-format.yml similarity index 87% rename from .github/workflows/run-checks-github-runners.yml rename to .github/workflows/built-test-check-format.yml index 2b3813d..17e2582 100644 --- a/.github/workflows/run-checks-github-runners.yml +++ b/.github/workflows/built-test-check-format.yml @@ -1,4 +1,4 @@ -name: Build Docker image, format check, and run tests +name: Build, Test, Check Format on: push: @@ -48,10 +48,10 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v3 + uses: actions/checkout@v4 - name: Log in to the Container registry - uses: docker/login-action@v2 + uses: docker/login-action@v3 with: registry: ${{ env.REGISTRY }} username: ${{ github.actor }} @@ -65,10 +65,10 @@ jobs: # images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} - name: Docker Setup Buildx - uses: docker/setup-buildx-action@v2 + uses: docker/setup-buildx-action@v3 - name: Build and push Docker image - uses: docker/build-push-action@v4 + uses: docker/build-push-action@v5 with: context: . push: true @@ -85,7 +85,7 @@ jobs: ] steps: - name: Log in to the Container registry - uses: docker/login-action@v2 + uses: docker/login-action@v3 with: registry: ${{ env.REGISTRY }} username: ${{ github.actor }} @@ -102,7 +102,7 @@ jobs: needs: build-and-push-image steps: - name: Log in to the Container registry - uses: docker/login-action@v2 + uses: docker/login-action@v3 with: registry: ${{ env.REGISTRY }} username: ${{ github.actor }} @@ -111,5 +111,7 @@ jobs: - name: Pull image run: docker pull ${{ env.IMAGE_TAG }} - - name: Rust format check - run: docker run ${{ env.IMAGE_TAG }} cargo fmt --manifest-path ./Cargo.toml -- --check + - name: Project-wide formatting check + run: | + docker run ${{ env.IMAGE_TAG }} \ + bash -c './fmt.sh && git status && [ -z "$(git status --porcelain)" ]' diff --git a/.gitignore b/.gitignore index cdee069..ecd123f 100644 --- a/.gitignore +++ b/.gitignore @@ -12,3 +12,7 @@ target Cargo.lock +.env + +**/*.so* +**/*.d diff --git a/.gitmodules b/.gitmodules deleted file mode 100644 index fb09743..0000000 --- a/.gitmodules +++ /dev/null @@ -1,3 +0,0 @@ -[submodule "lakeroad-egglog/yosys"] - path = yosys - url = git@github.com:uwsampl/yosys diff --git a/Dockerfile b/Dockerfile index e256966..694b41e 100644 --- a/Dockerfile +++ b/Dockerfile @@ -51,20 +51,67 @@ RUN apt install -y \ RUN curl https://sh.rustup.rs -sSf | sh -s -- -y ENV PATH="/root/.cargo/bin:$PATH" +# Build Rust package. +WORKDIR /root/churchroad +# ADD has weird behavior when it comes to directories. That's why we need so +# many ADDs. +ADD egglog_src egglog_src +ADD src src +ADD tests tests +ADD Cargo.toml . +RUN cargo build + +# Build Yosys. +WORKDIR /root +ARG MAKE_JOBS=2 +ADD dependencies.sh . +RUN source /root/dependencies.sh \ + && mkdir yosys && cd yosys \ + && wget -qO- https://github.com/YosysHQ/yosys/archive/$YOSYS_COMMIT_HASH.tar.gz | tar xz --strip-components=1 \ + && PREFIX="/root/.local" CPLUS_INCLUDE_PATH="/usr/include/tcl8.6/:$CPLUS_INCLUDE_PATH" make config-gcc \ + && PREFIX="/root/.local" CPLUS_INCLUDE_PATH="/usr/include/tcl8.6/:$CPLUS_INCLUDE_PATH" make -j ${MAKE_JOBS} install \ + && rm -rf /root/yosys + +# Add /root/.local/bin to PATH. +ENV PATH="/root/.local/bin:$PATH" + +# Build Yosys plugin. +WORKDIR /root/churchroad/yosys-plugin +ADD yosys-plugin . +RUN make -j ${MAKE_JOBS} + +# Make a binary for `lit`. If you're on Mac, you can install lit via Brew. +# Ubuntu doesn't have a binary for it, but it is available on pip and is +# installed later in this Dockerfile. +WORKDIR /root +RUN mkdir -p /root/.local/bin \ + && echo "#!/usr/bin/env python3" >> /root/.local/bin/lit \ + && echo "from lit.main import main" >> /root/.local/bin/lit \ + && echo "if __name__ == '__main__': main()" >> /root/.local/bin/lit \ + && chmod +x /root/.local/bin/lit + +# Point to llvm-config binary. Alternatively, make sure you have a binary called +# `llvm-config` on your PATH. +ENV LLVM_CONFIG=llvm-config-14 + +# Python dependencies. +WORKDIR /root/churchroad +ADD requirements.txt . +RUN pip install -r requirements.txt + +ENV CHURCHROAD_DIR=/root/churchroad + # Add other Churchroad files. It's useful to put this as far down as possible. # In general, only ADD files just before they're needed. This maximizes the # ability to cache intermediate containers and minimizes rebuilding. # -# In reality, we use the git functionality of ADD (enabled in our case via the -# optional flag --keep-git-dir) to add all of the checked-in files of the -# Churchroad repo (but not including the .git directory itself). We could cut -# this down further if we wanted, but I think this is a clean approach for now. +# We use the git functionality of ADD (enabled in our case via the optional flag +# --keep-git-dir) to add all of the checked-in files of the Churchroad repo (but +# not including the .git directory itself). We could cut this down further if we +# wanted, but I think this is a clean approach for now. WORKDIR /root/churchroad ADD --keep-git-dir=false . . -# Build Rust package. -WORKDIR /root/churchroad -RUN cargo build - WORKDIR /root/churchroad +ADD fmt.sh run-tests.sh ./ CMD [ "/bin/bash", "run-tests.sh" ] diff --git a/README.md b/README.md index 373e4c4..c2d3b8b 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,15 @@ # Churchroad :letsgo3: + +## Yosys Plugin + +[`./yosys-plugin/](./yosys-plugin/) + +We provide a Yosys plugin, + which can output Churchroad IR + from Yosys. +Please see the README + in that directory + for information on building + and using the plugin. diff --git a/dependencies.sh b/dependencies.sh new file mode 100644 index 0000000..5bab82e --- /dev/null +++ b/dependencies.sh @@ -0,0 +1,22 @@ +#!/bin/sh +# This script exports a number of variables that help to pin the versions of +# various Churchroad dependencies. +# +# To use, source this script before running relevant commands e.g. in a +# Dockerfile. +# +# This is just one possible way to implement dependency tracking. Some of the +# other options are: +# - No tracking at all. E.g. in the Dockerfile, we could clone Yosys at a +# specific commit, using a "magic constant" commit hash. This isn't ideal, +# because if we use Yosys elsewhere (e.g. in the evaluation) we have to make +# sure we keep the commit hashes in sync. +# - Git submodules. This is very similar to what we've chosen to do, but it's +# more directly supported by Git. However, it's a bit overkill to add a full +# repository as a submodule when we only need the resulting binary. +# +# This option is essentially a lighter-weight version of submodules. We track +# the commit hashes of the dependencies we need, but nothing additional is +# cloned on a `git clone --recursive`. + +export YOSYS_COMMIT_HASH="f8d4d7128cf72456cc03b0738a8651ac5dbe52e1" diff --git a/fmt.sh b/fmt.sh index 53c64ee..535a48b 100755 --- a/fmt.sh +++ b/fmt.sh @@ -2,6 +2,9 @@ set -e SCRIPT_DIR=$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" &>/dev/null && pwd) -raco fmt -i "$SCRIPT_DIR"/racket/*.rkt -raco fmt -i "$SCRIPT_DIR"/bin/*.rkt -cargo fmt --manifest-path "$SCRIPT_DIR"/rust/Cargo.toml + +# Rust formatting. +cargo fmt --manifest-path "$SCRIPT_DIR"/Cargo.toml + +# Python formatting. +black "$SCRIPT_DIR"/yosys-plugin/tests/lit.cfg diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..acc40b6 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,2 @@ +black==24.2.0 +lit==15.0.0 diff --git a/run-tests.sh b/run-tests.sh index 95a4935..ed7b9ed 100755 --- a/run-tests.sh +++ b/run-tests.sh @@ -2,4 +2,6 @@ set -e -cargo test \ No newline at end of file +cargo test + +./yosys-plugin/run_tests.sh \ No newline at end of file diff --git a/yosys b/yosys deleted file mode 160000 index 316e2bd..0000000 --- a/yosys +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 316e2bdcb8b2f227342af162a6d09c9b63a65ad4 diff --git a/yosys-plugin/Makefile b/yosys-plugin/Makefile new file mode 100644 index 0000000..b83dde8 --- /dev/null +++ b/yosys-plugin/Makefile @@ -0,0 +1,7 @@ +churchroad.so: churchroad.cc + $(CXX) $(shell yosys-config --cxxflags --ldflags) -shared -o $@ churchroad.cc -lboost_filesystem + +clean: + rm -rfv *.d *.o churchroad.so* + +-include *.d diff --git a/yosys-plugin/README.md b/yosys-plugin/README.md new file mode 100644 index 0000000..00e36a3 --- /dev/null +++ b/yosys-plugin/README.md @@ -0,0 +1,30 @@ +# Churchroad Backend Plugin for Yosys + +We provide a plugin + for Yosys + which produces Churchroad code. + +## Note on Building + +Yosys plugins are simply shared libraries + that can be loaded into Yosys at runtime. +As a result, + you are required to build + a Yosys plugin using the same + build environment + as your `yosys` binary. +The easiest way to ensure this + is to simply build Yosys + from source. +If you prefer to use + a prebuilt Yosys binary, + Yosys provides a Docker image + which sets up the appropriate + environment: + +We don't yet support + using this option easily. +If you would like this feature, + please open an issue. +Until then, all plugin users should build Yosys + from source. diff --git a/yosys-plugin/churchroad.cc b/yosys-plugin/churchroad.cc new file mode 100644 index 0000000..21f1377 --- /dev/null +++ b/yosys-plugin/churchroad.cc @@ -0,0 +1,1891 @@ +/* + * yosys -- Yosys Open SYnthesis Suite + * + * Copyright (C) 2012 Claire Xenia Wolf + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + * + */ + +#include "kernel/celltypes.h" +#include "kernel/json.h" +#include "kernel/log.h" +#include "kernel/mem.h" +#include "kernel/register.h" +#include "kernel/rtlil.h" +#include "kernel/sigtools.h" +#include "kernel/yw.h" +#include +#include + +USING_YOSYS_NAMESPACE +PRIVATE_NAMESPACE_BEGIN + +struct LakeroadWorker +{ + std::ostream &f; + SigMap sigmap; + RTLIL::Module *module; + bool verbose; + bool single_bad; + bool cover_mode; + bool print_internal_names; + + int next_nid = 1; + int initstate_nid = -1; + + // => + dict sorts_bv; + + // (, ) => + dict, int> sorts_mem; + + // SigBit => (, ) + dict> bit_nid; + + // => + dict nid_width; + + // SigSpec => + dict sig_nid; + + // bit to driving cell + dict bit_cell; + + // nids for constants + dict consts; + + // ff inputs that need to be evaluated (, ) + vector> ff_todo; + vector> mem_todo; + + pool cell_recursion_guard; + vector bad_properties; + dict initbits; + pool statewires; + pool srcsymbols; + vector memories; + dict mem_cells; + + string indent, info_filename; + vector info_lines; + dict info_clocks; + + struct ywmap_btor_sig + { + SigSpec sig; + Cell *cell = nullptr; + + ywmap_btor_sig(const SigSpec &sig) : sig(sig) {} + ywmap_btor_sig(Cell *cell) : cell(cell) {} + }; + + vector ywmap_inputs; + vector ywmap_states; + dict ywmap_clock_bits; + dict ywmap_clock_inputs; + + PrettyJson ywmap_json; + + void btorf(const char *fmt, ...) YS_ATTRIBUTE(format(printf, 2, 3)) + { + va_list ap; + va_start(ap, fmt); + f << indent << vstringf(fmt, ap); + va_end(ap); + } + + void infof(const char *fmt, ...) YS_ATTRIBUTE(format(printf, 2, 3)) + { + va_list ap; + va_start(ap, fmt); + info_lines.push_back(vstringf(fmt, ap)); + va_end(ap); + } + + template + string getinfo(T *obj, bool srcsym = false) + { + string infostr = log_id(obj); + if (!srcsym && !print_internal_names && infostr[0] == '$') + return ""; + if (obj->attributes.count(ID::src)) + { + string src = obj->attributes.at(ID::src).decode_string().c_str(); + if (srcsym && infostr[0] == '$') + { + std::replace(src.begin(), src.end(), ' ', '_'); + if (srcsymbols.count(src) || module->count_id("\\" + src)) + { + for (int i = 1;; i++) + { + string s = stringf("%s-%d", src.c_str(), i); + if (!srcsymbols.count(s) && !module->count_id("\\" + s)) + { + src = s; + break; + } + } + } + srcsymbols.insert(src); + infostr = src; + } + else + { + infostr += " ; " + src; + } + } + return " " + infostr; + } + + void ywmap_state(const SigSpec &sig) + { + if (ywmap_json.active()) + ywmap_states.emplace_back(sig); + } + + void ywmap_state(Cell *cell) + { + if (ywmap_json.active()) + ywmap_states.emplace_back(cell); + } + + void ywmap_input(const SigSpec &sig) + { + if (ywmap_json.active()) + ywmap_inputs.emplace_back(sig); + } + + void emit_ywmap_btor_sig(const ywmap_btor_sig &btor_sig) + { + if (btor_sig.cell == nullptr) + { + if (btor_sig.sig.empty()) + { + ywmap_json.value(nullptr); + return; + } + ywmap_json.begin_array(); + ywmap_json.compact(); + for (auto &chunk : btor_sig.sig.chunks()) + { + log_assert(chunk.is_wire()); + + ywmap_json.begin_object(); + ywmap_json.entry("path", witness_path(chunk.wire)); + ywmap_json.entry("width", chunk.width); + ywmap_json.entry("offset", chunk.offset); + ywmap_json.end_object(); + } + ywmap_json.end_array(); + } + else + { + ywmap_json.begin_object(); + ywmap_json.compact(); + ywmap_json.entry("path", witness_path(btor_sig.cell)); + Mem *mem = mem_cells[btor_sig.cell]; + ywmap_json.entry("width", mem->width); + ywmap_json.entry("size", mem->size); + ywmap_json.end_object(); + } + } + + void btorf_push(const string &id) + { + if (verbose) + { + f << indent << stringf(" ; begin %s\n", id.c_str()); + indent += " "; + } + } + + void btorf_pop(const string &id) + { + if (verbose) + { + indent = indent.substr(4); + f << indent << stringf(" ; end %s\n", id.c_str()); + } + } + + int get_bv_sid(int width) + { + if (sorts_bv.count(width) == 0) + { + int nid = next_nid++; + btorf("%d sort bitvec %d\n", nid, width); + sorts_bv[width] = nid; + } + return sorts_bv.at(width); + } + + int get_mem_sid(int abits, int dbits) + { + pair key(abits, dbits); + if (sorts_mem.count(key) == 0) + { + int addr_sid = get_bv_sid(abits); + int data_sid = get_bv_sid(dbits); + int nid = next_nid++; + btorf("%d sort array %d %d\n", nid, addr_sid, data_sid); + sorts_mem[key] = nid; + } + return sorts_mem.at(key); + } + + void add_nid_sig(int nid, const SigSpec &sig) + { + if (verbose) + f << indent << stringf("; %d %s\n", nid, log_signal(sig)); + + for (int i = 0; i < GetSize(sig); i++) + bit_nid[sig[i]] = make_pair(nid, i); + + sig_nid[sig] = nid; + nid_width[nid] = GetSize(sig); + } + + void export_cell(Cell *cell) + { + if (cell_recursion_guard.count(cell)) + { + string cell_list; + for (auto c : cell_recursion_guard) + cell_list += stringf("\n %s", log_id(c)); + log_error("Found topological loop while processing cell %s. Active cells:%s\n", log_id(cell), cell_list.c_str()); + } + + cell_recursion_guard.insert(cell); + btorf_push(log_id(cell)); + + if (cell->type.in(ID($add), ID($sub), ID($mul), ID($and), ID($or), ID($xor), ID($xnor), ID($shl), ID($sshl), ID($shr), ID($sshr), + ID($shift), ID($shiftx), ID($concat), ID($_AND_), ID($_NAND_), ID($_OR_), ID($_NOR_), ID($_XOR_), ID($_XNOR_))) + { + string btor_op; + if (cell->type == ID($add)) + btor_op = "add"; + if (cell->type == ID($sub)) + btor_op = "sub"; + if (cell->type == ID($mul)) + btor_op = "mul"; + if (cell->type.in(ID($shl), ID($sshl))) + btor_op = "sll"; + if (cell->type == ID($shr)) + btor_op = "srl"; + if (cell->type == ID($sshr)) + btor_op = "sra"; + if (cell->type.in(ID($shift), ID($shiftx))) + btor_op = "shift"; + if (cell->type.in(ID($and), ID($_AND_))) + btor_op = "and"; + if (cell->type.in(ID($or), ID($_OR_))) + btor_op = "or"; + if (cell->type.in(ID($xor), ID($_XOR_))) + btor_op = "xor"; + if (cell->type == ID($concat)) + btor_op = "concat"; + if (cell->type == ID($_NAND_)) + btor_op = "nand"; + if (cell->type == ID($_NOR_)) + btor_op = "nor"; + if (cell->type.in(ID($xnor), ID($_XNOR_))) + btor_op = "xnor"; + log_assert(!btor_op.empty()); + + int width_ay = std::max(GetSize(cell->getPort(ID::A)), GetSize(cell->getPort(ID::Y))); + int width = std::max(width_ay, GetSize(cell->getPort(ID::B))); + + bool a_signed = cell->hasParam(ID::A_SIGNED) ? cell->getParam(ID::A_SIGNED).as_bool() : false; + bool b_signed = cell->hasParam(ID::B_SIGNED) ? cell->getParam(ID::B_SIGNED).as_bool() : false; + + if (btor_op == "shift" && !b_signed) + btor_op = "srl"; + + if (cell->type.in(ID($shl), ID($sshl), ID($shr), ID($sshr))) + b_signed = false; + + if (cell->type == ID($sshr) && !a_signed) + btor_op = "srl"; + + int sid = get_bv_sid(width); + int nid; + + int nid_a; + if (cell->type.in(ID($shl), ID($shr), ID($shift), ID($shiftx)) && a_signed && width_ay < width) + { + // sign-extend A up to the width of Y + int nid_a_padded = get_sig_nid(cell->getPort(ID::A), width_ay, a_signed); + + // zero-extend the rest + int zeroes = get_sig_nid(Const(0, width - width_ay)); + nid_a = next_nid++; + btorf("%d concat %d %d %d\n", nid_a, sid, zeroes, nid_a_padded); + } + else + { + nid_a = get_sig_nid(cell->getPort(ID::A), width, a_signed); + } + + int nid_b = get_sig_nid(cell->getPort(ID::B), width, b_signed); + + if (btor_op == "shift") + { + int nid_r = next_nid++; + btorf("%d srl %d %d %d\n", nid_r, sid, nid_a, nid_b); + + int nid_b_neg = next_nid++; + btorf("%d neg %d %d\n", nid_b_neg, sid, nid_b); + + int nid_l = next_nid++; + btorf("%d sll %d %d %d\n", nid_l, sid, nid_a, nid_b_neg); + + int sid_bit = get_bv_sid(1); + int nid_zero = get_sig_nid(Const(0, width)); + int nid_b_ltz = next_nid++; + btorf("%d slt %d %d %d\n", nid_b_ltz, sid_bit, nid_b, nid_zero); + + nid = next_nid++; + btorf("%d ite %d %d %d %d%s\n", nid, sid, nid_b_ltz, nid_l, nid_r, getinfo(cell).c_str()); + } + else + { + nid = next_nid++; + btorf("%d %s %d %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, nid_b, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + if (GetSize(sig) < width) + { + int sid = get_bv_sid(GetSize(sig)); + int nid2 = next_nid++; + btorf("%d slice %d %d %d 0\n", nid2, sid, nid, GetSize(sig) - 1); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($div), ID($mod), ID($modfloor))) + { + bool a_signed = cell->hasParam(ID::A_SIGNED) ? cell->getParam(ID::A_SIGNED).as_bool() : false; + bool b_signed = cell->hasParam(ID::B_SIGNED) ? cell->getParam(ID::B_SIGNED).as_bool() : false; + + string btor_op; + if (cell->type == ID($div)) + btor_op = "div"; + // "rem" = truncating modulo + if (cell->type == ID($mod)) + btor_op = "rem"; + // "mod" = flooring modulo + if (cell->type == ID($modfloor)) + { + // "umod" doesn't exist because it's the same as "urem" + btor_op = a_signed || b_signed ? "mod" : "rem"; + } + log_assert(!btor_op.empty()); + + int width = GetSize(cell->getPort(ID::Y)); + width = std::max(width, GetSize(cell->getPort(ID::A))); + width = std::max(width, GetSize(cell->getPort(ID::B))); + + int nid_a = get_sig_nid(cell->getPort(ID::A), width, a_signed); + int nid_b = get_sig_nid(cell->getPort(ID::B), width, b_signed); + + int sid = get_bv_sid(width); + int nid = next_nid++; + btorf("%d %c%s %d %d %d%s\n", nid, a_signed || b_signed ? 's' : 'u', btor_op.c_str(), sid, nid_a, nid_b, + getinfo(cell).c_str()); + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + if (GetSize(sig) < width) + { + int sid = get_bv_sid(GetSize(sig)); + int nid2 = next_nid++; + btorf("%d slice %d %d %d 0\n", nid2, sid, nid, GetSize(sig) - 1); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($_ANDNOT_), ID($_ORNOT_))) + { + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A)); + int nid_b = get_sig_nid(cell->getPort(ID::B)); + + int nid1 = next_nid++; + int nid2 = next_nid++; + + if (cell->type == ID($_ANDNOT_)) + { + btorf("%d not %d %d\n", nid1, sid, nid_b); + btorf("%d and %d %d %d%s\n", nid2, sid, nid_a, nid1, getinfo(cell).c_str()); + } + + if (cell->type == ID($_ORNOT_)) + { + btorf("%d not %d %d\n", nid1, sid, nid_b); + btorf("%d or %d %d %d%s\n", nid2, sid, nid_a, nid1, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + add_nid_sig(nid2, sig); + goto okay; + } + + if (cell->type.in(ID($_OAI3_), ID($_AOI3_))) + { + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A)); + int nid_b = get_sig_nid(cell->getPort(ID::B)); + int nid_c = get_sig_nid(cell->getPort(ID::C)); + + int nid1 = next_nid++; + int nid2 = next_nid++; + int nid3 = next_nid++; + + if (cell->type == ID($_OAI3_)) + { + btorf("%d or %d %d %d\n", nid1, sid, nid_a, nid_b); + btorf("%d and %d %d %d\n", nid2, sid, nid1, nid_c); + btorf("%d not %d %d%s\n", nid3, sid, nid2, getinfo(cell).c_str()); + } + + if (cell->type == ID($_AOI3_)) + { + btorf("%d and %d %d %d\n", nid1, sid, nid_a, nid_b); + btorf("%d or %d %d %d\n", nid2, sid, nid1, nid_c); + btorf("%d not %d %d%s\n", nid3, sid, nid2, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + add_nid_sig(nid3, sig); + goto okay; + } + + if (cell->type.in(ID($_OAI4_), ID($_AOI4_))) + { + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A)); + int nid_b = get_sig_nid(cell->getPort(ID::B)); + int nid_c = get_sig_nid(cell->getPort(ID::C)); + int nid_d = get_sig_nid(cell->getPort(ID::D)); + + int nid1 = next_nid++; + int nid2 = next_nid++; + int nid3 = next_nid++; + int nid4 = next_nid++; + + if (cell->type == ID($_OAI4_)) + { + btorf("%d or %d %d %d\n", nid1, sid, nid_a, nid_b); + btorf("%d or %d %d %d\n", nid2, sid, nid_c, nid_d); + btorf("%d and %d %d %d\n", nid3, sid, nid1, nid2); + btorf("%d not %d %d%s\n", nid4, sid, nid3, getinfo(cell).c_str()); + } + + if (cell->type == ID($_AOI4_)) + { + btorf("%d and %d %d %d\n", nid1, sid, nid_a, nid_b); + btorf("%d and %d %d %d\n", nid2, sid, nid_c, nid_d); + btorf("%d or %d %d %d\n", nid3, sid, nid1, nid2); + btorf("%d not %d %d%s\n", nid4, sid, nid3, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + add_nid_sig(nid4, sig); + goto okay; + } + + if (cell->type.in(ID($lt), ID($le), ID($eq), ID($eqx), ID($ne), ID($nex), ID($ge), ID($gt))) + { + string btor_op; + if (cell->type == ID($lt)) + btor_op = "lt"; + if (cell->type == ID($le)) + btor_op = "lte"; + if (cell->type.in(ID($eq), ID($eqx))) + btor_op = "eq"; + if (cell->type.in(ID($ne), ID($nex))) + btor_op = "neq"; + if (cell->type == ID($ge)) + btor_op = "gte"; + if (cell->type == ID($gt)) + btor_op = "gt"; + log_assert(!btor_op.empty()); + + int width = 1; + width = std::max(width, GetSize(cell->getPort(ID::A))); + width = std::max(width, GetSize(cell->getPort(ID::B))); + + bool a_signed = cell->hasParam(ID::A_SIGNED) ? cell->getParam(ID::A_SIGNED).as_bool() : false; + bool b_signed = cell->hasParam(ID::B_SIGNED) ? cell->getParam(ID::B_SIGNED).as_bool() : false; + + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A), width, a_signed); + int nid_b = get_sig_nid(cell->getPort(ID::B), width, b_signed); + + int nid = next_nid++; + if (cell->type.in(ID($lt), ID($le), ID($ge), ID($gt))) + { + btorf("%d %c%s %d %d %d%s\n", nid, a_signed || b_signed ? 's' : 'u', btor_op.c_str(), sid, nid_a, nid_b, + getinfo(cell).c_str()); + } + else + { + btorf("%d %s %d %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, nid_b, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + if (GetSize(sig) > 1) + { + int sid = get_bv_sid(GetSize(sig)); + int nid2 = next_nid++; + btorf("%d uext %d %d %d\n", nid2, sid, nid, GetSize(sig) - 1); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($not), ID($neg), ID($_NOT_), ID($pos))) + { + string btor_op; + if (cell->type.in(ID($not), ID($_NOT_))) + btor_op = "not"; + if (cell->type == ID($neg)) + btor_op = "neg"; + + int width = std::max(GetSize(cell->getPort(ID::A)), GetSize(cell->getPort(ID::Y))); + + bool a_signed = cell->hasParam(ID::A_SIGNED) ? cell->getParam(ID::A_SIGNED).as_bool() : false; + int nid_a = get_sig_nid(cell->getPort(ID::A), width, a_signed); + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + // the $pos cell just passes through, all other cells need an actual operation applied + int nid = nid_a; + if (cell->type != ID($pos)) + { + log_assert(!btor_op.empty()); + int sid = get_bv_sid(width); + nid = next_nid++; + btorf("%d %s %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, getinfo(cell).c_str()); + } + + if (GetSize(sig) < width) + { + int sid = get_bv_sid(GetSize(sig)); + int nid2 = next_nid++; + btorf("%d slice %d %d %d 0\n", nid2, sid, nid, GetSize(sig) - 1); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($logic_and), ID($logic_or), ID($logic_not))) + { + string btor_op; + if (cell->type == ID($logic_and)) + btor_op = "and"; + if (cell->type == ID($logic_or)) + btor_op = "or"; + if (cell->type == ID($logic_not)) + btor_op = "not"; + log_assert(!btor_op.empty()); + + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A)); + int nid_b = btor_op != "not" ? get_sig_nid(cell->getPort(ID::B)) : 0; + + if (GetSize(cell->getPort(ID::A)) > 1) + { + int nid_red_a = next_nid++; + btorf("%d redor %d %d\n", nid_red_a, sid, nid_a); + nid_a = nid_red_a; + } + + if (btor_op != "not" && GetSize(cell->getPort(ID::B)) > 1) + { + int nid_red_b = next_nid++; + btorf("%d redor %d %d\n", nid_red_b, sid, nid_b); + nid_b = nid_red_b; + } + + int nid = next_nid++; + if (btor_op != "not") + btorf("%d %s %d %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, nid_b, getinfo(cell).c_str()); + else + btorf("%d %s %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, getinfo(cell).c_str()); + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + if (GetSize(sig) > 1) + { + int sid = get_bv_sid(GetSize(sig)); + int zeros_nid = get_sig_nid(Const(0, GetSize(sig) - 1)); + int nid2 = next_nid++; + btorf("%d concat %d %d %d\n", nid2, sid, zeros_nid, nid); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($reduce_and), ID($reduce_or), ID($reduce_bool), ID($reduce_xor), ID($reduce_xnor))) + { + string btor_op; + if (cell->type == ID($reduce_and)) + btor_op = "redand"; + if (cell->type.in(ID($reduce_or), ID($reduce_bool))) + btor_op = "redor"; + if (cell->type.in(ID($reduce_xor), ID($reduce_xnor))) + btor_op = "redxor"; + log_assert(!btor_op.empty()); + + int sid = get_bv_sid(1); + int nid_a = get_sig_nid(cell->getPort(ID::A)); + + int nid = next_nid++; + + if (cell->type == ID($reduce_xnor)) + { + int nid2 = next_nid++; + btorf("%d %s %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, getinfo(cell).c_str()); + btorf("%d not %d %d\n", nid2, sid, nid); + nid = nid2; + } + else + { + btorf("%d %s %d %d%s\n", nid, btor_op.c_str(), sid, nid_a, getinfo(cell).c_str()); + } + + SigSpec sig = sigmap(cell->getPort(ID::Y)); + + if (GetSize(sig) > 1) + { + int sid = get_bv_sid(GetSize(sig)); + int zeros_nid = get_sig_nid(Const(0, GetSize(sig) - 1)); + int nid2 = next_nid++; + btorf("%d concat %d %d %d\n", nid2, sid, zeros_nid, nid); + nid = nid2; + } + + add_nid_sig(nid, sig); + goto okay; + } + + if (cell->type.in(ID($mux), ID($_MUX_), ID($_NMUX_))) + { + SigSpec sig_a = sigmap(cell->getPort(ID::A)); + SigSpec sig_b = sigmap(cell->getPort(ID::B)); + SigSpec sig_s = sigmap(cell->getPort(ID::S)); + SigSpec sig_y = sigmap(cell->getPort(ID::Y)); + + int nid_a = get_sig_nid(sig_a); + int nid_b = get_sig_nid(sig_b); + int nid_s = get_sig_nid(sig_s); + + int sid = get_bv_sid(GetSize(sig_y)); + int nid = next_nid++; + + if (cell->type == ID($_NMUX_)) + { + int tmp = nid; + nid = next_nid++; + btorf("%d ite %d %d %d %d\n", tmp, sid, nid_s, nid_b, nid_a); + btorf("%d not %d %d%s\n", nid, sid, tmp, getinfo(cell).c_str()); + } + else + { + btorf("%d ite %d %d %d %d%s\n", nid, sid, nid_s, nid_b, nid_a, getinfo(cell).c_str()); + } + + add_nid_sig(nid, sig_y); + goto okay; + } + + if (cell->type == ID($pmux)) + { + SigSpec sig_a = sigmap(cell->getPort(ID::A)); + SigSpec sig_b = sigmap(cell->getPort(ID::B)); + SigSpec sig_s = sigmap(cell->getPort(ID::S)); + SigSpec sig_y = sigmap(cell->getPort(ID::Y)); + + int width = GetSize(sig_a); + int sid = get_bv_sid(width); + int nid = get_sig_nid(sig_a); + + for (int i = 0; i < GetSize(sig_s); i++) + { + int nid_b = get_sig_nid(sig_b.extract(i * width, width)); + int nid_s = get_sig_nid(sig_s.extract(i)); + int nid2 = next_nid++; + if (i == GetSize(sig_s) - 1) + btorf("%d ite %d %d %d %d%s\n", nid2, sid, nid_s, nid_b, nid, getinfo(cell).c_str()); + else + btorf("%d ite %d %d %d %d\n", nid2, sid, nid_s, nid_b, nid); + nid = nid2; + } + + add_nid_sig(nid, sig_y); + goto okay; + } + + if (cell->type.in(ID($dff), ID($ff), ID($anyinit), ID($_DFF_P_), ID($_DFF_N), ID($_FF_))) + { + SigSpec sig_d = sigmap(cell->getPort(ID::D)); + SigSpec sig_q = sigmap(cell->getPort(ID::Q)); + + if ((!info_filename.empty() || ywmap_json.active()) && cell->type.in(ID($dff), ID($_DFF_P_), ID($_DFF_N_))) + { + SigSpec sig_c = sigmap(cell->getPort(cell->type == ID($dff) ? ID::CLK : ID::C)); + int nid = get_sig_nid(sig_c); + bool negedge = false; + + if (cell->type == ID($_DFF_N_)) + negedge = true; + + if (cell->type == ID($dff) && !cell->getParam(ID::CLK_POLARITY).as_bool()) + negedge = true; + + if (!info_filename.empty()) + info_clocks[nid] |= negedge ? 2 : 1; + + if (ywmap_json.active()) + ywmap_clock_bits[sig_c] |= negedge ? 2 : 1; + } + + IdString symbol; + + if (sig_q.is_wire()) + { + Wire *w = sig_q.as_wire(); + if (w->port_id == 0) + { + statewires.insert(w); + symbol = w->name; + } + } + + Const initval; + for (int i = 0; i < GetSize(sig_q); i++) + if (initbits.count(sig_q[i])) + initval.bits.push_back(initbits.at(sig_q[i]) ? State::S1 : State::S0); + else + initval.bits.push_back(State::Sx); + + int nid_init_val = -1; + + if (!initval.is_fully_undef()) + nid_init_val = get_sig_nid(initval, -1, false, true); + + int sid = get_bv_sid(GetSize(sig_q)); + int nid = next_nid++; + + if (symbol.empty() || (!print_internal_names && symbol[0] == '$')) + btorf("%d state %d\n", nid, sid); + else + btorf("%d state %d %s\n", nid, sid, log_id(symbol)); + + if (cell->get_bool_attribute(ID(clk2fflogic))) + ywmap_state(cell->getPort(ID::D)); // For a clk2fflogic FF the named signal is the D input not the Q output + else + ywmap_state(sig_q); + + if (nid_init_val >= 0) + { + int nid_init = next_nid++; + if (verbose) + btorf("; initval = %s\n", log_signal(initval)); + btorf("%d init %d %d %d\n", nid_init, sid, nid, nid_init_val); + } + + ff_todo.push_back(make_pair(nid, cell)); + add_nid_sig(nid, sig_q); + goto okay; + } + + if (cell->type.in(ID($anyconst), ID($anyseq))) + { + SigSpec sig_y = sigmap(cell->getPort(ID::Y)); + + int sid = get_bv_sid(GetSize(sig_y)); + int nid = next_nid++; + + btorf("%d state %d%s\n", nid, sid, getinfo(cell).c_str()); + + ywmap_state(sig_y); + + if (cell->type == ID($anyconst)) + { + int nid2 = next_nid++; + btorf("%d next %d %d %d\n", nid2, sid, nid, nid); + } + + add_nid_sig(nid, sig_y); + goto okay; + } + + if (cell->type == ID($initstate)) + { + SigSpec sig_y = sigmap(cell->getPort(ID::Y)); + + if (initstate_nid < 0) + { + int sid = get_bv_sid(1); + int one_nid = get_sig_nid(State::S1); + int zero_nid = get_sig_nid(State::S0); + initstate_nid = next_nid++; + btorf("%d state %d%s\n", initstate_nid, sid, getinfo(cell).c_str()); + btorf("%d init %d %d %d\n", next_nid++, sid, initstate_nid, one_nid); + btorf("%d next %d %d %d\n", next_nid++, sid, initstate_nid, zero_nid); + + ywmap_state(sig_y); + } + + add_nid_sig(initstate_nid, sig_y); + goto okay; + } + + if (cell->is_mem_cell()) + { + Mem *mem = mem_cells[cell]; + + int abits = ceil_log2(mem->size); + + bool asyncwr = false; + bool syncwr = false; + + for (auto &port : mem->wr_ports) + { + if (port.clk_enable) + syncwr = true; + else + asyncwr = true; + } + + if (asyncwr && syncwr) + log_error("Memory %s.%s has mixed async/sync write ports.\n", log_id(module), log_id(mem->memid)); + + for (auto &port : mem->rd_ports) + { + if (port.clk_enable) + log_error("Memory %s.%s has sync read ports. Please use memory_nordff to convert them first.\n", + log_id(module), log_id(mem->memid)); + } + + int data_sid = get_bv_sid(mem->width); + int bool_sid = get_bv_sid(1); + int sid = get_mem_sid(abits, mem->width); + + int nid_init_val = -1; + + if (!mem->inits.empty()) + { + Const initdata = mem->get_init_data(); + bool constword = true; + Const firstword = initdata.extract(0, mem->width); + + for (int i = 1; i < mem->size; i++) + { + Const thisword = initdata.extract(i * mem->width, mem->width); + if (thisword != firstword) + { + constword = false; + break; + } + } + + if (constword) + { + if (verbose) + btorf("; initval = %s\n", log_signal(firstword)); + nid_init_val = get_sig_nid(firstword, -1, false, true); + } + else + { + nid_init_val = next_nid++; + btorf("%d state %d\n", nid_init_val, sid); + + ywmap_state(nullptr); + + for (int i = 0; i < mem->size; i++) + { + Const thisword = initdata.extract(i * mem->width, mem->width); + if (thisword.is_fully_undef()) + continue; + Const thisaddr(i, abits); + int nid_thisword = get_sig_nid(thisword, -1, false, true); + int nid_thisaddr = get_sig_nid(thisaddr, -1, false, true); + int last_nid_init_val = nid_init_val; + nid_init_val = next_nid++; + if (verbose) + btorf("; initval[%d] = %s\n", i, log_signal(thisword)); + btorf("%d write %d %d %d %d\n", nid_init_val, sid, last_nid_init_val, nid_thisaddr, nid_thisword); + } + } + } + + int nid = next_nid++; + int nid_head = nid; + + if (mem->memid[0] == '$') + btorf("%d state %d\n", nid, sid); + else + btorf("%d state %d %s\n", nid, sid, log_id(mem->memid)); + + ywmap_state(cell); + + if (nid_init_val >= 0) + { + int nid_init = next_nid++; + btorf("%d init %d %d %d\n", nid_init, sid, nid, nid_init_val); + } + + if (asyncwr) + { + for (auto &port : mem->wr_ports) + { + SigSpec wa = port.addr; + wa.extend_u0(abits); + + int wa_nid = get_sig_nid(wa); + int wd_nid = get_sig_nid(port.data); + int we_nid = get_sig_nid(port.en); + + int nid2 = next_nid++; + btorf("%d read %d %d %d\n", nid2, data_sid, nid_head, wa_nid); + + int nid3 = next_nid++; + btorf("%d not %d %d\n", nid3, data_sid, we_nid); + + int nid4 = next_nid++; + btorf("%d and %d %d %d\n", nid4, data_sid, nid2, nid3); + + int nid5 = next_nid++; + btorf("%d and %d %d %d\n", nid5, data_sid, wd_nid, we_nid); + + int nid6 = next_nid++; + btorf("%d or %d %d %d\n", nid6, data_sid, nid5, nid4); + + int nid7 = next_nid++; + btorf("%d write %d %d %d %d\n", nid7, sid, nid_head, wa_nid, nid6); + + int nid8 = next_nid++; + btorf("%d redor %d %d\n", nid8, bool_sid, we_nid); + + int nid9 = next_nid++; + btorf("%d ite %d %d %d %d\n", nid9, sid, nid8, nid7, nid_head); + + nid_head = nid9; + } + } + + for (auto &port : mem->rd_ports) + { + SigSpec ra = port.addr; + ra.extend_u0(abits); + + int ra_nid = get_sig_nid(ra); + int rd_nid = next_nid++; + + btorf("%d read %d %d %d\n", rd_nid, data_sid, nid_head, ra_nid); + + add_nid_sig(rd_nid, port.data); + } + + if (!asyncwr) + { + mem_todo.push_back(make_pair(nid, mem)); + } + else + { + int nid2 = next_nid++; + btorf("%d next %d %d %d\n", nid2, sid, nid, nid_head); + } + + goto okay; + } + + if (cell->type.in(ID($dffe), ID($sdff), ID($sdffe), ID($sdffce)) || cell->type.str().substr(0, 6) == "$_SDFF" || + (cell->type.str().substr(0, 6) == "$_DFFE" && cell->type.str().size() == 10)) + { + log_error("Unsupported cell type %s for cell %s.%s -- please run `dffunmap` before `write_btor`.\n", log_id(cell->type), + log_id(module), log_id(cell)); + } + if (cell->type.in(ID($adff), ID($adffe), ID($aldff), ID($aldffe), ID($dffsr), ID($dffsre)) || + cell->type.str().substr(0, 5) == "$_DFF" || cell->type.str().substr(0, 7) == "$_ALDFF") + { + log_error( + "Unsupported cell type %s for cell %s.%s -- please run `async2sync; dffunmap` or `clk2fflogic` before `write_btor`.\n", + log_id(cell->type), log_id(module), log_id(cell)); + } + if (cell->type.in(ID($sr), ID($dlatch), ID($adlatch), ID($dlatchsr)) || cell->type.str().substr(0, 8) == "$_DLATCH" || + cell->type.str().substr(0, 5) == "$_SR_") + { + log_error("Unsupported cell type %s for cell %s.%s -- please run `clk2fflogic` before `write_btor`.\n", log_id(cell->type), + log_id(module), log_id(cell)); + } + log_error("Unsupported cell type %s for cell %s.%s.\n", log_id(cell->type), log_id(module), log_id(cell)); + + okay: + btorf_pop(log_id(cell)); + cell_recursion_guard.erase(cell); + } + + int get_sig_nid(SigSpec sig, int to_width = -1, bool is_signed = false, bool is_init = false) + { + int nid = -1; + sigmap.apply(sig); + + for (auto bit : sig) + if (bit == State::Sx) + goto has_undef_bits; + + if (0) + { + has_undef_bits: + SigSpec sig_mask_undef, sig_noundef; + int first_undef = -1; + + for (int i = 0; i < GetSize(sig); i++) + if (sig[i] == State::Sx) + { + if (first_undef < 0) + first_undef = i; + sig_mask_undef.append(State::S1); + sig_noundef.append(State::S0); + } + else + { + sig_mask_undef.append(State::S0); + sig_noundef.append(sig[i]); + } + + if (to_width < 0 || first_undef < to_width) + { + int sid = get_bv_sid(GetSize(sig)); + + int nid_input = next_nid++; + if (is_init) + { + btorf("%d state %d\n", nid_input, sid); + ywmap_state(sig); + } + else + { + btorf("%d input %d\n", nid_input, sid); + ywmap_input(sig); + } + + int nid_masked_input; + if (sig_mask_undef.is_fully_ones()) + { + nid_masked_input = nid_input; + } + else + { + int nid_mask_undef = get_sig_nid(sig_mask_undef); + nid_masked_input = next_nid++; + btorf("%d and %d %d %d\n", nid_masked_input, sid, nid_input, nid_mask_undef); + } + + if (sig_noundef.is_fully_zero()) + { + nid = nid_masked_input; + } + else + { + int nid_noundef = get_sig_nid(sig_noundef); + nid = next_nid++; + btorf("%d or %d %d %d\n", nid, sid, nid_masked_input, nid_noundef); + } + + goto extend_or_trim; + } + + sig = sig_noundef; + } + + if (sig_nid.count(sig) == 0) + { + // , + vector> nidbits; + + // collect all bits + for (int i = 0; i < GetSize(sig); i++) + { + SigBit bit = sig[i]; + + if (bit_nid.count(bit) == 0) + { + if (bit.wire == nullptr) + { + Const c(bit.data); + + while (i + GetSize(c) < GetSize(sig) && sig[i + GetSize(c)].wire == nullptr) + c.bits.push_back(sig[i + GetSize(c)].data); + + if (consts.count(c) == 0) + { + int sid = get_bv_sid(GetSize(c)); + int nid = next_nid++; + btorf("%d const %d %s\n", nid, sid, c.as_string().c_str()); + consts[c] = nid; + nid_width[nid] = GetSize(c); + } + + int nid = consts.at(c); + + for (int j = 0; j < GetSize(c); j++) + nidbits.push_back(make_pair(nid, j)); + + i += GetSize(c) - 1; + continue; + } + else + { + if (bit_cell.count(bit) == 0) + { + SigSpec s = bit; + + while (i + GetSize(s) < GetSize(sig) && sig[i + GetSize(s)].wire != nullptr && + bit_cell.count(sig[i + GetSize(s)]) == 0) + s.append(sig[i + GetSize(s)]); + + log_warning("No driver for signal %s.\n", log_signal(s)); + + int sid = get_bv_sid(GetSize(s)); + int nid = next_nid++; + btorf("%d input %d\n", nid, sid); + ywmap_input(s); + nid_width[nid] = GetSize(s); + + for (int j = 0; j < GetSize(s); j++) + nidbits.push_back(make_pair(nid, j)); + + i += GetSize(s) - 1; + continue; + } + else + { + export_cell(bit_cell.at(bit)); + log_assert(bit_nid.count(bit)); + } + } + } + + nidbits.push_back(bit_nid.at(bit)); + } + + int width = 0; + int nid = -1; + + // group bits and emit slice-concat chain + for (int i = 0; i < GetSize(nidbits); i++) + { + int nid2 = nidbits[i].first; + int lower = nidbits[i].second; + int upper = lower; + + while (i + 1 < GetSize(nidbits) && nidbits[i + 1].first == nidbits[i].first && + nidbits[i + 1].second == nidbits[i].second + 1) + upper++, i++; + + int nid3 = nid2; + + if (lower != 0 || upper + 1 != nid_width.at(nid2)) + { + int sid = get_bv_sid(upper - lower + 1); + nid3 = next_nid++; + btorf("%d slice %d %d %d %d\n", nid3, sid, nid2, upper, lower); + } + + int nid4 = nid3; + + if (nid >= 0) + { + int sid = get_bv_sid(width + upper - lower + 1); + nid4 = next_nid++; + btorf("%d concat %d %d %d\n", nid4, sid, nid3, nid); + } + + width += upper - lower + 1; + nid = nid4; + } + + sig_nid[sig] = nid; + nid_width[nid] = width; + } + + nid = sig_nid.at(sig); + + extend_or_trim: + if (to_width >= 0 && to_width != GetSize(sig)) + { + if (to_width < GetSize(sig)) + { + int sid = get_bv_sid(to_width); + int nid2 = next_nid++; + btorf("%d slice %d %d %d 0\n", nid2, sid, nid, to_width - 1); + nid = nid2; + } + else + { + int sid = get_bv_sid(to_width); + int nid2 = next_nid++; + btorf("%d %s %d %d %d\n", nid2, is_signed ? "sext" : "uext", sid, nid, to_width - GetSize(sig)); + nid = nid2; + } + } + + return nid; + } + + LakeroadWorker(std::ostream &f, RTLIL::Module *module) : f(f), sigmap(module), module(module) {} + + void run() + { + if (module->processes.size() > 0) + log_error("Module has processes.\n"); + + // IDs used to generate let expressions. + int id = 0; + auto get_new_id_str = [&]() + { return stringf("v%d", id++); }; + + // Get signal name. + auto get_signal_name = [&](const SigSpec &sig) + { + auto signal_name = std::string(log_signal(sig)); + assert(signal_name.size() > 0); + if (signal_name[0] == '\\') + signal_name = signal_name.substr(1); + return signal_name; + }; + + // Function to generate a let expression. + auto let = [&](const std::string &id_str, const std::string &expr) + { return stringf("(let %s %s)", id_str.c_str(), expr.c_str()); }; + + // Wire expressions (which we can eventually delete) + vector wire_exprs; + + // Generate wire expression. + auto wire_expr = [&](const std::string &name, const int bitwidth) + { + auto s = stringf("(Wire \"%s\" %d)", name.c_str(), bitwidth); + wire_exprs.push_back(s); + return s; + }; + + std::map signal_let_bound_name; + + // Does not sigmap the signal; you should sigmap the signal if you need it + // sigmapped. + // + // to_width = -1 means no extension. + std::function get_expression_for_signal = [&](const SigSpec &sig, int to_width) + { + // If we've already handled the expression, return it. + if (signal_let_bound_name.count(sig)) + return signal_let_bound_name.at(sig); + + // SigSpecs are either constants, wires, or concatenations and selections + // of wires. We simply need to handle each case. + + // The output expression will go in here. + std::string out_expr; + + if (sig.is_fully_const()) + { + // If the signal is a constant, we can just use the constant. + auto const_str = stringf("(BV %d %d)", Const(sig.as_const()).as_int(), sig.size()); + auto new_id = get_new_id_str(); + auto let_expr = let(new_id, const_str); + auto signal_name = get_signal_name(sig); + f << "; " << signal_name << "\n"; + f << let_expr << "\n"; + out_expr = new_id; + } + else if (sig.is_wire()) + { + // If the signal is a simple wire, we can just use the name of the wire. + auto signal_name = get_signal_name(sig); + auto let_bound_id = get_new_id_str(); + // Generate wire expression + auto let_expr = let(let_bound_id, wire_expr(let_bound_id, GetSize(sig))); + f << "; " << signal_name << "\n"; + f << let_expr << "\n"; + out_expr = let_bound_id; + } + else if (sig.chunks().size() > 1) + { + // If the signal is multiple chunks, we need to generate a concatenation. + + f << "; " << log_signal(sig) << "\n"; + + // Generate expression for each chunk. + std::vector chunk_exprs; + for (auto chunk : sig.chunks()) + { + chunk_exprs.push_back(get_expression_for_signal(chunk, -1)); + } + + // Generate concatenation expressions. + auto concat_expr = chunk_exprs[0]; + for (size_t i = 1; i < chunk_exprs.size(); i++) + { + auto new_id = get_new_id_str(); + auto let_expr = let(new_id, stringf("(Concat %s %s)", concat_expr.c_str(), chunk_exprs[i].c_str())); + f << let_expr << "\n"; + concat_expr = new_id; + } + + out_expr = concat_expr; + } + else if (sig.chunks().size() == 1 && sig.chunks()[0].wire->width != sig.size()) + { + // This branch is meant to capture the case where the signal is a + // slice/extraction. I'm not quite sure how to check this in Yosys + // though. + // + // It may be the case that we've set up these if statements in a way + // that this is the only possible case that's left. That would be nice. + // Currently, the condition in this else if branch is a little messy. + + auto chunk = sig.chunks()[0]; + + if (chunk.wire->upto) + { + // You can copy implementation for upto from verilog_backend.cc. + log_error("Unimplemented case of `upto==true` in %s.\n", log_signal(sig)); + } + + // The let-bound ID string of the expression to extract from. + auto extract_from_expr = get_expression_for_signal(sigmap(sig.chunks()[0].wire), -1); + auto new_id = get_new_id_str(); + auto extract_expr = stringf("(Extract %d %d %s)", (chunk.offset + chunk.width - 1) + chunk.wire->start_offset, + chunk.offset + chunk.wire->start_offset, extract_from_expr.c_str()); + + auto let_expr = let(new_id, extract_expr); + f << let_expr << "\n"; + out_expr = new_id; + } + else + { + log_error("Unhandled case of signal for %s.\n", log_signal(sig)); + } + + // If we need to extend the signal, do so. + if (to_width >= 0 && to_width != GetSize(sig)) + { + if (to_width < GetSize(sig)) + { + auto new_id = get_new_id_str(); + auto extend_expr = stringf("(Extract %d %d %s)", to_width - 1, 0, out_expr.c_str()); + f << let(new_id, extend_expr) << "\n"; + out_expr = new_id; + } + else + { + + auto new_id = get_new_id_str(); + f << "; TODO not handling signedness\n"; + auto extend_expr = stringf("(ZeroExtend %s %d)", out_expr.c_str(), to_width); + f << let(new_id, extend_expr) << "\n"; + out_expr = new_id; + } + } + + signal_let_bound_name.insert({sig, out_expr}); + return out_expr; + }; + + // Create Wire expression for each wire. + // We don't have to do this upfront anymore. Expressions can just be created + // as needed. + f << "; wire declarations\n"; + for (auto wire : module->wires()) + { + get_expression_for_signal(sigmap(wire), -1); + } + + // Handle cells + f << "\n; cells\n"; + for (auto cell : module->cells()) + { + + if (cell->type.in(ID($logic_not))) + { + // Unary ops. + assert(cell->connections().size() == 2); + auto y = sigmap(cell->getPort(ID::Y)); + auto a_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::A)), y.size()); + auto y_let_name = get_expression_for_signal(y, -1); + + std::string op_str; + if (cell->type == ID($logic_not)) + op_str = "(LogicNot)"; + else + log_error("This should be unreachable. You are missing an else if branch.\n"); + + f << stringf("(union %s (Op1 %s %s))\n", y_let_name.c_str(), op_str.c_str(), a_let_name.c_str()).c_str(); + } + else if (cell->type.in(ID($and), ID($or), ID($xor), ID($shr))) + { + // Binary ops that preserve width. + assert(cell->connections().size() == 3); + auto y = sigmap(cell->getPort(ID::Y)); + auto a_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::A)), y.size()); + auto b_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::B)), y.size()); + auto y_let_name = get_expression_for_signal(y, -1); + + std::string op_str; + if (cell->type == ID($and)) + op_str = "(And)"; + else if (cell->type == ID($or)) + op_str = "(Or)"; + else if (cell->type == ID($xor)) + op_str = "(Xor)"; + else if (cell->type == ID($shr)) + op_str = "(Shr)"; + else + log_error("This should be unreachable. You are missing an else if branch.\n"); + + f << stringf("(union %s (Op2 %s %s %s))\n", y_let_name.c_str(), op_str.c_str(), a_let_name.c_str(), + b_let_name.c_str()) + .c_str(); + } + else if (cell->type.in(ID($mux))) + { + assert(cell->connections().size() == 4); + auto y = sigmap(cell->getPort(ID::Y)); + auto a_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::A)), y.size()); + auto b_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::B)), y.size()); + auto s_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::S)), -1); + auto y_let_name = get_expression_for_signal(y, -1); + + f << stringf("(union %s (Op3 (Mux) %s %s %s))\n", y_let_name.c_str(), s_let_name.c_str(), a_let_name.c_str(), + b_let_name.c_str()) + .c_str(); + } + else if (cell->type.in(ID($eq))) + { + // Binary ops that result in one bit. + assert(cell->connections().size() == 3); + auto y = sigmap(cell->getPort(ID::Y)); + auto a = sigmap(cell->getPort(ID::A)); + auto b = sigmap(cell->getPort(ID::B)); + + if (y.size() != 1) + log_error("Expected 1-bit output for cell %s.\n", log_id(cell)); + + // Extend the inputs to the same width. + int to_width = std::max(a.size(), b.size()); + auto a_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::A)), to_width); + auto b_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::B)), to_width); + + auto y_let_name = get_expression_for_signal(y, -1); + + std::string op_str; + if (cell->type == ID($eq)) + op_str = "(Eq)"; + else + log_error("This should be unreachable. You are missing an else if branch.\n"); + + f << stringf("(union %s (Op2 %s %s %s))\n", y_let_name.c_str(), op_str.c_str(), a_let_name.c_str(), + b_let_name.c_str()) + .c_str(); + } + else if (cell->type == ID($dff)) + { + assert(cell->connections().size() == 3); + auto q = sigmap(cell->getPort(ID::Q)); + auto d_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::D)), q.size()); + auto clk_let_name = get_expression_for_signal(sigmap(cell->getPort(ID::CLK)), -1); + auto q_let_name = get_expression_for_signal(q, -1); + + f << "; TODO: assuming 0 default for Reg\n"; + f << stringf("(union %s (Reg 0 %s %s))\n", q_let_name.c_str(), clk_let_name.c_str(), d_let_name.c_str()); + } + else if (cell->type == ID($pmux)) + { + // Don't support $pmux: require them to run pmuxtree instead. + log_error("Unsupported cell type %s for cell %s.%s -- please run `pmuxtree` before `write_lakeroad`.\n", + log_id(cell->type), log_id(module), log_id(cell)); + } + else + { + log_error("Unimplemented cell type %s for cell %s.%s.\n", log_id(cell->type), log_id(module), log_id(cell)); + } + } + + // For each input, generate Var expression and a let binding. + f << "\n; inputs\n"; + for (auto wire : module->wires()) + { + if (!wire->port_id || !wire->port_input) + continue; + + auto sigspec = sigmap(wire); + auto signal_name = get_signal_name(sigspec); + + assert(signal_let_bound_name.count(sigspec)); + auto let_bound_id = signal_let_bound_name.at(sigspec); + + f << stringf("(let %s (Var \"%s\" %d))\n", signal_name.c_str(), signal_name.c_str(), GetSize(sigspec)).c_str(); + f << stringf("(union %s %s)\n", let_bound_id.c_str(), signal_name.c_str()).c_str(); + } + + // For each output, generate a let binding. + f << "\n; outputs\n"; + for (auto wire : module->wires()) + { + if (!wire->port_id || !wire->port_output) + continue; + + // The name before sigmapping should be the output port name, i hope! + auto signal_name_pre_sigmap = get_signal_name(wire); + + // The name after sigmapping will help us figure out which let ID holds the value of this signal. + auto sigspec = sigmap(wire); + auto signal_name = get_signal_name(sigspec); + assert(signal_let_bound_name.count(sigspec)); + auto let_bound_id = signal_let_bound_name.at(sigspec); + + f << stringf("(let %s %s)\n", signal_name_pre_sigmap.c_str(), let_bound_id.c_str()).c_str(); + } + + // Delete Wire expressions. + f << "\n; delete wire expressions\n"; + for (auto wire_expr : wire_exprs) + { + f << stringf("(delete %s)\n", wire_expr.c_str()).c_str(); + } + + return; + for (auto wire : module->wires()) + { + // Some kind of initialization thing? + // if (wire->attributes.count(ID::init)) { + // Const attrval = wire->attributes.at(ID::init); + // for (int i = 0; i < GetSize(wire) && i < GetSize(attrval); i++) + // if (attrval[i] == State::S0 || attrval[i] == State::S1) + // initbits[sigmap(SigBit(wire, i))] = (attrval[i] == State::S1); + // } + + // Skip if not an input? + if (!wire->port_id || !wire->port_input) + continue; + + SigSpec sig = sigmap(wire); + auto signal_name = std::string(log_signal(sig)); + assert(signal_name.size() > 1); + signal_name = signal_name.substr(1); + auto id_str = "v0"; + btorf("(let %s (Var %s %d))\n", id_str, signal_name.c_str(), GetSize(sig)); + + // if (!info_filename.empty()) { + // auto gclk_attr = wire->attributes.find(ID::replaced_by_gclk); + // if (gclk_attr != wire->attributes.end()) { + // if (gclk_attr->second == State::S1) + // info_clocks[nid] |= 1; + // else if (gclk_attr->second == State::S0) + // info_clocks[nid] |= 2; + // } + // } + + // if (ywmap_json.active()) { + // for (int i = 0; i < GetSize(sig); i++) { + // auto input_bit = SigBit(wire, i); + // auto bit = sigmap(input_bit); + // if (!ywmap_clock_bits.count(bit)) + // continue; + // ywmap_clock_inputs[input_bit] = ywmap_clock_bits[bit]; + // } + // } + } + + // btorf_pop("inputs"); + + // for (auto cell : module->cells()) + // for (auto &conn : cell->connections()) { + // if (!cell->output(conn.first)) + // continue; + + // for (auto bit : sigmap(conn.second)) + // bit_cell[bit] = cell; + // } + + // for (auto wire : module->wires()) { + // if (!wire->port_id || !wire->port_output) + // continue; + + // btorf_push(stringf("output %s", log_id(wire))); + + // int nid = get_sig_nid(wire); + // btorf("%d output %d%s\n", next_nid++, nid, getinfo(wire).c_str()); + + // btorf_pop(stringf("output %s", log_id(wire))); + // } + + // for (auto wire : module->wires()) { + // if (wire->port_id || wire->name[0] == '$') + // continue; + + // btorf_push(stringf("wire %s", log_id(wire))); + + // int sid = get_bv_sid(GetSize(wire)); + // int nid = get_sig_nid(sigmap(wire)); + + // if (statewires.count(wire)) + // continue; + + // int this_nid = next_nid++; + // btorf("%d uext %d %d %d%s\n", this_nid, sid, nid, 0, getinfo(wire).c_str()); + // if (info_clocks.count(nid)) + // info_clocks[this_nid] |= info_clocks[nid]; + + // btorf_pop(stringf("wire %s", log_id(wire))); + // continue; + // } + + // while (!ff_todo.empty() || !mem_todo.empty()) { + // vector> todo; + // todo.swap(ff_todo); + + // for (auto &it : todo) { + // int nid = it.first; + // Cell *cell = it.second; + + // btorf_push(stringf("next %s", log_id(cell))); + + // SigSpec sig = sigmap(cell->getPort(ID::D)); + // int nid_q = get_sig_nid(sig); + // int sid = get_bv_sid(GetSize(sig)); + // btorf("%d next %d %d %d%s\n", next_nid++, sid, nid, nid_q, getinfo(cell).c_str()); + + // btorf_pop(stringf("next %s", log_id(cell))); + // } + + // vector> mtodo; + // mtodo.swap(mem_todo); + + // for (auto &it : mtodo) { + // int nid = it.first; + // Mem *mem = it.second; + + // btorf_push(stringf("next %s", log_id(mem->memid))); + + // int abits = ceil_log2(mem->size); + + // int data_sid = get_bv_sid(mem->width); + // int bool_sid = get_bv_sid(1); + // int sid = get_mem_sid(abits, mem->width); + // int nid_head = nid; + + // for (auto &port : mem->wr_ports) { + // SigSpec wa = port.addr; + // wa.extend_u0(abits); + + // int wa_nid = get_sig_nid(wa); + // int wd_nid = get_sig_nid(port.data); + // int we_nid = get_sig_nid(port.en); + + // int nid2 = next_nid++; + // btorf("%d read %d %d %d\n", nid2, data_sid, nid_head, wa_nid); + + // int nid3 = next_nid++; + // btorf("%d not %d %d\n", nid3, data_sid, we_nid); + + // int nid4 = next_nid++; + // btorf("%d and %d %d %d\n", nid4, data_sid, nid2, nid3); + + // int nid5 = next_nid++; + // btorf("%d and %d %d %d\n", nid5, data_sid, wd_nid, we_nid); + + // int nid6 = next_nid++; + // btorf("%d or %d %d %d\n", nid6, data_sid, nid5, nid4); + + // int nid7 = next_nid++; + // btorf("%d write %d %d %d %d\n", nid7, sid, nid_head, wa_nid, nid6); + + // int nid8 = next_nid++; + // btorf("%d redor %d %d\n", nid8, bool_sid, we_nid); + + // int nid9 = next_nid++; + // btorf("%d ite %d %d %d %d\n", nid9, sid, nid8, nid7, nid_head); + + // nid_head = nid9; + // } + + // int nid2 = next_nid++; + // btorf("%d next %d %d %d%s\n", nid2, sid, nid, nid_head, (mem->cell ? getinfo(mem->cell) : + // getinfo(mem->mem)).c_str()); + + // btorf_pop(stringf("next %s", log_id(mem->memid))); + // } + // } + + // while (!bad_properties.empty()) { + // vector todo; + // bad_properties.swap(todo); + + // int sid = get_bv_sid(1); + // int cursor = 0; + + // while (cursor + 1 < GetSize(todo)) { + // int nid_a = todo[cursor++]; + // int nid_b = todo[cursor++]; + // int nid = next_nid++; + + // bad_properties.push_back(nid); + // btorf("%d or %d %d %d\n", nid, sid, nid_a, nid_b); + // } + + // if (!bad_properties.empty()) { + // if (cursor < GetSize(todo)) + // bad_properties.push_back(todo[cursor++]); + // log_assert(cursor == GetSize(todo)); + // } else { + // int nid = next_nid++; + // log_assert(cursor == 0); + // log_assert(GetSize(todo) == 1); + // btorf("%d bad %d\n", nid, todo[cursor]); + // } + // } + + // if (!info_filename.empty()) { + // for (auto &it : info_clocks) { + // switch (it.second) { + // case 1: + // infof("posedge %d\n", it.first); + // break; + // case 2: + // infof("negedge %d\n", it.first); + // break; + // case 3: + // infof("event %d\n", it.first); + // break; + // default: + // log_abort(); + // } + // } + + // std::ofstream f; + // f.open(info_filename.c_str(), std::ofstream::trunc); + // if (f.fail()) + // log_error("Can't open file `%s' for writing: %s\n", info_filename.c_str(), strerror(errno)); + // for (auto &it : info_lines) + // f << it; + // f.close(); + // } + + // if (ywmap_json.active()) { + // ywmap_json.begin_object(); + // ywmap_json.entry("version", "Yosys Witness BTOR map"); + // ywmap_json.entry("generator", yosys_version_str); + + // ywmap_json.name("clocks"); + // ywmap_json.begin_array(); + // for (auto &entry : ywmap_clock_inputs) { + // if (entry.second != 1 && entry.second != 2) + // continue; + // log_assert(entry.first.is_wire()); + // ywmap_json.begin_object(); + // ywmap_json.compact(); + // ywmap_json.entry("path", witness_path(entry.first.wire)); + // ywmap_json.entry("offset", entry.first.offset); + // ywmap_json.entry("edge", entry.second == 1 ? "posedge" : "negedge"); + // ywmap_json.end_object(); + // } + // ywmap_json.end_array(); + + // ywmap_json.name("inputs"); + // ywmap_json.begin_array(); + // for (auto &entry : ywmap_inputs) + // emit_ywmap_btor_sig(entry); + // ywmap_json.end_array(); + + // ywmap_json.name("states"); + // ywmap_json.begin_array(); + // for (auto &entry : ywmap_states) + // emit_ywmap_btor_sig(entry); + // ywmap_json.end_array(); + + // ywmap_json.end_object(); + // } + } +}; + +struct BtorBackend : public Backend +{ + BtorBackend() : Backend("lakeroad", "write design to egglog Lakeroad IR") {} + void help() override + { + // |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---| + // log("\n"); + // log(" write_btor [options] [filename]\n"); + // log("\n"); + // log("Write a BTOR description of the current design.\n"); + // log("\n"); + // log(" -v\n"); + // log(" Add comments and indentation to BTOR output file\n"); + // log("\n"); + // log(" -s\n"); + // log(" Output only a single bad property for all asserts\n"); + // log("\n"); + // log(" -c\n"); + // log(" Output cover properties using 'bad' statements instead of asserts\n"); + // log("\n"); + // log(" -i \n"); + // log(" Create additional info file with auxiliary information\n"); + // log("\n"); + // log(" -x\n"); + // log(" Output symbols for internal netnames (starting with '$')\n"); + // log("\n"); + // log(" -ywmap \n"); + // log(" Create a map file for conversion to and from Yosys witness traces\n"); + // log("\n"); + } + void execute(std::ostream *&f, std::string filename, std::vector args, RTLIL::Design *design) override + { + log_header(design, "Executing Lakeroad egglog backend.\n"); + + RTLIL::Module *topmod = design->top_module(); + + // Has to come after other arg parsing. + extra_args(f, filename, args, args.size()); + + if (topmod == nullptr) + log_cmd_error("No top module found.\n"); + + LakeroadWorker(*f, topmod).run(); + + // *f << stringf("; end of yosys output\n"); + } +} BtorBackend; + +PRIVATE_NAMESPACE_END diff --git a/yosys-plugin/run_tests.sh b/yosys-plugin/run_tests.sh new file mode 100755 index 0000000..4698f68 --- /dev/null +++ b/yosys-plugin/run_tests.sh @@ -0,0 +1,8 @@ +#!/bin/bash +# Each test file should have a corresponding .expect file. The .expect file +# should contain the expected output of the Lakeroad Yosys backend for the +# given test file, minus any comment lines or blank lines. + +SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) + +lit -v "$SCRIPT_DIR"/tests diff --git a/yosys-plugin/tests/example.ys b/yosys-plugin/tests/example.ys new file mode 100644 index 0000000..4f44cea --- /dev/null +++ b/yosys-plugin/tests/example.ys @@ -0,0 +1,25 @@ +# RUN: $YOSYS -m $CHURCHROAD_DIR/yosys-plugin/churchroad.so %s \ +# RUN: | FileCheck %s + + +read_verilog <