From f6e16e7f4c65440783047d3f2b03563f36f6dae8 Mon Sep 17 00:00:00 2001 From: whitequark Date: Wed, 27 May 2020 00:21:15 +0000 Subject: cxxrtl: generate debug information for non-localized public wires. Debug information describes values, wires, and memories with a simple C-compatible layout. It can be emitted on demand into a map, which has no runtime cost when it is unused, and allows late bound designs. The `hdlname` attribute is used as the lookup key such that original names, as emitted by the frontend, can be used for debugging and introspection. --- backends/cxxrtl/cxxrtl.h | 46 +++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 45 insertions(+), 1 deletion(-) (limited to 'backends/cxxrtl/cxxrtl.h') diff --git a/backends/cxxrtl/cxxrtl.h b/backends/cxxrtl/cxxrtl.h index 7b91742e0..14613afb0 100644 --- a/backends/cxxrtl/cxxrtl.h +++ b/backends/cxxrtl/cxxrtl.h @@ -49,6 +49,8 @@ namespace cxxrtl { // invisible to the compiler, (b) we often operate on non-power-of-2 values and have to clear the high bits anyway. // Therefore, using relatively wide chunks and clearing the high bits explicitly and only when we know they may be // clobbered results in simpler generated code. +typedef uint32_t chunk_t; + template struct chunk_traits { static_assert(std::is_integral::value && std::is_unsigned::value, @@ -65,7 +67,7 @@ template struct value : public expr_base> { static constexpr size_t bits = Bits; - using chunk = chunk_traits; + using chunk = chunk_traits; static constexpr chunk::type msb_mask = (Bits % chunk::bits == 0) ? chunk::mask : chunk::mask >> (chunk::bits - (Bits % chunk::bits)); @@ -712,6 +714,46 @@ struct metadata { typedef std::map metadata_map; +// This structure is intended for consumption via foreign function interfaces, like Python's ctypes. +// Because of this it uses a C-style layout that is easy to parse rather than more idiomatic C++. +struct debug_item { + enum : uint32_t { + VALUE = 0, + WIRE = 1, + MEMORY = 2, + } type; + + size_t width; // in bits + size_t depth; // 1 if `type != MEMORY` + chunk_t *curr; + chunk_t *next; // nullptr if `type == VALUE || type == MEMORY` + + template + debug_item(value &item) : type(VALUE), width(Bits), depth(1), + curr(item.data), next(nullptr) { + static_assert(sizeof(item) == value::chunks * sizeof(chunk_t), + "value is not compatible with C layout"); + } + + template + debug_item(wire &item) : type(WIRE), width(Bits), depth(1), + curr(item.curr.data), next(item.next.data) { + static_assert(sizeof(item.curr) == value::chunks * sizeof(chunk_t) && + sizeof(item.next) == value::chunks * sizeof(chunk_t), + "wire is not compatible with C layout"); + } + + template + debug_item(memory &item) : type(MEMORY), width(Width), depth(item.data.size()), + curr(item.data.empty() ? nullptr : item.data[0].data), next(nullptr) { + static_assert(sizeof(item.data[0]) == value::chunks * sizeof(chunk_t), + "memory is not compatible with C layout"); + } +}; +static_assert(std::is_standard_layout::value, "debug_item is not compatible with C layout"); + +typedef std::map debug_items; + struct module { module() {} virtual ~module() {} @@ -731,6 +773,8 @@ struct module { } while (commit() && !converged); return deltas; } + + virtual void debug_info(debug_items &items, std::string path = "") {} }; } // namespace cxxrtl -- cgit v1.2.3 From c399359ed6ea8f1d380a88779664f95f9c2e58a9 Mon Sep 17 00:00:00 2001 From: whitequark Date: Fri, 5 Jun 2020 13:52:30 +0000 Subject: cxxrtl: add a C API for driving and introspecting designs. Compared to the C++ API, the C API currently has two limitations: 1. Memories cannot be updated in a race-free way. 2. Black boxes cannot be implemented in C. --- backends/cxxrtl/cxxrtl.h | 75 +++++++++++++++++++++++++++++------------------- 1 file changed, 46 insertions(+), 29 deletions(-) (limited to 'backends/cxxrtl/cxxrtl.h') diff --git a/backends/cxxrtl/cxxrtl.h b/backends/cxxrtl/cxxrtl.h index 14613afb0..aba2c77a1 100644 --- a/backends/cxxrtl/cxxrtl.h +++ b/backends/cxxrtl/cxxrtl.h @@ -33,13 +33,15 @@ #include #include -// The cxxrtl support library implements compile time specialized arbitrary width arithmetics, as well as provides +#include + +// The CXXRTL support library implements compile time specialized arbitrary width arithmetics, as well as provides // composite lvalues made out of bit slices and concatenations of lvalues. This allows the `write_cxxrtl` pass // to perform a straightforward translation of RTLIL structures to readable C++, relying on the C++ compiler // to unwrap the abstraction and generate efficient code. namespace cxxrtl { -// All arbitrary-width values in cxxrtl are backed by arrays of unsigned integers called chunks. The chunk size +// All arbitrary-width values in CXXRTL are backed by arrays of unsigned integers called chunks. The chunk size // is the same regardless of the value width to simplify manipulating values via FFI interfaces, e.g. driving // and introspecting the simulation in Python. // @@ -716,39 +718,49 @@ typedef std::map metadata_map; // This structure is intended for consumption via foreign function interfaces, like Python's ctypes. // Because of this it uses a C-style layout that is easy to parse rather than more idiomatic C++. -struct debug_item { +// +// To avoid violating strict aliasing rules, this structure has to be a subclass of the one used +// in the C API, or it would not be possible to cast between the pointers to these. +struct debug_item : ::cxxrtl_object { enum : uint32_t { - VALUE = 0, - WIRE = 1, - MEMORY = 2, - } type; - - size_t width; // in bits - size_t depth; // 1 if `type != MEMORY` - chunk_t *curr; - chunk_t *next; // nullptr if `type == VALUE || type == MEMORY` + VALUE = CXXRTL_VALUE, + WIRE = CXXRTL_WIRE, + MEMORY = CXXRTL_MEMORY, + }; template - debug_item(value &item) : type(VALUE), width(Bits), depth(1), - curr(item.data), next(nullptr) { - static_assert(sizeof(item) == value::chunks * sizeof(chunk_t), - "value is not compatible with C layout"); - } + debug_item(value &item) { + static_assert(sizeof(item) == value::chunks * sizeof(chunk_t), + "value is not compatible with C layout"); + type = VALUE; + width = Bits; + depth = 1; + curr = item.data; + next = item.data; + } template - debug_item(wire &item) : type(WIRE), width(Bits), depth(1), - curr(item.curr.data), next(item.next.data) { - static_assert(sizeof(item.curr) == value::chunks * sizeof(chunk_t) && - sizeof(item.next) == value::chunks * sizeof(chunk_t), - "wire is not compatible with C layout"); - } + debug_item(wire &item) { + static_assert(sizeof(item.curr) == value::chunks * sizeof(chunk_t) && + sizeof(item.next) == value::chunks * sizeof(chunk_t), + "wire is not compatible with C layout"); + type = WIRE; + width = Bits; + depth = 1; + curr = item.curr.data; + next = item.next.data; + } template - debug_item(memory &item) : type(MEMORY), width(Width), depth(item.data.size()), - curr(item.data.empty() ? nullptr : item.data[0].data), next(nullptr) { - static_assert(sizeof(item.data[0]) == value::chunks * sizeof(chunk_t), - "memory is not compatible with C layout"); - } + debug_item(memory &item) { + static_assert(sizeof(item.data[0]) == value::chunks * sizeof(chunk_t), + "memory is not compatible with C layout"); + type = MEMORY; + width = Width; + depth = item.data.size(); + curr = item.data.empty() ? nullptr : item.data[0].data; + next = nullptr; + } }; static_assert(std::is_standard_layout::value, "debug_item is not compatible with C layout"); @@ -779,7 +791,12 @@ struct module { } // namespace cxxrtl -// Definitions of internal Yosys cells. Other than the functions in this namespace, cxxrtl is fully generic +// Internal structure used to communicate with the implementation of the C interface. +typedef struct _cxxrtl_toplevel { + std::unique_ptr module; +} *cxxrtl_toplevel; + +// Definitions of internal Yosys cells. Other than the functions in this namespace, CXXRTL is fully generic // and indepenent of Yosys implementation details. // // The `write_cxxrtl` pass translates internal cells (cells with names that start with `$`) to calls of these -- cgit v1.2.3