yosys/kernel/drivertools.h

1389 lines
33 KiB
C++

/*
* yosys -- Yosys Open SYnthesis Suite
*
* Copyright (C) 2024 Jannis Harder <jix@yosyshq.com> <me@jix.one>
*
* 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.
*
*/
#ifndef DRIVERTOOLS_H
#define DRIVERTOOLS_H
#include <type_traits>
#include "kernel/rtlil.h"
#include "kernel/sigtools.h"
#include "kernel/celltypes.h"
YOSYS_NAMESPACE_BEGIN
// TODO move implementation into a .cc file
struct DriveBit;
struct DriveChunkWire;
struct DriveChunkPort;
struct DriveChunkMarker;
struct DriveChunk;
struct DriveSpec;
const char *log_signal(DriveChunkWire const &chunk);
const char *log_signal(DriveChunkPort const &chunk);
const char *log_signal(DriveChunkMarker const &chunk);
const char *log_signal(DriveChunk const &chunk);
const char *log_signal(DriveSpec const &chunk);
enum class DriveType : unsigned char
{
NONE,
CONSTANT,
WIRE,
PORT,
MULTIPLE,
MARKER,
};
struct DriveBitWire
{
Wire *wire;
int offset;
DriveBitWire(Wire *wire, int offset) : wire(wire), offset(offset) {}
bool operator==(const DriveBitWire &other) const
{
return wire == other.wire && offset == other.offset;
}
bool operator<(const DriveBitWire &other) const
{
if (wire != other.wire)
return wire->name < other.wire->name;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
operator SigBit() const
{
return SigBit(wire, offset);
}
};
struct DriveBitPort
{
Cell *cell;
IdString port;
int offset;
DriveBitPort(Cell *cell, IdString port, int offset) : cell(cell), port(port), offset(offset) {}
bool operator==(const DriveBitPort &other) const
{
return cell == other.cell && port == other.port && offset == other.offset;
}
bool operator<(const DriveBitPort &other) const
{
if (cell != other.cell)
return cell->name < other.cell->name;
if (port != other.port)
return port < other.port;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
};
struct DriveBitMarker
{
int marker;
int offset;
DriveBitMarker(int marker, int offset) : marker(marker), offset(offset) {}
bool operator==(const DriveBitMarker &other) const
{
return marker == other.marker && offset == other.offset;
}
bool operator<(const DriveBitMarker &other) const
{
if (marker != other.marker)
return marker < other.marker;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
};
struct DriveBitMultiple
{
private:
pool<DriveBit> multiple_;
public:
DriveBitMultiple();
DriveBitMultiple(DriveBit const &single);
pool<DriveBit> const &multiple() const { return multiple_; }
void merge(DriveBitMultiple const &other)
{
for (DriveBit const &single : other.multiple_)
merge(single);
}
void merge(DriveBitMultiple &&other)
{
for (DriveBit &single : other.multiple_)
merge(std::move(single));
}
void merge(DriveBit const &single);
void merge(DriveBit &&single);
bool operator==(const DriveBitMultiple &other) const
{
return multiple_ == other.multiple_;
}
Hasher hash_into(Hasher h) const;
};
struct DriveBit
{
private:
DriveType type_ = DriveType::NONE;
union
{
State constant_;
DriveBitWire wire_;
DriveBitPort port_;
DriveBitMarker marker_;
DriveBitMultiple multiple_;
};
public:
DriveBit() {}
DriveBit(SigBit const &bit);
DriveBit(DriveBit const &other) { *this = other; }
DriveBit(DriveBit &&other) { *this = other; }
DriveBit(State constant) { *this = constant; }
DriveBit(DriveBitWire const &wire) { *this = wire; }
DriveBit(DriveBitWire &&wire) { *this = wire; }
DriveBit(DriveBitPort const &port) { *this = port; }
DriveBit(DriveBitPort &&port) { *this = port; }
DriveBit(DriveBitMarker const &marker) { *this = marker; }
DriveBit(DriveBitMarker &&marker) { *this = marker; }
DriveBit(DriveBitMultiple const &multiple) { *this = multiple; }
DriveBit(DriveBitMultiple &&multiple) { *this = multiple; }
~DriveBit() { set_none(); }
void set_none()
{
switch (type_)
{
case DriveType::NONE:
break;
case DriveType::CONSTANT:
break;
case DriveType::WIRE:
wire_.~DriveBitWire();
break;
case DriveType::PORT:
port_.~DriveBitPort();
break;
case DriveType::MARKER:
marker_.~DriveBitMarker();
break;
case DriveType::MULTIPLE:
multiple_.~DriveBitMultiple();
break;
}
type_ = DriveType::NONE;
}
DriveBit &operator=(DriveBit const &other)
{
switch (other.type_)
{
case DriveType::NONE:
set_none();
break;
case DriveType::CONSTANT:
*this = other.constant_;
break;
case DriveType::WIRE:
*this = other.wire_;
break;
case DriveType::PORT:
*this = other.port_;
break;
case DriveType::MARKER:
*this = other.marker_;
break;
case DriveType::MULTIPLE:
*this = other.multiple_;
break;
}
return *this;
}
DriveBit &operator=(DriveBit &&other)
{
switch (other.type_)
{
case DriveType::NONE:
set_none();
break;
case DriveType::CONSTANT:
*this = std::move(other.constant_);
break;
case DriveType::WIRE:
*this = std::move(other.wire_);
break;
case DriveType::PORT:
*this = std::move(other.port_);
break;
case DriveType::MARKER:
*this = std::move(other.marker_);
break;
case DriveType::MULTIPLE:
*this = std::move(other.multiple_);
break;
}
return *this;
}
DriveBit &operator=(State constant)
{
set_none();
constant_ = constant;
type_ = DriveType::CONSTANT;
return *this;
}
DriveBit &operator=(DriveBitWire const &wire)
{
set_none();
new (&wire_) DriveBitWire(wire);
type_ = DriveType::WIRE;
return *this;
}
DriveBit &operator=(DriveBitWire &&wire)
{
set_none();
new (&wire_) DriveBitWire(wire);
type_ = DriveType::WIRE;
return *this;
}
DriveBit &operator=(DriveBitPort const &port)
{
set_none();
new (&port_) DriveBitPort(port);
type_ = DriveType::PORT;
return *this;
}
DriveBit &operator=(DriveBitPort &&port)
{
set_none();
new (&port_) DriveBitPort(port);
type_ = DriveType::PORT;
return *this;
}
DriveBit &operator=(DriveBitMarker const &marker)
{
set_none();
new (&marker_) DriveBitMarker(marker);
type_ = DriveType::MARKER;
return *this;
}
DriveBit &operator=(DriveBitMarker &&marker)
{
set_none();
new (&marker_) DriveBitMarker(marker);
type_ = DriveType::MARKER;
return *this;
}
DriveBit &operator=(DriveBitMultiple const &multiple)
{
set_none();
if (multiple.multiple().empty())
return *this;
new (&multiple_) DriveBitMultiple(multiple);
type_ = DriveType::MULTIPLE;
return *this;
}
DriveBit &operator=(DriveBitMultiple &&multiple)
{
set_none();
if (multiple.multiple().empty())
return *this;
new (&multiple_) DriveBitMultiple(multiple);
type_ = DriveType::MULTIPLE;
return *this;
}
Hasher hash_into(Hasher h) const;
bool operator==(const DriveBit &other) const
{
if (type_ != other.type_)
return false;
switch (type_)
{
case DriveType::NONE:
return true;
case DriveType::CONSTANT:
return constant_ == other.constant_;
case DriveType::WIRE:
return wire_ == other.wire_;
case DriveType::PORT:
return port_ == other.port_;
case DriveType::MARKER:
return marker_ == other.marker_;
case DriveType::MULTIPLE:
return multiple_ == other.multiple_;
}
log_abort();
}
bool operator!=(const DriveBit &other) const
{
return !(*this == other);
}
bool operator<(const DriveBit &other) const
{
if (type_ != other.type_)
return type_ < other.type_;
switch (type_)
{
case DriveType::NONE:
return false;
case DriveType::CONSTANT:
return constant_ < other.constant_;
case DriveType::WIRE:
return wire_ < other.wire_;
case DriveType::PORT:
return port_ < other.port_;
case DriveType::MARKER:
return marker_ < other.marker_;
case DriveType::MULTIPLE:
log_assert(!"TODO");
}
log_abort();
}
DriveType type() const { return type_; }
bool is_none() const { return type_ == DriveType::NONE; }
bool is_constant() const { return type_ == DriveType::CONSTANT; }
bool is_wire() const { return type_ == DriveType::WIRE; }
bool is_port() const { return type_ == DriveType::PORT; }
bool is_marker() const { return type_ == DriveType::MARKER; }
bool is_multiple() const { return type_ == DriveType::MULTIPLE; }
State &constant() { log_assert(is_constant()); return constant_; }
State const &constant() const { log_assert(is_constant()); return constant_; }
DriveBitWire &wire() { log_assert(is_wire()); return wire_; }
DriveBitWire const &wire() const { log_assert(is_wire()); return wire_; }
DriveBitPort &port() { log_assert(is_port()); return port_; }
DriveBitPort const &port() const { log_assert(is_port()); return port_; }
DriveBitMarker &marker() { log_assert(is_marker()); return marker_; }
DriveBitMarker const &marker() const { log_assert(is_marker()); return marker_; }
DriveBitMultiple &multiple() { log_assert(is_multiple()); return multiple_; }
DriveBitMultiple const &multiple() const { log_assert(is_multiple()); return multiple_; }
void merge(DriveBit const &other);
};
inline DriveBitMultiple::DriveBitMultiple() {}
inline DriveBitMultiple::DriveBitMultiple(DriveBit const &single)
{
multiple_.emplace(single);
}
struct DriveChunkWire
{
Wire *wire;
int offset;
int width;
DriveChunkWire(Wire *wire, int offset, int width) : wire(wire), offset(offset), width(width) {}
DriveChunkWire(DriveBitWire const &bit) : wire(bit.wire), offset(bit.offset), width(1) {}
int size() const { return width; }
DriveBitWire operator[](int i) const
{
log_assert(i >= 0 && i < width);
return DriveBitWire(wire, offset + i);
}
bool can_append(DriveBitWire const &bit) const;
bool try_append(DriveBitWire const &bit);
bool try_append(DriveChunkWire const &chunk);
// Whether this chunk is a whole wire
bool is_whole() const { return offset == 0 && width == wire->width; }
bool operator==(const DriveChunkWire &other) const
{
return wire == other.wire && offset == other.offset && width == other.width;
}
bool operator<(const DriveChunkWire &other) const
{
if (wire != other.wire)
return wire->name < other.wire->name;
if (width != other.width)
return width < other.width;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
explicit operator SigChunk() const
{
return SigChunk(wire, offset, width);
}
};
struct DriveChunkPort
{
Cell *cell;
IdString port;
int offset;
int width;
DriveChunkPort(Cell *cell, IdString port, int offset, int width) :
cell(cell), port(port), offset(offset), width(width) { }
DriveChunkPort(Cell *cell, IdString port) :
cell(cell), port(port), offset(0), width(GetSize(cell->connections().at(port))) { }
DriveChunkPort(Cell *cell, std::pair<IdString, SigSpec> const &conn) :
cell(cell), port(conn.first), offset(0), width(GetSize(conn.second)) { }
DriveChunkPort(DriveBitPort const &bit) :
cell(bit.cell), port(bit.port), offset(bit.offset), width(1) { }
int size() const { return width; }
DriveBitPort operator[](int i) const
{
log_assert(i >= 0 && i < width);
return DriveBitPort(cell, port, offset + i);
}
bool can_append(DriveBitPort const &bit) const;
bool try_append(DriveBitPort const &bit);
bool try_append(DriveChunkPort const &chunk);
// Whether this chunk is a whole port
bool is_whole() const
{
return offset == 0 && width == cell->connections().at(port).size();
}
bool operator==(const DriveChunkPort &other) const
{
return cell == other.cell && port == other.port && offset == other.offset && width == other.width;
}
bool operator<(const DriveChunkPort &other) const
{
if (cell != other.cell)
return cell->name < other.cell->name;
if (port != other.port)
return port < other.port;
if (width != other.width)
return width < other.width;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
};
struct DriveChunkMarker
{
int marker;
int offset;
int width;
DriveChunkMarker(int marker, int offset, int width) :
marker(marker), offset(offset), width(width) {}
DriveChunkMarker(DriveBitMarker const &bit) :
marker(bit.marker), offset(bit.offset), width(1) {}
int size() const { return width; }
DriveBitMarker operator[](int i) const
{
log_assert(i >= 0 && i < width);
return DriveBitMarker(marker, offset + i);
}
bool can_append(DriveBitMarker const &bit) const;
bool try_append(DriveBitMarker const &bit);
bool try_append(DriveChunkMarker const &chunk);
bool operator==(const DriveChunkMarker &other) const
{
return marker == other.marker && offset == other.offset && width == other.width;
}
bool operator<(const DriveChunkMarker &other) const
{
if (marker != other.marker)
return marker < other.marker;
if (width != other.width)
return width < other.width;
return offset < other.offset;
}
Hasher hash_into(Hasher h) const;
};
struct DriveChunkMultiple
{
private:
mutable pool<DriveChunk> multiple_;
int width_;
public:
pool<DriveChunk> const &multiple() const { return multiple_; }
DriveChunkMultiple(DriveBitMultiple const &bit);
int size() const { return width_; }
DriveBitMultiple operator[](int i) const;
bool can_append(DriveBitMultiple const &bit) const;
bool try_append(DriveBitMultiple const &bit);
bool can_append(DriveChunkMultiple const &bit) const;
bool try_append(DriveChunkMultiple const &bit);
bool operator==(const DriveChunkMultiple &other) const
{
return width_ == other.width_ && multiple_ == other.multiple_;
}
bool operator<(const DriveChunkMultiple &other) const
{
if (multiple_.size() < other.multiple_.size())
multiple_.sort();
return false; // TODO implement, canonicalize order
}
Hasher hash_into(Hasher h) const;
};
struct DriveChunk
{
private:
DriveType type_ = DriveType::NONE;
union
{
int none_;
Const constant_;
DriveChunkWire wire_;
DriveChunkPort port_;
DriveChunkMarker marker_;
DriveChunkMultiple multiple_;
};
public:
DriveChunk() { set_none(); }
DriveChunk(DriveChunk const &other) { *this = other; }
DriveChunk(DriveChunk &&other) { *this = other; }
DriveChunk(DriveBit const &other) { *this = other; }
DriveChunk(Const const &constant) { *this = constant; }
DriveChunk(Const &&constant) { *this = constant; }
DriveChunk(DriveChunkWire const &wire) { *this = wire; }
DriveChunk(DriveChunkWire &&wire) { *this = wire; }
DriveChunk(DriveChunkPort const &port) { *this = port; }
DriveChunk(DriveChunkPort &&port) { *this = port; }
DriveChunk(DriveChunkMarker const &marker) { *this = marker; }
DriveChunk(DriveChunkMarker &&marker) { *this = marker; }
DriveChunk(DriveChunkMultiple const &multiple) { *this = multiple; }
DriveChunk(DriveChunkMultiple &&multiple) { *this = multiple; }
~DriveChunk() { set_none(); }
DriveBit operator[](int i) const
{
switch (type_)
{
case DriveType::NONE:
return DriveBit();
case DriveType::CONSTANT:
return constant_[i];
case DriveType::WIRE:
return wire_[i];
case DriveType::PORT:
return port_[i];
case DriveType::MARKER:
return marker_[i];
case DriveType::MULTIPLE:
return multiple_[i];
}
log_abort();
}
void set_none(int width = 0)
{
switch (type_)
{
case DriveType::NONE:
none_ = width;
break;
case DriveType::CONSTANT:
constant_.~Const();
break;
case DriveType::WIRE:
wire_.~DriveChunkWire();
break;
case DriveType::PORT:
port_.~DriveChunkPort();
break;
case DriveType::MARKER:
marker_.~DriveChunkMarker();
break;
case DriveType::MULTIPLE:
multiple_.~DriveChunkMultiple();
break;
}
type_ = DriveType::NONE;
none_ = width;
}
DriveChunk &operator=(DriveChunk const &other)
{
switch (other.type_)
{
case DriveType::NONE:
set_none(other.none_);
break;
case DriveType::CONSTANT:
*this = other.constant_;
break;
case DriveType::WIRE:
*this = other.wire_;
break;
case DriveType::PORT:
*this = other.port_;
break;
case DriveType::MARKER:
*this = other.marker_;
break;
case DriveType::MULTIPLE:
*this = other.multiple_;
break;
}
return *this;
}
DriveChunk &operator=(DriveChunk &&other)
{
switch (other.type_)
{
case DriveType::NONE:
set_none(other.none_);
break;
case DriveType::CONSTANT:
*this = std::move(other.constant_);
break;
case DriveType::WIRE:
*this = std::move(other.wire_);
break;
case DriveType::PORT:
*this = std::move(other.port_);
break;
case DriveType::MARKER:
*this = std::move(other.marker_);
break;
case DriveType::MULTIPLE:
*this = std::move(other.multiple_);
break;
}
return *this;
}
DriveChunk &operator=(Const const &constant)
{
set_none();
new (&constant_) Const(constant);
type_ = DriveType::CONSTANT;
return *this;
}
DriveChunk &operator=(Const &&constant)
{
set_none();
new (&constant_) Const(std::move(constant));
type_ = DriveType::CONSTANT;
return *this;
}
DriveChunk &operator=(DriveChunkWire const &wire)
{
set_none();
new (&wire_) DriveChunkWire(wire);
type_ = DriveType::WIRE;
return *this;
}
DriveChunk &operator=(DriveChunkWire &&wire)
{
set_none();
new (&wire_) DriveChunkWire(wire);
type_ = DriveType::WIRE;
return *this;
}
DriveChunk &operator=(DriveChunkPort const &port)
{
set_none();
new (&port_) DriveChunkPort(port);
type_ = DriveType::PORT;
return *this;
}
DriveChunk &operator=(DriveChunkPort &&port)
{
set_none();
new (&port_) DriveChunkPort(port);
type_ = DriveType::PORT;
return *this;
}
DriveChunk &operator=(DriveChunkMarker const &marker)
{
set_none();
new (&marker_) DriveChunkMarker(marker);
type_ = DriveType::MARKER;
return *this;
}
DriveChunk &operator=(DriveChunkMarker &&marker)
{
set_none();
new (&marker_) DriveChunkMarker(marker);
type_ = DriveType::MARKER;
return *this;
}
DriveChunk &operator=(DriveChunkMultiple const &multiple)
{
set_none(multiple.size());
if (multiple.multiple().empty())
return *this;
new (&multiple_) DriveChunkMultiple(multiple);
type_ = DriveType::MULTIPLE;
return *this;
}
DriveChunk &operator=(DriveChunkMultiple &&multiple)
{
set_none(multiple.size());
if (multiple.multiple().empty())
return *this;
new (&multiple_) DriveChunkMultiple(multiple);
type_ = DriveType::MULTIPLE;
return *this;
}
DriveChunk &operator=(DriveBit const &other)
{
switch (other.type())
{
case DriveType::NONE:
set_none(1);
break;
case DriveType::CONSTANT:
*this = Const(other.constant());
break;
case DriveType::WIRE:
*this = DriveChunkWire(other.wire());
break;
case DriveType::PORT:
*this = DriveChunkPort(other.port());
break;
case DriveType::MARKER:
*this = DriveChunkMarker(other.marker());
break;
case DriveType::MULTIPLE:
*this = DriveChunkMultiple(other.multiple());
break;
}
return *this;
}
bool can_append(DriveBit const &bit) const;
bool try_append(DriveBit const &bit);
bool try_append(DriveChunk const &chunk);
Hasher hash_into(Hasher h) const;
bool operator==(const DriveChunk &other) const
{
if (type_ != other.type_)
return false;
switch (type_)
{
case DriveType::NONE:
return true;
case DriveType::CONSTANT:
return constant_ == other.constant_;
case DriveType::WIRE:
return wire_ == other.wire_;
case DriveType::PORT:
return port_ == other.port_;
case DriveType::MARKER:
return marker_ == other.marker_;
case DriveType::MULTIPLE:
return multiple_ == other.multiple_;
}
log_abort();
}
bool operator!=(const DriveChunk &other) const
{
return !(*this == other);
}
bool operator<(const DriveChunk &other) const
{
if (type_ != other.type_)
return type_ < other.type_;
switch (type_)
{
case DriveType::NONE:
return false;
case DriveType::CONSTANT:
return constant_ < other.constant_;
case DriveType::WIRE:
return wire_ < other.wire_;
case DriveType::PORT:
return port_ < other.port_;
case DriveType::MARKER:
return marker_ < other.marker_;
case DriveType::MULTIPLE:
return multiple_ < other.multiple_;
}
log_abort();
}
DriveType type() const { return type_; }
bool is_none() const { return type_ == DriveType::NONE; }
bool is_constant() const { return type_ == DriveType::CONSTANT; }
bool is_wire() const { return type_ == DriveType::WIRE; }
bool is_port() const { return type_ == DriveType::PORT; }
bool is_marker() const { return type_ == DriveType::MARKER; }
bool is_multiple() const { return type_ == DriveType::MULTIPLE; }
Const &constant() { log_assert(is_constant()); return constant_; }
Const const &constant() const { log_assert(is_constant()); return constant_; }
DriveChunkWire &wire() { log_assert(is_wire()); return wire_; }
DriveChunkWire const &wire() const { log_assert(is_wire()); return wire_; }
DriveChunkPort &port() { log_assert(is_port()); return port_; }
DriveChunkPort const &port() const { log_assert(is_port()); return port_; }
DriveChunkMarker &marker() { log_assert(is_marker()); return marker_; }
DriveChunkMarker const &marker() const { log_assert(is_marker()); return marker_; }
DriveChunkMultiple &multiple() { log_assert(is_multiple()); return multiple_; }
DriveChunkMultiple const &multiple() const { log_assert(is_multiple()); return multiple_; }
int size() const
{
switch (type_)
{
case DriveType::NONE:
return none_;
case DriveType::CONSTANT:
return constant_.size();
case DriveType::WIRE:
return wire_.size();
case DriveType::PORT:
return port_.size();
case DriveType::MARKER:
return marker_.size();
case DriveType::MULTIPLE:
return multiple_.size();
}
log_abort();
}
};
inline DriveChunkMultiple::DriveChunkMultiple(DriveBitMultiple const &bit)
: width_(1)
{
for (auto const &bit : bit.multiple())
multiple_.emplace(bit);
}
struct DriveSpec
{
private:
int width_ = 0;
mutable std::vector<DriveChunk> chunks_;
mutable std::vector<DriveBit> bits_;
mutable unsigned int hash_ = 0;
public:
inline bool packed() const {
return bits_.empty();
}
DriveSpec() {}
DriveSpec(DriveChunk const &chunk) { *this = chunk; }
DriveSpec(DriveChunkWire const &chunk) { *this = chunk; }
DriveSpec(DriveChunkPort const &chunk) { *this = chunk; }
DriveSpec(DriveChunkMarker const &chunk) { *this = chunk; }
DriveSpec(DriveChunkMultiple const &chunk) { *this = chunk; }
DriveSpec(DriveBit const &bit) { *this = bit; }
DriveSpec(DriveBitWire const &bit) { *this = bit; }
DriveSpec(DriveBitPort const &bit) { *this = bit; }
DriveSpec(DriveBitMarker const &bit) { *this = bit; }
DriveSpec(DriveBitMultiple const &bit) { *this = bit; }
DriveSpec(std::vector<DriveChunk> const &chunks) : chunks_(chunks) { compute_width(); }
DriveSpec(std::vector<DriveBit> const &bits)
{
for (auto const &bit : bits)
append(bit);
}
DriveSpec(SigSpec const &sig)
{
// TODO: converting one chunk at a time would be faster
for (auto const &bit : sig.bits())
append(bit);
}
std::vector<DriveChunk> const &chunks() const { pack(); return chunks_; }
std::vector<DriveBit> const &bits() const { unpack(); return bits_; }
int size() const { return width_; }
void append(DriveBit const &bit);
void append(DriveChunk const &chunk);
void pack() const;
void unpack() const;
DriveBit &operator[](int index)
{
log_assert(index >= 0 && index < size());
unpack();
return bits_[index];
}
const DriveBit &operator[](int index) const
{
log_assert(index >= 0 && index < size());
unpack();
return bits_[index];
}
void clear()
{
chunks_.clear();
bits_.clear();
width_ = 0;
}
DriveSpec &operator=(DriveChunk const &chunk)
{
chunks_.clear();
bits_.clear();
append(chunk);
return *this;
}
DriveSpec &operator=(DriveChunkWire const &chunk) { return *this = DriveChunk(chunk); }
DriveSpec &operator=(DriveChunkPort const &chunk) { return *this = DriveChunk(chunk); }
DriveSpec &operator=(DriveChunkMarker const &chunk) { return *this = DriveChunk(chunk); }
DriveSpec &operator=(DriveChunkMultiple const &chunk) { return *this = DriveChunk(chunk); }
DriveSpec &operator=(DriveBit const &bit)
{
chunks_.clear();
bits_.clear();
append(bit);
return *this;
}
DriveSpec &operator=(DriveBitWire const &bit) { return *this = DriveBit(bit); }
DriveSpec &operator=(DriveBitPort const &bit) { return *this = DriveBit(bit); }
DriveSpec &operator=(DriveBitMarker const &bit) { return *this = DriveBit(bit); }
DriveSpec &operator=(DriveBitMultiple const &bit) { return *this = DriveBit(bit); }
void updhash() const {
if (hash_ != 0)
return;
pack();
hash_ = run_hash(chunks_);
hash_ |= (hash_ == 0);
}
Hasher hash_into(Hasher h) const;
bool operator==(DriveSpec const &other) const {
updhash();
other.updhash();
if (size() != other.size() || hash_ != other.hash_)
return false;
return chunks() == other.chunks();
}
private:
void compute_width();
};
struct DriverMap
{
CellTypes celltypes;
DriverMap() { celltypes.setup(); }
DriverMap(Design *design) { celltypes.setup(); celltypes.setup_design(design); }
private:
// Internally we represent all DriveBits by mapping them to DriveBitIds
// which use less memory and are cheaper to compare.
struct DriveBitId
{
int id = -1;
DriveBitId() {};
DriveBitId(int id) : id(id) { }
bool operator==(const DriveBitId &other) const { return id == other.id; }
bool operator!=(const DriveBitId &other) const { return id != other.id; }
bool operator<(const DriveBitId &other) const { return id < other.id; }
Hasher hash_into(Hasher h) const;
};
// Essentially a dict<DriveBitId, pool<DriveBitId>> but using less memory
// and fewer allocations
struct DriveBitGraph {
dict<DriveBitId, DriveBitId> first_edges;
dict<DriveBitId, DriveBitId> second_edges;
dict<DriveBitId, pool<DriveBitId>> more_edges;
void add_edge(DriveBitId src, DriveBitId dst);
DriveBitId pop_edge(DriveBitId src);
void clear(DriveBitId src);
bool contains(DriveBitId src);
int count(DriveBitId src);
DriveBitId at(DriveBitId src, int index);
};
// The following two maps maintain a sparse DriveBit to DriveBitId mapping.
// This saves a lot of memory compared to a `dict<DriveBit, DriveBitId>` or
// `idict<DriveBit>`.
// Maps wires to the first DriveBitId of the consecutive range used for
// that wire.
dict<Wire *, DriveBitId> wire_offsets;
// Maps cell ports to a the first DriveBitId of the consecutive range used
// for that cell port.
dict<pair<Cell *, IdString>, DriveBitId> port_offsets;
// For the inverse map that maps DriveBitIds back to DriveBits we use a
// sorted map containing only the first DriveBit for each wire and cell
// port.
std::map<DriveBitId, DriveBit> drive_bits;
// As a memory optimization for gate level net lists we store single-bit
// wires and cell ports in a `dict` which requires less memory and fewer
// allocations than `std::map` but doesn't support the kind of lookups we
// need for a sparse coarse grained mapping.
dict<DriveBitId, DriveBit> isolated_drive_bits;
// Used for allocating DriveBitIds, none and constant states use a fixewd
// mapping to the first few ids, which we need to skip.
int next_offset = 1 + (int)State::Sm;
// Union-Find over directly connected bits that share the same single
// driver or are undriven. We never merge connections between drivers
// and/or kept wires.
mfp<DriveBitId> same_driver;
// For each bit, store a set of connected driver bits for which the
// explicit connection should be preserved and the driving direction is
// locally unambiguous (one side only drives or requires a driven value).
DriveBitGraph connected_drivers;
// For each bit, store a set of connected driver bits for which the
// explicit connection should be preserved and the driving direction is
// locally ambiguous. Every such ambiguous connection is also present in
// the reverse direction and has to be resolved when querying drivers.
DriveBitGraph connected_undirected;
// Subset of `connected_undirected` for caching the resolved driving
// direction. In case multiple drivers are present this can still contain
// both orientations of a single connection, but for a single driver only
// one will be present.
DriveBitGraph connected_oriented;
// Stores for which bits we already resolved the orientation (cached in
// `connected_oriented`).
pool<DriveBitId> oriented_present;
enum class BitMode {
NONE = 0, // Not driven, no need to keep wire
DRIVEN = 1, // Not driven, uses a value driven elsewhere
DRIVEN_UNIQUE = 2, // Uses a value driven elsewhere, has at most one direct connection
KEEP = 3, // Wire that should be kept
TRISTATE = 4, // Can drive a value but can also use a value driven elsewhere
DRIVER = 5, // Drives a value
};
BitMode bit_mode(DriveBit const &bit);
DriveBitId id_from_drive_bit(DriveBit const &bit);
DriveBit drive_bit_from_id(DriveBitId id);
void connect_directed_merge(DriveBitId driven_id, DriveBitId driver_id);
void connect_directed_buffer(DriveBitId driven_id, DriveBitId driver_id);
void connect_undirected(DriveBitId a_id, DriveBitId b_id);
public:
void add(Module *module);
// Add a single bit connection to the driver map.
void add(DriveBit const &a, DriveBit const &b);
template<typename T>
static constexpr bool is_sig_type() {
return
std::is_same<T, SigSpec>::value ||
std::is_same<T, SigChunk>::value ||
std::is_same<T, DriveSpec>::value ||
std::is_same<T, DriveChunk>::value ||
std::is_same<T, DriveChunkPort>::value ||
std::is_same<T, DriveChunkWire>::value ||
std::is_same<T, Const>::value;
}
// We use the enable_if to produce better compiler errors when unsupported
// types are used
template<typename T, typename U>
typename std::enable_if<is_sig_type<T>() && is_sig_type<U>()>::type
add(T const &a, U const &b)
{
log_assert(a.size() == b.size());
for (int i = 0; i != GetSize(a); ++i)
add(DriveBit(a[i]), DriveBit(b[i]));
}
// Specialized version that avoids unpacking
void add(SigSpec const &a, SigSpec const &b);
private:
void add_port(Cell *cell, IdString const &port, SigSpec const &b);
// Only used a local variables in `orient_undirected`, always cleared, only
// stored to reduce allocations.
pool<DriveBitId> orient_undirected_seen;
pool<DriveBitId> orient_undirected_drivers;
dict<DriveBitId, int> orient_undirected_distance;
void orient_undirected(DriveBitId id);
public:
DriveBit operator()(DriveBit const &bit);
DriveSpec operator()(DriveSpec spec);
private:
bool keep_wire(Wire *wire) {
// TODO configurable
return wire->has_attribute(ID(keep));
}
};
inline Hasher DriveBitWire::hash_into(Hasher h) const
{
h.eat(wire->name);
h.eat(offset);
return h;
}
inline Hasher DriveBitPort::hash_into(Hasher h) const
{
h.eat(cell->name);
h.eat(port);
h.eat(offset);
return h;
}
inline Hasher DriveBitMarker::hash_into(Hasher h) const
{
h.eat(marker);
h.eat(offset);
return h;
}
inline Hasher DriveBitMultiple::hash_into(Hasher h) const
{
h.eat(multiple_);
return h;
}
inline Hasher DriveBit::hash_into(Hasher h) const
{
switch (type_) {
case DriveType::NONE:
h.eat(0);
break;
case DriveType::CONSTANT:
h.eat(constant_);
break;
case DriveType::WIRE:
h.eat(wire_);
break;
case DriveType::PORT:
h.eat(port_);
break;
case DriveType::MARKER:
h.eat(marker_);
break;
case DriveType::MULTIPLE:
h.eat(multiple_);
break;
}
h.eat(type_);
return h;
}
inline Hasher DriveChunkWire::hash_into(Hasher h) const
{
h.eat(wire->name);
h.eat(width);
h.eat(offset);
return h;
}
inline Hasher DriveChunkPort::hash_into(Hasher h) const
{
h.eat(cell->name);
h.eat(port);
h.eat(width);
h.eat(offset);
return h;
}
inline Hasher DriveChunkMarker::hash_into(Hasher h) const
{
h.eat(marker);
h.eat(width);
h.eat(offset);
return h;
}
inline Hasher DriveChunkMultiple::hash_into(Hasher h) const
{
h.eat(width_);
h.eat(multiple_);
return h;
}
inline Hasher DriveChunk::hash_into(Hasher h) const
{
switch (type_) {
case DriveType::NONE:
h.eat(0);
break;
case DriveType::CONSTANT:
h.eat(constant_);
break;
case DriveType::WIRE:
h.eat(wire_);
break;
case DriveType::PORT:
h.eat(port_);
break;
case DriveType::MARKER:
h.eat(marker_);
break;
case DriveType::MULTIPLE:
h.eat(multiple_);
break;
}
h.eat(type_);
return h;
}
inline Hasher DriveSpec::hash_into(Hasher h) const
{
updhash();
h.eat(hash_);
return h;
}
inline Hasher DriverMap::DriveBitId::hash_into(Hasher h) const
{
h.eat(id);
return h;
}
YOSYS_NAMESPACE_END
#endif