2018-07-21 01:41:18 -05:00
|
|
|
/* -*- c++ -*-
|
2013-01-05 04:13:26 -06:00
|
|
|
* yosys -- Yosys Open SYnthesis Suite
|
|
|
|
*
|
2021-06-07 17:39:36 -05:00
|
|
|
* Copyright (C) 2012 Claire Xenia Wolf <claire@yosyshq.com>
|
2015-07-02 04:14:30 -05:00
|
|
|
*
|
2013-01-05 04:13:26 -06:00
|
|
|
* 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.
|
2015-07-02 04:14:30 -05:00
|
|
|
*
|
2013-01-05 04:13:26 -06:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2014-07-30 07:10:15 -05:00
|
|
|
#include "kernel/yosys.h"
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
#ifndef RTLIL_H
|
|
|
|
#define RTLIL_H
|
|
|
|
|
2014-07-31 06:19:47 -05:00
|
|
|
YOSYS_NAMESPACE_BEGIN
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
namespace RTLIL
|
|
|
|
{
|
2014-07-26 19:00:04 -05:00
|
|
|
enum State : unsigned char {
|
2013-01-05 04:13:26 -06:00
|
|
|
S0 = 0,
|
|
|
|
S1 = 1,
|
|
|
|
Sx = 2, // undefined value or conflict
|
|
|
|
Sz = 3, // high-impedance / not-connected
|
|
|
|
Sa = 4, // don't care (used only in cases)
|
|
|
|
Sm = 5 // marker (used internally by some passes)
|
|
|
|
};
|
2013-12-04 07:14:05 -06:00
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
enum SyncType : unsigned char {
|
2013-01-05 04:13:26 -06:00
|
|
|
ST0 = 0, // level sensitive: 0
|
|
|
|
ST1 = 1, // level sensitive: 1
|
|
|
|
STp = 2, // edge sensitive: posedge
|
|
|
|
STn = 3, // edge sensitive: negedge
|
|
|
|
STe = 4, // edge sensitive: both edges
|
2013-11-21 06:49:00 -06:00
|
|
|
STa = 5, // always active
|
2016-10-14 05:33:56 -05:00
|
|
|
STg = 6, // global clock
|
|
|
|
STi = 7 // init
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
enum ConstFlags : unsigned char {
|
2013-12-04 07:14:05 -06:00
|
|
|
CONST_FLAG_NONE = 0,
|
|
|
|
CONST_FLAG_STRING = 1,
|
2013-12-04 07:24:44 -06:00
|
|
|
CONST_FLAG_SIGNED = 2, // only used for parameters
|
2019-05-04 01:46:24 -05:00
|
|
|
CONST_FLAG_REAL = 4 // only used for parameters
|
2013-12-04 07:14:05 -06:00
|
|
|
};
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
struct Const;
|
2015-04-24 15:04:05 -05:00
|
|
|
struct AttrObject;
|
2013-01-05 04:13:26 -06:00
|
|
|
struct Selection;
|
2014-07-31 07:34:12 -05:00
|
|
|
struct Monitor;
|
2013-01-05 04:13:26 -06:00
|
|
|
struct Design;
|
|
|
|
struct Module;
|
|
|
|
struct Wire;
|
|
|
|
struct Memory;
|
|
|
|
struct Cell;
|
|
|
|
struct SigChunk;
|
2013-11-21 21:07:13 -06:00
|
|
|
struct SigBit;
|
2014-07-22 16:49:26 -05:00
|
|
|
struct SigSpecIterator;
|
2014-07-27 07:47:23 -05:00
|
|
|
struct SigSpecConstIterator;
|
2013-01-05 04:13:26 -06:00
|
|
|
struct SigSpec;
|
|
|
|
struct CaseRule;
|
|
|
|
struct SwitchRule;
|
2021-02-22 17:21:46 -06:00
|
|
|
struct MemWriteAction;
|
2013-01-05 04:13:26 -06:00
|
|
|
struct SyncRule;
|
|
|
|
struct Process;
|
2020-04-20 10:06:53 -05:00
|
|
|
struct Binding;
|
2013-01-05 04:13:26 -06:00
|
|
|
|
|
|
|
typedef std::pair<SigSpec, SigSpec> SigSig;
|
|
|
|
|
2014-08-02 06:11:01 -05:00
|
|
|
struct IdString
|
|
|
|
{
|
2019-03-11 14:12:28 -05:00
|
|
|
#undef YOSYS_XTRACE_GET_PUT
|
|
|
|
#undef YOSYS_SORT_ID_FREE_LIST
|
2019-08-10 04:41:09 -05:00
|
|
|
#undef YOSYS_USE_STICKY_IDS
|
2019-08-11 05:23:16 -05:00
|
|
|
#undef YOSYS_NO_IDS_REFCNT
|
2019-03-11 14:12:28 -05:00
|
|
|
|
2014-08-02 08:52:21 -05:00
|
|
|
// the global id string cache
|
2014-08-02 08:11:35 -05:00
|
|
|
|
2014-12-11 14:46:36 -06:00
|
|
|
static struct destruct_guard_t {
|
2014-12-16 03:38:25 -06:00
|
|
|
bool ok; // POD, will be initialized to zero
|
2014-12-11 14:46:36 -06:00
|
|
|
destruct_guard_t() { ok = true; }
|
|
|
|
~destruct_guard_t() { ok = false; }
|
|
|
|
} destruct_guard;
|
|
|
|
|
2014-08-02 08:44:10 -05:00
|
|
|
static std::vector<char*> global_id_storage_;
|
2014-12-26 14:59:41 -06:00
|
|
|
static dict<char*, int, hash_cstr_ops> global_id_index_;
|
2019-08-11 05:23:16 -05:00
|
|
|
#ifndef YOSYS_NO_IDS_REFCNT
|
|
|
|
static std::vector<int> global_refcount_storage_;
|
2014-08-02 08:11:35 -05:00
|
|
|
static std::vector<int> global_free_idx_list_;
|
2019-08-11 05:23:16 -05:00
|
|
|
#endif
|
2014-08-02 08:11:35 -05:00
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
#ifdef YOSYS_USE_STICKY_IDS
|
2019-03-11 14:12:28 -05:00
|
|
|
static int last_created_idx_ptr_;
|
|
|
|
static int last_created_idx_[8];
|
2019-08-10 04:41:09 -05:00
|
|
|
#endif
|
2019-03-11 14:12:28 -05:00
|
|
|
|
|
|
|
static inline void xtrace_db_dump()
|
|
|
|
{
|
|
|
|
#ifdef YOSYS_XTRACE_GET_PUT
|
|
|
|
for (int idx = 0; idx < GetSize(global_id_storage_); idx++)
|
|
|
|
{
|
|
|
|
if (global_id_storage_.at(idx) == nullptr)
|
|
|
|
log("#X# DB-DUMP index %d: FREE\n", idx);
|
|
|
|
else
|
|
|
|
log("#X# DB-DUMP index %d: '%s' (ref %d)\n", idx, global_id_storage_.at(idx), global_refcount_storage_.at(idx));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void checkpoint()
|
|
|
|
{
|
2019-08-10 04:41:09 -05:00
|
|
|
#ifdef YOSYS_USE_STICKY_IDS
|
2019-03-11 14:12:28 -05:00
|
|
|
last_created_idx_ptr_ = 0;
|
|
|
|
for (int i = 0; i < 8; i++) {
|
|
|
|
if (last_created_idx_[i])
|
|
|
|
put_reference(last_created_idx_[i]);
|
|
|
|
last_created_idx_[i] = 0;
|
|
|
|
}
|
2019-08-10 04:41:09 -05:00
|
|
|
#endif
|
2019-03-11 14:12:28 -05:00
|
|
|
#ifdef YOSYS_SORT_ID_FREE_LIST
|
|
|
|
std::sort(global_free_idx_list_.begin(), global_free_idx_list_.end(), std::greater<int>());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-08-02 08:11:35 -05:00
|
|
|
static inline int get_reference(int idx)
|
|
|
|
{
|
2019-08-10 04:41:09 -05:00
|
|
|
if (idx) {
|
2019-08-11 05:23:16 -05:00
|
|
|
#ifndef YOSYS_NO_IDS_REFCNT
|
2019-08-10 04:41:09 -05:00
|
|
|
global_refcount_storage_[idx]++;
|
2019-08-11 05:23:16 -05:00
|
|
|
#endif
|
2019-03-11 14:12:28 -05:00
|
|
|
#ifdef YOSYS_XTRACE_GET_PUT
|
2019-08-10 04:41:09 -05:00
|
|
|
if (yosys_xtrace)
|
|
|
|
log("#X# GET-BY-INDEX '%s' (index %d, refcount %d)\n", global_id_storage_.at(idx), idx, global_refcount_storage_.at(idx));
|
2019-03-11 14:12:28 -05:00
|
|
|
#endif
|
2019-08-10 04:41:09 -05:00
|
|
|
}
|
2014-08-02 08:11:35 -05:00
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
static int get_reference(const char *p)
|
2014-08-02 08:11:35 -05:00
|
|
|
{
|
2014-12-11 14:46:36 -06:00
|
|
|
log_assert(destruct_guard.ok);
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
if (!p[0])
|
|
|
|
return 0;
|
|
|
|
|
2014-08-02 08:44:10 -05:00
|
|
|
auto it = global_id_index_.find((char*)p);
|
2014-08-02 08:11:35 -05:00
|
|
|
if (it != global_id_index_.end()) {
|
2019-08-11 05:23:16 -05:00
|
|
|
#ifndef YOSYS_NO_IDS_REFCNT
|
2014-08-02 08:11:35 -05:00
|
|
|
global_refcount_storage_.at(it->second)++;
|
2019-08-11 05:23:16 -05:00
|
|
|
#endif
|
2019-03-11 14:12:28 -05:00
|
|
|
#ifdef YOSYS_XTRACE_GET_PUT
|
2019-08-10 04:41:09 -05:00
|
|
|
if (yosys_xtrace)
|
2019-03-11 14:12:28 -05:00
|
|
|
log("#X# GET-BY-NAME '%s' (index %d, refcount %d)\n", global_id_storage_.at(it->second), it->second, global_refcount_storage_.at(it->second));
|
|
|
|
#endif
|
2014-08-02 08:11:35 -05:00
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
2020-05-27 00:20:39 -05:00
|
|
|
log_assert(p[0] == '$' || p[0] == '\\');
|
|
|
|
log_assert(p[1] != 0);
|
|
|
|
for (const char *c = p; *c; c++)
|
2021-03-16 21:43:25 -05:00
|
|
|
if ((unsigned)*c <= (unsigned)' ')
|
|
|
|
log_error("Found control character or space (0x%02hhx) in string '%s' which is not allowed in RTLIL identifiers\n", *c, p);
|
2020-05-27 00:20:39 -05:00
|
|
|
|
2019-08-11 05:23:16 -05:00
|
|
|
#ifndef YOSYS_NO_IDS_REFCNT
|
2014-08-02 08:11:35 -05:00
|
|
|
if (global_free_idx_list_.empty()) {
|
2019-08-10 04:41:09 -05:00
|
|
|
if (global_id_storage_.empty()) {
|
|
|
|
global_refcount_storage_.push_back(0);
|
|
|
|
global_id_storage_.push_back((char*)"");
|
|
|
|
global_id_index_[global_id_storage_.back()] = 0;
|
|
|
|
}
|
2014-08-02 08:11:35 -05:00
|
|
|
log_assert(global_id_storage_.size() < 0x40000000);
|
|
|
|
global_free_idx_list_.push_back(global_id_storage_.size());
|
2014-08-02 08:44:10 -05:00
|
|
|
global_id_storage_.push_back(nullptr);
|
2014-08-02 08:11:35 -05:00
|
|
|
global_refcount_storage_.push_back(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int idx = global_free_idx_list_.back();
|
|
|
|
global_free_idx_list_.pop_back();
|
2014-08-02 08:44:10 -05:00
|
|
|
global_id_storage_.at(idx) = strdup(p);
|
2014-08-02 08:11:35 -05:00
|
|
|
global_id_index_[global_id_storage_.at(idx)] = idx;
|
|
|
|
global_refcount_storage_.at(idx)++;
|
2019-08-11 05:23:16 -05:00
|
|
|
#else
|
|
|
|
if (global_id_storage_.empty()) {
|
|
|
|
global_id_storage_.push_back((char*)"");
|
|
|
|
global_id_index_[global_id_storage_.back()] = 0;
|
|
|
|
}
|
|
|
|
int idx = global_id_storage_.size();
|
|
|
|
global_id_storage_.push_back(strdup(p));
|
|
|
|
global_id_index_[global_id_storage_.back()] = idx;
|
|
|
|
#endif
|
2014-12-29 06:33:33 -06:00
|
|
|
|
|
|
|
if (yosys_xtrace) {
|
|
|
|
log("#X# New IdString '%s' with index %d.\n", p, idx);
|
|
|
|
log_backtrace("-X- ", yosys_xtrace-1);
|
|
|
|
}
|
|
|
|
|
2019-03-11 14:12:28 -05:00
|
|
|
#ifdef YOSYS_XTRACE_GET_PUT
|
2019-08-10 04:41:09 -05:00
|
|
|
if (yosys_xtrace)
|
2019-03-11 14:12:28 -05:00
|
|
|
log("#X# GET-BY-NAME '%s' (index %d, refcount %d)\n", global_id_storage_.at(idx), idx, global_refcount_storage_.at(idx));
|
|
|
|
#endif
|
2019-08-10 04:41:09 -05:00
|
|
|
|
|
|
|
#ifdef YOSYS_USE_STICKY_IDS
|
|
|
|
// Avoid Create->Delete->Create pattern
|
|
|
|
if (last_created_idx_[last_created_idx_ptr_])
|
|
|
|
put_reference(last_created_idx_[last_created_idx_ptr_]);
|
|
|
|
last_created_idx_[last_created_idx_ptr_] = idx;
|
|
|
|
get_reference(last_created_idx_[last_created_idx_ptr_]);
|
|
|
|
last_created_idx_ptr_ = (last_created_idx_ptr_ + 1) & 7;
|
|
|
|
#endif
|
|
|
|
|
2014-08-02 08:11:35 -05:00
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2019-08-11 05:23:16 -05:00
|
|
|
#ifndef YOSYS_NO_IDS_REFCNT
|
2014-08-02 08:11:35 -05:00
|
|
|
static inline void put_reference(int idx)
|
|
|
|
{
|
2014-12-11 14:46:36 -06:00
|
|
|
// put_reference() may be called from destructors after the destructor of
|
|
|
|
// global_refcount_storage_ has been run. in this case we simply do nothing.
|
2019-08-10 04:41:09 -05:00
|
|
|
if (!destruct_guard.ok || !idx)
|
2014-12-11 14:46:36 -06:00
|
|
|
return;
|
|
|
|
|
2019-03-11 14:12:28 -05:00
|
|
|
#ifdef YOSYS_XTRACE_GET_PUT
|
|
|
|
if (yosys_xtrace) {
|
|
|
|
log("#X# PUT '%s' (index %d, refcount %d)\n", global_id_storage_.at(idx), idx, global_refcount_storage_.at(idx));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
int &refcount = global_refcount_storage_[idx];
|
2014-08-02 09:03:18 -05:00
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
if (--refcount > 0)
|
2014-08-02 08:11:35 -05:00
|
|
|
return;
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
log_assert(refcount == 0);
|
2020-03-12 16:42:07 -05:00
|
|
|
free_reference(idx);
|
|
|
|
}
|
|
|
|
static inline void free_reference(int idx)
|
|
|
|
{
|
2014-12-29 06:33:33 -06:00
|
|
|
if (yosys_xtrace) {
|
|
|
|
log("#X# Removed IdString '%s' with index %d.\n", global_id_storage_.at(idx), idx);
|
|
|
|
log_backtrace("-X- ", yosys_xtrace-1);
|
|
|
|
}
|
|
|
|
|
2014-08-02 08:11:35 -05:00
|
|
|
global_id_index_.erase(global_id_storage_.at(idx));
|
2014-08-02 08:44:10 -05:00
|
|
|
free(global_id_storage_.at(idx));
|
2014-08-02 09:03:18 -05:00
|
|
|
global_id_storage_.at(idx) = nullptr;
|
2014-08-02 08:11:35 -05:00
|
|
|
global_free_idx_list_.push_back(idx);
|
|
|
|
}
|
2019-08-11 05:23:16 -05:00
|
|
|
#else
|
|
|
|
static inline void put_reference(int) { }
|
|
|
|
#endif
|
2014-08-02 08:11:35 -05:00
|
|
|
|
2014-08-02 08:52:21 -05:00
|
|
|
// the actual IdString object is just is a single int
|
2014-08-02 08:11:35 -05:00
|
|
|
|
|
|
|
int index_;
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline IdString() : index_(0) { }
|
|
|
|
inline IdString(const char *str) : index_(get_reference(str)) { }
|
|
|
|
inline IdString(const IdString &str) : index_(get_reference(str.index_)) { }
|
|
|
|
inline IdString(IdString &&str) : index_(str.index_) { str.index_ = 0; }
|
|
|
|
inline IdString(const std::string &str) : index_(get_reference(str.c_str())) { }
|
|
|
|
inline ~IdString() { put_reference(index_); }
|
2014-08-02 08:11:35 -05:00
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline void operator=(const IdString &rhs) {
|
2014-08-02 08:11:35 -05:00
|
|
|
put_reference(index_);
|
|
|
|
index_ = get_reference(rhs.index_);
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline void operator=(const char *rhs) {
|
2014-08-02 08:11:35 -05:00
|
|
|
IdString id(rhs);
|
|
|
|
*this = id;
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline void operator=(const std::string &rhs) {
|
2014-08-02 08:11:35 -05:00
|
|
|
IdString id(rhs);
|
|
|
|
*this = id;
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline const char *c_str() const {
|
2014-08-02 08:11:35 -05:00
|
|
|
return global_id_storage_.at(index_);
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline std::string str() const {
|
2014-08-02 08:44:10 -05:00
|
|
|
return std::string(global_id_storage_.at(index_));
|
|
|
|
}
|
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline bool operator<(const IdString &rhs) const {
|
2014-08-02 08:11:35 -05:00
|
|
|
return index_ < rhs.index_;
|
|
|
|
}
|
2014-08-02 06:11:01 -05:00
|
|
|
|
2019-08-10 04:41:09 -05:00
|
|
|
inline bool operator==(const IdString &rhs) const { return index_ == rhs.index_; }
|
|
|
|
inline bool operator!=(const IdString &rhs) const { return index_ != rhs.index_; }
|
2014-08-02 06:11:01 -05:00
|
|
|
|
2015-08-14 03:56:05 -05:00
|
|
|
// The methods below are just convenience functions for better compatibility with std::string.
|
2014-08-02 06:11:01 -05:00
|
|
|
|
2020-05-26 05:51:06 -05:00
|
|
|
bool operator==(const std::string &rhs) const { return c_str() == rhs; }
|
|
|
|
bool operator!=(const std::string &rhs) const { return c_str() != rhs; }
|
2014-08-02 08:11:35 -05:00
|
|
|
|
2014-08-04 08:08:35 -05:00
|
|
|
bool operator==(const char *rhs) const { return strcmp(c_str(), rhs) == 0; }
|
|
|
|
bool operator!=(const char *rhs) const { return strcmp(c_str(), rhs) != 0; }
|
2014-08-02 06:11:01 -05:00
|
|
|
|
2014-08-02 11:58:40 -05:00
|
|
|
char operator[](size_t i) const {
|
2014-08-02 12:08:02 -05:00
|
|
|
const char *p = c_str();
|
2015-07-02 04:14:30 -05:00
|
|
|
for (; i != 0; i--, p++)
|
2014-08-02 12:08:02 -05:00
|
|
|
log_assert(*p != 0);
|
|
|
|
return *p;
|
2014-08-02 11:58:40 -05:00
|
|
|
}
|
|
|
|
|
2014-08-02 08:44:10 -05:00
|
|
|
std::string substr(size_t pos = 0, size_t len = std::string::npos) const {
|
2014-08-02 09:03:18 -05:00
|
|
|
if (len == std::string::npos || len >= strlen(c_str() + pos))
|
2014-08-02 08:44:10 -05:00
|
|
|
return std::string(c_str() + pos);
|
|
|
|
else
|
|
|
|
return std::string(c_str() + pos, len);
|
|
|
|
}
|
|
|
|
|
2019-08-06 18:42:25 -05:00
|
|
|
int compare(size_t pos, size_t len, const char* s) const {
|
|
|
|
return strncmp(c_str()+pos, s, len);
|
|
|
|
}
|
|
|
|
|
2019-06-14 14:25:06 -05:00
|
|
|
bool begins_with(const char* prefix) const {
|
|
|
|
size_t len = strlen(prefix);
|
|
|
|
if (size() < len) return false;
|
2019-08-06 21:07:45 -05:00
|
|
|
return compare(0, len, prefix) == 0;
|
2019-06-14 14:25:06 -05:00
|
|
|
}
|
|
|
|
|
2019-02-26 14:04:16 -06:00
|
|
|
bool ends_with(const char* suffix) const {
|
|
|
|
size_t len = strlen(suffix);
|
|
|
|
if (size() < len) return false;
|
2019-08-06 21:07:45 -05:00
|
|
|
return compare(size()-len, len, suffix) == 0;
|
2019-02-26 14:04:16 -06:00
|
|
|
}
|
|
|
|
|
verilog: significant block scoping improvements
This change set contains a number of bug fixes and improvements related to
scoping and resolution in generate and procedural blocks. While many of the
frontend changes are interdependent, it may be possible bring the techmap
changes in under a separate PR.
Declarations within unnamed generate blocks previously encountered issues
because the data declarations were left un-prefixed, breaking proper scoping.
The LRM outlines behavior for generating names for unnamed generate blocks. The
original goal was to add this implicit labelling, but doing so exposed a number
of issues downstream. Additional testing highlighted other closely related scope
resolution issues, which have been fixed. This change also adds support for
block item declarations within unnamed blocks in SystemVerilog mode.
1. Unlabled generate blocks are now implicitly named according to the LRM in
`label_genblks`, which is invoked at the beginning of module elaboration
2. The Verilog parser no longer wraps explicitly named generate blocks in a
synthetic unnamed generate block to avoid creating extra hierarchy levels
where they should not exist
3. The techmap phase now allows special control identifiers to be used outside
of the topmost scope, which is necessary because such wires and cells often
appear in unlabeled generate blocks, which now prefix the declarations within
4. Some techlibs required modifications because they relied on the previous
invalid scope resolution behavior
5. `expand_genblock` has been simplified, now only expanding the outermost
scope, completely deferring the inspection and elaboration of nested scopes;
names are now resolved by looking in the innermost scope and stepping outward
6. Loop variables now always become localparams during unrolling, allowing them
to be resolved and shadowed like any other identifier
7. Identifiers in synthetic function call scopes are now prefixed and resolved
in largely the same manner as other blocks
before: `$func$\func_01$tests/simple/scopes.blk.v:60$5$\blk\x`
after: `\func_01$func$tests/simple/scopes.v:60$5.blk.x`
8. Support identifiers referencing a local generate scope nested more
than 1 level deep, i.e. `B.C.x` while within generate scope `A`, or using a
prefix of a current or parent scope, i.e. `B.C.D.x` while in `A.B`, `A.B.C`,
or `A.B.C.D`
9. Variables can now be declared within unnamed blocks in SystemVerilog mode
Addresses the following issues: 656, 2423, 2493
2021-01-27 12:30:22 -06:00
|
|
|
bool contains(const char* str) const {
|
|
|
|
return strstr(c_str(), str);
|
|
|
|
}
|
|
|
|
|
2014-08-02 08:44:10 -05:00
|
|
|
size_t size() const {
|
2019-08-06 18:42:25 -05:00
|
|
|
return strlen(c_str());
|
2014-08-02 08:44:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool empty() const {
|
|
|
|
return c_str()[0] == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void clear() {
|
|
|
|
*this = IdString();
|
|
|
|
}
|
2014-08-02 09:03:18 -05:00
|
|
|
|
2014-12-26 14:35:22 -06:00
|
|
|
unsigned int hash() const {
|
|
|
|
return index_;
|
|
|
|
}
|
|
|
|
|
2014-12-28 19:01:42 -06:00
|
|
|
// The following is a helper key_compare class. Instead of for example std::set<Cell*>
|
|
|
|
// use std::set<Cell*, IdString::compare_ptr_by_name<Cell>> if the order of cells in the
|
2014-08-02 09:03:18 -05:00
|
|
|
// set has an influence on the algorithm.
|
|
|
|
|
|
|
|
template<typename T> struct compare_ptr_by_name {
|
2014-11-07 08:21:03 -06:00
|
|
|
bool operator()(const T *a, const T *b) const {
|
2014-08-02 09:03:18 -05:00
|
|
|
return (a == nullptr || b == nullptr) ? (a < b) : (a->name < b->name);
|
|
|
|
}
|
|
|
|
};
|
2014-08-04 08:08:35 -05:00
|
|
|
|
|
|
|
// often one needs to check if a given IdString is part of a list (for example a list
|
|
|
|
// of cell types). the following functions helps with that.
|
|
|
|
|
2020-03-15 11:47:20 -05:00
|
|
|
template<typename... Args>
|
|
|
|
bool in(Args... args) const {
|
2020-03-16 14:58:55 -05:00
|
|
|
// Credit: https://articles.emptycrate.com/2016/05/14/folds_in_cpp11_ish.html
|
|
|
|
bool result = false;
|
|
|
|
(void) std::initializer_list<int>{ (result = result || in(args), 0)... };
|
|
|
|
return result;
|
2014-08-04 08:08:35 -05:00
|
|
|
}
|
|
|
|
|
2020-04-01 16:10:24 -05:00
|
|
|
bool in(const IdString &rhs) const { return *this == rhs; }
|
2014-12-31 15:50:08 -06:00
|
|
|
bool in(const char *rhs) const { return *this == rhs; }
|
|
|
|
bool in(const std::string &rhs) const { return *this == rhs; }
|
|
|
|
bool in(const pool<IdString> &rhs) const { return rhs.count(*this) != 0; }
|
2020-09-03 10:37:58 -05:00
|
|
|
|
2020-12-12 14:50:37 -06:00
|
|
|
bool isPublic() const { return begins_with("\\"); }
|
2014-08-01 17:45:25 -05:00
|
|
|
};
|
2013-01-05 04:13:26 -06:00
|
|
|
|
2019-08-10 05:24:16 -05:00
|
|
|
namespace ID {
|
2020-03-12 14:54:30 -05:00
|
|
|
#define X(_id) extern IdString _id;
|
2020-04-09 11:14:03 -05:00
|
|
|
#include "kernel/constids.inc"
|
2020-03-12 14:54:30 -05:00
|
|
|
#undef X
|
2019-08-10 05:24:16 -05:00
|
|
|
};
|
|
|
|
|
2020-01-07 14:52:03 -06:00
|
|
|
extern dict<std::string, std::string> constpad;
|
|
|
|
|
2020-04-08 10:36:12 -05:00
|
|
|
const pool<IdString> &builtin_ff_cell_types();
|
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
static inline std::string escape_id(const std::string &str) {
|
2013-01-05 04:13:26 -06:00
|
|
|
if (str.size() > 0 && str[0] != '\\' && str[0] != '$')
|
|
|
|
return "\\" + str;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
static inline std::string unescape_id(const std::string &str) {
|
2015-01-30 15:51:16 -06:00
|
|
|
if (str.size() < 2)
|
|
|
|
return str;
|
|
|
|
if (str[0] != '\\')
|
|
|
|
return str;
|
|
|
|
if (str[1] == '$' || str[1] == '\\')
|
|
|
|
return str;
|
|
|
|
if (str[1] >= '0' && str[1] <= '9')
|
|
|
|
return str;
|
|
|
|
return str.substr(1);
|
2013-01-05 04:13:26 -06:00
|
|
|
}
|
|
|
|
|
2014-08-01 17:45:25 -05:00
|
|
|
static inline std::string unescape_id(RTLIL::IdString str) {
|
|
|
|
return unescape_id(str.str());
|
|
|
|
}
|
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
static inline const char *id2cstr(RTLIL::IdString str) {
|
2014-08-02 09:03:18 -05:00
|
|
|
return log_id(str);
|
2014-08-01 17:45:25 -05:00
|
|
|
}
|
|
|
|
|
2014-08-14 19:40:46 -05:00
|
|
|
template <typename T> struct sort_by_name_id {
|
2013-08-09 05:42:32 -05:00
|
|
|
bool operator()(T *a, T *b) const {
|
|
|
|
return a->name < b->name;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-08-14 19:40:46 -05:00
|
|
|
template <typename T> struct sort_by_name_str {
|
|
|
|
bool operator()(T *a, T *b) const {
|
|
|
|
return strcmp(a->name.c_str(), b->name.c_str()) < 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sort_by_id_str {
|
|
|
|
bool operator()(RTLIL::IdString a, RTLIL::IdString b) const {
|
|
|
|
return strcmp(a.c_str(), b.c_str()) < 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
// see calc.cc for the implementation of this functions
|
|
|
|
RTLIL::Const const_not (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_and (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_or (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_xor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_xnor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
|
|
|
|
RTLIL::Const const_reduce_and (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_reduce_or (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_reduce_xor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_reduce_xnor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_reduce_bool (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
|
|
|
|
RTLIL::Const const_logic_not (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_logic_and (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_logic_or (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
|
|
|
|
RTLIL::Const const_shl (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_shr (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_sshl (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_sshr (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2014-07-29 07:42:33 -05:00
|
|
|
RTLIL::Const const_shift (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_shiftx (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2013-01-05 04:13:26 -06:00
|
|
|
|
|
|
|
RTLIL::Const const_lt (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_le (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_eq (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_ne (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2013-12-27 06:50:08 -06:00
|
|
|
RTLIL::Const const_eqx (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_nex (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::Const const_ge (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_gt (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
|
|
|
|
RTLIL::Const const_add (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_sub (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_mul (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_div (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2020-04-21 05:51:58 -05:00
|
|
|
RTLIL::Const const_divfloor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2020-04-08 12:30:47 -05:00
|
|
|
RTLIL::Const const_modfloor (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::Const const_mod (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_pow (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
|
|
|
|
RTLIL::Const const_pos (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
|
|
|
RTLIL::Const const_neg (const RTLIL::Const &arg1, const RTLIL::Const &arg2, bool signed1, bool signed2, int result_len);
|
2014-07-27 03:13:22 -05:00
|
|
|
|
|
|
|
|
|
|
|
// This iterator-range-pair is used for Design::modules(), Module::wires() and Module::cells().
|
|
|
|
// It maintains a reference counter that is used to make sure that the container is not modified while being iterated over.
|
|
|
|
|
|
|
|
template<typename T>
|
2019-07-25 02:51:09 -05:00
|
|
|
struct ObjIterator {
|
|
|
|
using iterator_category = std::forward_iterator_tag;
|
|
|
|
using value_type = T;
|
|
|
|
using difference_type = ptrdiff_t;
|
|
|
|
using pointer = T*;
|
|
|
|
using reference = T&;
|
2014-12-26 03:53:21 -06:00
|
|
|
typename dict<RTLIL::IdString, T>::iterator it;
|
|
|
|
dict<RTLIL::IdString, T> *list_p;
|
2014-07-27 03:13:22 -05:00
|
|
|
int *refcount_p;
|
|
|
|
|
|
|
|
ObjIterator() : list_p(nullptr), refcount_p(nullptr) {
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjIterator(decltype(list_p) list_p, int *refcount_p) : list_p(list_p), refcount_p(refcount_p) {
|
|
|
|
if (list_p->empty()) {
|
|
|
|
this->list_p = nullptr;
|
|
|
|
this->refcount_p = nullptr;
|
|
|
|
} else {
|
|
|
|
it = list_p->begin();
|
|
|
|
(*refcount_p)++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjIterator(const RTLIL::ObjIterator<T> &other) {
|
|
|
|
it = other.it;
|
|
|
|
list_p = other.list_p;
|
|
|
|
refcount_p = other.refcount_p;
|
|
|
|
if (refcount_p)
|
|
|
|
(*refcount_p)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ObjIterator &operator=(const RTLIL::ObjIterator<T> &other) {
|
|
|
|
if (refcount_p)
|
|
|
|
(*refcount_p)--;
|
|
|
|
it = other.it;
|
|
|
|
list_p = other.list_p;
|
|
|
|
refcount_p = other.refcount_p;
|
|
|
|
if (refcount_p)
|
|
|
|
(*refcount_p)++;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
~ObjIterator() {
|
|
|
|
if (refcount_p)
|
|
|
|
(*refcount_p)--;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline T operator*() const {
|
2014-07-28 04:08:55 -05:00
|
|
|
log_assert(list_p != nullptr);
|
2014-07-27 03:13:22 -05:00
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool operator!=(const RTLIL::ObjIterator<T> &other) const {
|
|
|
|
if (list_p == nullptr || other.list_p == nullptr)
|
|
|
|
return list_p != other.list_p;
|
|
|
|
return it != other.it;
|
|
|
|
}
|
|
|
|
|
2019-07-24 06:33:07 -05:00
|
|
|
|
|
|
|
inline bool operator==(const RTLIL::ObjIterator<T> &other) const {
|
|
|
|
return !(*this != other);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ObjIterator<T>& operator++() {
|
2014-07-28 04:08:55 -05:00
|
|
|
log_assert(list_p != nullptr);
|
2014-07-27 03:13:22 -05:00
|
|
|
if (++it == list_p->end()) {
|
|
|
|
(*refcount_p)--;
|
|
|
|
list_p = nullptr;
|
|
|
|
refcount_p = nullptr;
|
|
|
|
}
|
2019-07-24 06:33:07 -05:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2020-06-17 17:32:34 -05:00
|
|
|
inline ObjIterator<T>& operator+=(int amt) {
|
|
|
|
log_assert(list_p != nullptr);
|
|
|
|
it += amt;
|
|
|
|
if (it == list_p->end()) {
|
|
|
|
(*refcount_p)--;
|
|
|
|
list_p = nullptr;
|
|
|
|
refcount_p = nullptr;
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ObjIterator<T> operator+(int amt) {
|
|
|
|
log_assert(list_p != nullptr);
|
|
|
|
ObjIterator<T> new_obj(*this);
|
|
|
|
new_obj.it += amt;
|
|
|
|
if (new_obj.it == list_p->end()) {
|
|
|
|
(*(new_obj.refcount_p))--;
|
|
|
|
new_obj.list_p = nullptr;
|
|
|
|
new_obj.refcount_p = nullptr;
|
|
|
|
}
|
|
|
|
return new_obj;
|
|
|
|
}
|
|
|
|
|
2019-07-24 06:33:07 -05:00
|
|
|
inline const ObjIterator<T> operator++(int) {
|
|
|
|
ObjIterator<T> result(*this);
|
|
|
|
++(*this);
|
|
|
|
return result;
|
2014-07-27 03:13:22 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct ObjRange
|
|
|
|
{
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, T> *list_p;
|
2014-07-27 03:13:22 -05:00
|
|
|
int *refcount_p;
|
|
|
|
|
|
|
|
ObjRange(decltype(list_p) list_p, int *refcount_p) : list_p(list_p), refcount_p(refcount_p) { }
|
|
|
|
RTLIL::ObjIterator<T> begin() { return RTLIL::ObjIterator<T>(list_p, refcount_p); }
|
|
|
|
RTLIL::ObjIterator<T> end() { return RTLIL::ObjIterator<T>(); }
|
2014-07-27 03:41:06 -05:00
|
|
|
|
2014-07-31 07:11:39 -05:00
|
|
|
size_t size() const {
|
|
|
|
return list_p->size();
|
|
|
|
}
|
|
|
|
|
2014-12-26 14:59:41 -06:00
|
|
|
operator pool<T>() const {
|
|
|
|
pool<T> result;
|
2014-07-27 03:41:06 -05:00
|
|
|
for (auto &it : *list_p)
|
|
|
|
result.insert(it.second);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
operator std::vector<T>() const {
|
|
|
|
std::vector<T> result;
|
|
|
|
result.reserve(list_p->size());
|
|
|
|
for (auto &it : *list_p)
|
|
|
|
result.push_back(it.second);
|
|
|
|
return result;
|
|
|
|
}
|
2014-07-30 15:04:30 -05:00
|
|
|
|
2015-01-19 06:59:08 -06:00
|
|
|
pool<T> to_pool() const { return *this; }
|
2014-07-30 15:04:30 -05:00
|
|
|
std::vector<T> to_vector() const { return *this; }
|
2014-07-27 03:13:22 -05:00
|
|
|
};
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
struct RTLIL::Const
|
|
|
|
{
|
2013-12-04 07:14:05 -06:00
|
|
|
int flags;
|
2013-01-05 04:13:26 -06:00
|
|
|
std::vector<RTLIL::State> bits;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-12-04 07:14:05 -06:00
|
|
|
Const();
|
|
|
|
Const(std::string str);
|
2013-01-05 04:13:26 -06:00
|
|
|
Const(int val, int width = 32);
|
|
|
|
Const(RTLIL::State bit, int width = 1);
|
2015-11-26 11:11:34 -06:00
|
|
|
Const(const std::vector<RTLIL::State> &bits) : bits(bits) { flags = CONST_FLAG_NONE; }
|
2014-09-19 08:50:55 -05:00
|
|
|
Const(const std::vector<bool> &bits);
|
2018-08-13 08:18:46 -05:00
|
|
|
Const(const RTLIL::Const &c);
|
2019-05-05 03:00:27 -05:00
|
|
|
RTLIL::Const &operator =(const RTLIL::Const &other) = default;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
bool operator <(const RTLIL::Const &other) const;
|
|
|
|
bool operator ==(const RTLIL::Const &other) const;
|
|
|
|
bool operator !=(const RTLIL::Const &other) const;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
bool as_bool() const;
|
2014-08-24 08:14:00 -05:00
|
|
|
int as_int(bool is_signed = false) const;
|
2013-01-05 04:13:26 -06:00
|
|
|
std::string as_string() const;
|
2020-03-18 13:21:53 -05:00
|
|
|
static Const from_string(const std::string &str);
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-12-04 07:14:05 -06:00
|
|
|
std::string decode_string() const;
|
2014-08-31 11:07:48 -05:00
|
|
|
|
|
|
|
inline int size() const { return bits.size(); }
|
2019-10-04 15:31:33 -05:00
|
|
|
inline bool empty() const { return bits.empty(); }
|
2015-04-06 10:06:15 -05:00
|
|
|
inline RTLIL::State &operator[](int index) { return bits.at(index); }
|
2015-11-26 11:11:34 -06:00
|
|
|
inline const RTLIL::State &operator[](int index) const { return bits.at(index); }
|
2019-10-04 15:31:33 -05:00
|
|
|
inline decltype(bits)::iterator begin() { return bits.begin(); }
|
|
|
|
inline decltype(bits)::iterator end() { return bits.end(); }
|
2015-04-06 10:06:15 -05:00
|
|
|
|
2017-08-18 04:40:08 -05:00
|
|
|
bool is_fully_zero() const;
|
2017-12-13 19:06:39 -06:00
|
|
|
bool is_fully_ones() const;
|
2017-08-18 04:40:08 -05:00
|
|
|
bool is_fully_def() const;
|
|
|
|
bool is_fully_undef() const;
|
2021-06-09 11:41:57 -05:00
|
|
|
bool is_onehot(int *pos = nullptr) const;
|
2017-08-18 04:40:08 -05:00
|
|
|
|
2015-04-06 10:06:15 -05:00
|
|
|
inline RTLIL::Const extract(int offset, int len = 1, RTLIL::State padding = RTLIL::State::S0) const {
|
|
|
|
RTLIL::Const ret;
|
|
|
|
ret.bits.reserve(len);
|
|
|
|
for (int i = offset; i < offset + len; i++)
|
|
|
|
ret.bits.push_back(i < GetSize(bits) ? bits[i] : padding);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-12-28 19:01:42 -06:00
|
|
|
|
2019-03-23 08:38:48 -05:00
|
|
|
void extu(int width) {
|
|
|
|
bits.resize(width, RTLIL::State::S0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void exts(int width) {
|
|
|
|
bits.resize(width, bits.empty() ? RTLIL::State::Sx : bits.back());
|
|
|
|
}
|
|
|
|
|
2014-12-28 19:01:42 -06:00
|
|
|
inline unsigned int hash() const {
|
2014-12-30 11:51:24 -06:00
|
|
|
unsigned int h = mkhash_init;
|
2014-12-28 19:01:42 -06:00
|
|
|
for (auto b : bits)
|
|
|
|
mkhash(h, b);
|
|
|
|
return h;
|
|
|
|
}
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::AttrObject
|
|
|
|
{
|
|
|
|
dict<RTLIL::IdString, RTLIL::Const> attributes;
|
|
|
|
|
2020-04-16 16:49:49 -05:00
|
|
|
bool has_attribute(RTLIL::IdString id) const;
|
|
|
|
|
2019-04-20 04:04:46 -05:00
|
|
|
void set_bool_attribute(RTLIL::IdString id, bool value=true);
|
2015-04-24 15:04:05 -05:00
|
|
|
bool get_bool_attribute(RTLIL::IdString id) const;
|
2017-08-30 04:39:11 -05:00
|
|
|
|
2019-04-18 12:19:45 -05:00
|
|
|
bool get_blackbox_attribute(bool ignore_wb=false) const {
|
2019-08-10 05:24:16 -05:00
|
|
|
return get_bool_attribute(ID::blackbox) || (!ignore_wb && get_bool_attribute(ID::whitebox));
|
2019-04-18 10:42:12 -05:00
|
|
|
}
|
|
|
|
|
2020-04-16 16:43:03 -05:00
|
|
|
void set_string_attribute(RTLIL::IdString id, string value);
|
|
|
|
string get_string_attribute(RTLIL::IdString id) const;
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
void set_strpool_attribute(RTLIL::IdString id, const pool<string> &data);
|
|
|
|
void add_strpool_attribute(RTLIL::IdString id, const pool<string> &data);
|
|
|
|
pool<string> get_strpool_attribute(RTLIL::IdString id) const;
|
2017-08-30 04:39:11 -05:00
|
|
|
|
2020-04-16 16:43:03 -05:00
|
|
|
void set_src_attribute(const std::string &src) {
|
|
|
|
set_string_attribute(ID::src, src);
|
|
|
|
}
|
|
|
|
std::string get_src_attribute() const {
|
|
|
|
return get_string_attribute(ID::src);
|
|
|
|
}
|
2020-06-04 05:46:54 -05:00
|
|
|
|
|
|
|
void set_hdlname_attribute(const vector<string> &hierarchy);
|
|
|
|
vector<string> get_hdlname_attribute() const;
|
2015-04-24 15:04:05 -05:00
|
|
|
};
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
struct RTLIL::SigChunk
|
|
|
|
{
|
|
|
|
RTLIL::Wire *wire;
|
|
|
|
std::vector<RTLIL::State> data; // only used if wire == NULL, LSB at index 0
|
|
|
|
int width, offset;
|
|
|
|
|
|
|
|
SigChunk();
|
|
|
|
SigChunk(const RTLIL::Const &value);
|
|
|
|
SigChunk(RTLIL::Wire *wire);
|
|
|
|
SigChunk(RTLIL::Wire *wire, int offset, int width = 1);
|
|
|
|
SigChunk(const std::string &str);
|
|
|
|
SigChunk(int val, int width = 32);
|
|
|
|
SigChunk(RTLIL::State bit, int width = 1);
|
2020-03-18 13:21:53 -05:00
|
|
|
SigChunk(const RTLIL::SigBit &bit);
|
2018-08-13 08:18:46 -05:00
|
|
|
SigChunk(const RTLIL::SigChunk &sigchunk);
|
2019-05-05 03:00:27 -05:00
|
|
|
RTLIL::SigChunk &operator =(const RTLIL::SigChunk &other) = default;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
RTLIL::SigChunk extract(int offset, int length) const;
|
2019-05-28 09:43:25 -05:00
|
|
|
inline int size() const { return width; }
|
2021-02-21 22:00:31 -06:00
|
|
|
inline bool is_wire() const { return wire != NULL; }
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
bool operator <(const RTLIL::SigChunk &other) const;
|
|
|
|
bool operator ==(const RTLIL::SigChunk &other) const;
|
|
|
|
bool operator !=(const RTLIL::SigChunk &other) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RTLIL::SigBit
|
|
|
|
{
|
|
|
|
RTLIL::Wire *wire;
|
|
|
|
union {
|
|
|
|
RTLIL::State data; // used if wire == NULL
|
|
|
|
int offset; // used if wire != NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
SigBit();
|
|
|
|
SigBit(RTLIL::State bit);
|
|
|
|
SigBit(bool bit);
|
|
|
|
SigBit(RTLIL::Wire *wire);
|
|
|
|
SigBit(RTLIL::Wire *wire, int offset);
|
|
|
|
SigBit(const RTLIL::SigChunk &chunk);
|
|
|
|
SigBit(const RTLIL::SigChunk &chunk, int index);
|
|
|
|
SigBit(const RTLIL::SigSpec &sig);
|
2020-05-22 10:59:24 -05:00
|
|
|
SigBit(const RTLIL::SigBit &sigbit) = default;
|
2019-05-05 03:00:27 -05:00
|
|
|
RTLIL::SigBit &operator =(const RTLIL::SigBit &other) = default;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
2021-02-21 22:00:31 -06:00
|
|
|
inline bool is_wire() const { return wire != NULL; }
|
2021-02-20 13:46:30 -06:00
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
bool operator <(const RTLIL::SigBit &other) const;
|
|
|
|
bool operator ==(const RTLIL::SigBit &other) const;
|
|
|
|
bool operator !=(const RTLIL::SigBit &other) const;
|
2014-12-26 14:35:22 -06:00
|
|
|
unsigned int hash() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
struct RTLIL::SigSpecIterator : public std::iterator<std::input_iterator_tag, RTLIL::SigSpec>
|
|
|
|
{
|
|
|
|
RTLIL::SigSpec *sig_p;
|
|
|
|
int index;
|
|
|
|
|
|
|
|
inline RTLIL::SigBit &operator*() const;
|
|
|
|
inline bool operator!=(const RTLIL::SigSpecIterator &other) const { return index != other.index; }
|
|
|
|
inline bool operator==(const RTLIL::SigSpecIterator &other) const { return index == other.index; }
|
|
|
|
inline void operator++() { index++; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RTLIL::SigSpecConstIterator : public std::iterator<std::input_iterator_tag, RTLIL::SigSpec>
|
|
|
|
{
|
|
|
|
const RTLIL::SigSpec *sig_p;
|
|
|
|
int index;
|
|
|
|
|
|
|
|
inline const RTLIL::SigBit &operator*() const;
|
|
|
|
inline bool operator!=(const RTLIL::SigSpecConstIterator &other) const { return index != other.index; }
|
|
|
|
inline bool operator==(const RTLIL::SigSpecIterator &other) const { return index == other.index; }
|
|
|
|
inline void operator++() { index++; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RTLIL::SigSpec
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
int width_;
|
|
|
|
unsigned long hash_;
|
|
|
|
std::vector<RTLIL::SigChunk> chunks_; // LSB at index 0
|
|
|
|
std::vector<RTLIL::SigBit> bits_; // LSB at index 0
|
|
|
|
|
|
|
|
void pack() const;
|
|
|
|
void unpack() const;
|
2014-12-26 14:35:22 -06:00
|
|
|
void updhash() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
inline bool packed() const {
|
|
|
|
return bits_.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void inline_unpack() const {
|
|
|
|
if (!chunks_.empty())
|
|
|
|
unpack();
|
|
|
|
}
|
|
|
|
|
2020-03-12 17:55:54 -05:00
|
|
|
// Only used by Module::remove(const pool<Wire*> &wires)
|
|
|
|
// but cannot be more specific as it isn't yet declared
|
|
|
|
friend struct RTLIL::Module;
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
public:
|
|
|
|
SigSpec();
|
|
|
|
SigSpec(const RTLIL::SigSpec &other);
|
|
|
|
SigSpec(std::initializer_list<RTLIL::SigSpec> parts);
|
2020-03-18 13:51:00 -05:00
|
|
|
RTLIL::SigSpec &operator=(const RTLIL::SigSpec &other);
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
SigSpec(const RTLIL::Const &value);
|
|
|
|
SigSpec(const RTLIL::SigChunk &chunk);
|
|
|
|
SigSpec(RTLIL::Wire *wire);
|
|
|
|
SigSpec(RTLIL::Wire *wire, int offset, int width = 1);
|
|
|
|
SigSpec(const std::string &str);
|
|
|
|
SigSpec(int val, int width = 32);
|
|
|
|
SigSpec(RTLIL::State bit, int width = 1);
|
2020-03-18 13:21:53 -05:00
|
|
|
SigSpec(const RTLIL::SigBit &bit, int width = 1);
|
|
|
|
SigSpec(const std::vector<RTLIL::SigChunk> &chunks);
|
|
|
|
SigSpec(const std::vector<RTLIL::SigBit> &bits);
|
|
|
|
SigSpec(const pool<RTLIL::SigBit> &bits);
|
|
|
|
SigSpec(const std::set<RTLIL::SigBit> &bits);
|
2014-12-26 03:53:21 -06:00
|
|
|
SigSpec(bool bit);
|
|
|
|
|
|
|
|
SigSpec(RTLIL::SigSpec &&other) {
|
|
|
|
width_ = other.width_;
|
|
|
|
hash_ = other.hash_;
|
|
|
|
chunks_ = std::move(other.chunks_);
|
|
|
|
bits_ = std::move(other.bits_);
|
|
|
|
}
|
|
|
|
|
|
|
|
const RTLIL::SigSpec &operator=(RTLIL::SigSpec &&other) {
|
|
|
|
width_ = other.width_;
|
|
|
|
hash_ = other.hash_;
|
|
|
|
chunks_ = std::move(other.chunks_);
|
|
|
|
bits_ = std::move(other.bits_);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t get_hash() const {
|
|
|
|
if (!hash_) hash();
|
|
|
|
return hash_;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const std::vector<RTLIL::SigChunk> &chunks() const { pack(); return chunks_; }
|
|
|
|
inline const std::vector<RTLIL::SigBit> &bits() const { inline_unpack(); return bits_; }
|
|
|
|
|
|
|
|
inline int size() const { return width_; }
|
|
|
|
inline bool empty() const { return width_ == 0; }
|
|
|
|
|
|
|
|
inline RTLIL::SigBit &operator[](int index) { inline_unpack(); return bits_.at(index); }
|
|
|
|
inline const RTLIL::SigBit &operator[](int index) const { inline_unpack(); return bits_.at(index); }
|
|
|
|
|
|
|
|
inline RTLIL::SigSpecIterator begin() { RTLIL::SigSpecIterator it; it.sig_p = this; it.index = 0; return it; }
|
|
|
|
inline RTLIL::SigSpecIterator end() { RTLIL::SigSpecIterator it; it.sig_p = this; it.index = width_; return it; }
|
|
|
|
|
|
|
|
inline RTLIL::SigSpecConstIterator begin() const { RTLIL::SigSpecConstIterator it; it.sig_p = this; it.index = 0; return it; }
|
|
|
|
inline RTLIL::SigSpecConstIterator end() const { RTLIL::SigSpecConstIterator it; it.sig_p = this; it.index = width_; return it; }
|
|
|
|
|
|
|
|
void sort();
|
|
|
|
void sort_and_unify();
|
|
|
|
|
|
|
|
void replace(const RTLIL::SigSpec &pattern, const RTLIL::SigSpec &with);
|
|
|
|
void replace(const RTLIL::SigSpec &pattern, const RTLIL::SigSpec &with, RTLIL::SigSpec *other) const;
|
|
|
|
|
|
|
|
void replace(const dict<RTLIL::SigBit, RTLIL::SigBit> &rules);
|
|
|
|
void replace(const dict<RTLIL::SigBit, RTLIL::SigBit> &rules, RTLIL::SigSpec *other) const;
|
|
|
|
|
|
|
|
void replace(const std::map<RTLIL::SigBit, RTLIL::SigBit> &rules);
|
|
|
|
void replace(const std::map<RTLIL::SigBit, RTLIL::SigBit> &rules, RTLIL::SigSpec *other) const;
|
|
|
|
|
|
|
|
void replace(int offset, const RTLIL::SigSpec &with);
|
|
|
|
|
|
|
|
void remove(const RTLIL::SigSpec &pattern);
|
|
|
|
void remove(const RTLIL::SigSpec &pattern, RTLIL::SigSpec *other) const;
|
|
|
|
void remove2(const RTLIL::SigSpec &pattern, RTLIL::SigSpec *other);
|
|
|
|
|
2014-12-26 14:59:41 -06:00
|
|
|
void remove(const pool<RTLIL::SigBit> &pattern);
|
|
|
|
void remove(const pool<RTLIL::SigBit> &pattern, RTLIL::SigSpec *other) const;
|
|
|
|
void remove2(const pool<RTLIL::SigBit> &pattern, RTLIL::SigSpec *other);
|
2016-01-30 00:03:12 -06:00
|
|
|
void remove2(const std::set<RTLIL::SigBit> &pattern, RTLIL::SigSpec *other);
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
void remove(int offset, int length = 1);
|
|
|
|
void remove_const();
|
|
|
|
|
|
|
|
RTLIL::SigSpec extract(const RTLIL::SigSpec &pattern, const RTLIL::SigSpec *other = NULL) const;
|
2014-12-26 14:59:41 -06:00
|
|
|
RTLIL::SigSpec extract(const pool<RTLIL::SigBit> &pattern, const RTLIL::SigSpec *other = NULL) const;
|
2014-12-26 03:53:21 -06:00
|
|
|
RTLIL::SigSpec extract(int offset, int length = 1) const;
|
2019-08-06 17:25:11 -05:00
|
|
|
RTLIL::SigSpec extract_end(int offset) const { return extract(offset, width_ - offset); }
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
void append(const RTLIL::SigSpec &signal);
|
2020-03-13 10:17:39 -05:00
|
|
|
inline void append(Wire *wire) { append(RTLIL::SigSpec(wire)); }
|
|
|
|
inline void append(const RTLIL::SigChunk &chunk) { append(RTLIL::SigSpec(chunk)); }
|
|
|
|
inline void append(const RTLIL::Const &const_) { append(RTLIL::SigSpec(const_)); }
|
|
|
|
|
|
|
|
void append(const RTLIL::SigBit &bit);
|
|
|
|
inline void append(RTLIL::State state) { append(RTLIL::SigBit(state)); }
|
|
|
|
inline void append(bool bool_) { append(RTLIL::SigBit(bool_)); }
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
void extend_u0(int width, bool is_signed = false);
|
|
|
|
|
|
|
|
RTLIL::SigSpec repeat(int num) const;
|
|
|
|
|
2020-01-28 12:37:16 -06:00
|
|
|
void reverse() { inline_unpack(); std::reverse(bits_.begin(), bits_.end()); }
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
bool operator <(const RTLIL::SigSpec &other) const;
|
|
|
|
bool operator ==(const RTLIL::SigSpec &other) const;
|
|
|
|
inline bool operator !=(const RTLIL::SigSpec &other) const { return !(*this == other); }
|
|
|
|
|
|
|
|
bool is_wire() const;
|
|
|
|
bool is_chunk() const;
|
2015-10-24 15:56:40 -05:00
|
|
|
inline bool is_bit() const { return width_ == 1; }
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
bool is_fully_const() const;
|
2015-04-29 00:28:15 -05:00
|
|
|
bool is_fully_zero() const;
|
2017-12-12 14:48:31 -06:00
|
|
|
bool is_fully_ones() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
bool is_fully_def() const;
|
|
|
|
bool is_fully_undef() const;
|
2015-02-07 17:01:51 -06:00
|
|
|
bool has_const() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
bool has_marked_bits() const;
|
2021-06-09 11:41:57 -05:00
|
|
|
bool is_onehot(int *pos = nullptr) const;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
bool as_bool() const;
|
|
|
|
int as_int(bool is_signed = false) const;
|
|
|
|
std::string as_string() const;
|
|
|
|
RTLIL::Const as_const() const;
|
|
|
|
RTLIL::Wire *as_wire() const;
|
|
|
|
RTLIL::SigChunk as_chunk() const;
|
2015-10-24 15:56:40 -05:00
|
|
|
RTLIL::SigBit as_bit() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
bool match(const char* pattern) const;
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
std::set<RTLIL::SigBit> to_sigbit_set() const;
|
2014-12-27 05:02:57 -06:00
|
|
|
pool<RTLIL::SigBit> to_sigbit_pool() const;
|
2014-12-26 03:53:21 -06:00
|
|
|
std::vector<RTLIL::SigBit> to_sigbit_vector() const;
|
|
|
|
std::map<RTLIL::SigBit, RTLIL::SigBit> to_sigbit_map(const RTLIL::SigSpec &other) const;
|
|
|
|
dict<RTLIL::SigBit, RTLIL::SigBit> to_sigbit_dict(const RTLIL::SigSpec &other) const;
|
|
|
|
|
|
|
|
static bool parse(RTLIL::SigSpec &sig, RTLIL::Module *module, std::string str);
|
|
|
|
static bool parse_sel(RTLIL::SigSpec &sig, RTLIL::Design *design, RTLIL::Module *module, std::string str);
|
|
|
|
static bool parse_rhs(const RTLIL::SigSpec &lhs, RTLIL::SigSpec &sig, RTLIL::Module *module, std::string str);
|
|
|
|
|
|
|
|
operator std::vector<RTLIL::SigChunk>() const { return chunks(); }
|
|
|
|
operator std::vector<RTLIL::SigBit>() const { return bits(); }
|
2020-03-18 13:21:53 -05:00
|
|
|
const RTLIL::SigBit &at(int offset, const RTLIL::SigBit &defval) { return offset < width_ ? (*this)[offset] : defval; }
|
2014-12-26 03:53:21 -06:00
|
|
|
|
2014-12-26 14:35:22 -06:00
|
|
|
unsigned int hash() const { if (!hash_) updhash(); return hash_; };
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
#ifndef NDEBUG
|
2021-09-10 09:51:34 -05:00
|
|
|
void check(Module *mod = nullptr) const;
|
2014-12-26 03:53:21 -06:00
|
|
|
#else
|
2021-09-10 09:51:34 -05:00
|
|
|
void check(Module *mod = nullptr) const { }
|
2014-12-26 03:53:21 -06:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
struct RTLIL::Selection
|
|
|
|
{
|
2013-01-05 04:13:26 -06:00
|
|
|
bool full_selection;
|
2014-12-26 14:59:41 -06:00
|
|
|
pool<RTLIL::IdString> selected_modules;
|
|
|
|
dict<RTLIL::IdString, pool<RTLIL::IdString>> selected_members;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
Selection(bool full = true) : full_selection(full) { }
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-06-02 10:53:30 -05:00
|
|
|
bool selected_module(RTLIL::IdString mod_name) const;
|
|
|
|
bool selected_whole_module(RTLIL::IdString mod_name) const;
|
|
|
|
bool selected_member(RTLIL::IdString mod_name, RTLIL::IdString memb_name) const;
|
2013-01-05 04:13:26 -06:00
|
|
|
void optimize(RTLIL::Design *design);
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-11-23 08:58:06 -06:00
|
|
|
template<typename T1> void select(T1 *module) {
|
|
|
|
if (!full_selection && selected_modules.count(module->name) == 0) {
|
|
|
|
selected_modules.insert(module->name);
|
|
|
|
selected_members.erase(module->name);
|
|
|
|
}
|
|
|
|
}
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-08-09 08:20:22 -05:00
|
|
|
template<typename T1, typename T2> void select(T1 *module, T2 *member) {
|
|
|
|
if (!full_selection && selected_modules.count(module->name) == 0)
|
|
|
|
selected_members[module->name].insert(member->name);
|
|
|
|
}
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2014-01-17 09:34:50 -06:00
|
|
|
bool empty() const {
|
|
|
|
return !full_selection && selected_modules.empty() && selected_members.empty();
|
|
|
|
}
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2014-07-31 07:34:12 -05:00
|
|
|
struct RTLIL::Monitor
|
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
|
|
|
Monitor() {
|
2014-12-28 17:12:36 -06:00
|
|
|
static unsigned int hashidx_count = 123456789;
|
|
|
|
hashidx_count = mkhash_xorshift(hashidx_count);
|
|
|
|
hashidx_ = hashidx_count;
|
2014-12-28 10:51:16 -06:00
|
|
|
}
|
|
|
|
|
2014-07-31 09:38:54 -05:00
|
|
|
virtual ~Monitor() { }
|
2014-07-31 07:34:12 -05:00
|
|
|
virtual void notify_module_add(RTLIL::Module*) { }
|
|
|
|
virtual void notify_module_del(RTLIL::Module*) { }
|
2020-03-18 13:21:53 -05:00
|
|
|
virtual void notify_connect(RTLIL::Cell*, const RTLIL::IdString&, const RTLIL::SigSpec&, const RTLIL::SigSpec&) { }
|
2014-07-31 07:34:12 -05:00
|
|
|
virtual void notify_connect(RTLIL::Module*, const RTLIL::SigSig&) { }
|
2014-08-01 09:53:15 -05:00
|
|
|
virtual void notify_connect(RTLIL::Module*, const std::vector<RTLIL::SigSig>&) { }
|
2014-07-31 07:34:12 -05:00
|
|
|
virtual void notify_blackout(RTLIL::Module*) { }
|
|
|
|
};
|
|
|
|
|
2020-03-17 04:34:31 -05:00
|
|
|
// Forward declaration; defined in preproc.h.
|
|
|
|
struct define_map_t;
|
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
struct RTLIL::Design
|
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
|
|
|
pool<RTLIL::Monitor*> monitors;
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<std::string, std::string> scratchpad;
|
2014-07-31 07:34:12 -05:00
|
|
|
|
2014-07-27 03:40:31 -05:00
|
|
|
int refcount_modules_;
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, RTLIL::Module*> modules_;
|
2020-04-20 10:06:53 -05:00
|
|
|
std::vector<RTLIL::Binding*> bindings_;
|
|
|
|
|
2016-11-15 05:42:43 -06:00
|
|
|
std::vector<AST::AstNode*> verilog_packages, verilog_globals;
|
2020-03-17 04:34:31 -05:00
|
|
|
std::unique_ptr<define_map_t> verilog_defines;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
std::vector<RTLIL::Selection> selection_stack;
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, RTLIL::Selection> selection_vars;
|
2013-01-05 04:13:26 -06:00
|
|
|
std::string selected_active_module;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2014-07-29 09:06:27 -05:00
|
|
|
Design();
|
2013-01-05 04:13:26 -06:00
|
|
|
~Design();
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2014-07-27 14:12:09 -05:00
|
|
|
RTLIL::ObjRange<RTLIL::Module*> modules();
|
|
|
|
RTLIL::Module *module(RTLIL::IdString name);
|
2021-03-24 10:23:23 -05:00
|
|
|
const RTLIL::Module *module(RTLIL::IdString name) const;
|
2015-06-17 02:38:56 -05:00
|
|
|
RTLIL::Module *top_module();
|
2014-07-27 14:12:09 -05:00
|
|
|
|
2014-07-29 09:06:27 -05:00
|
|
|
bool has(RTLIL::IdString id) const {
|
|
|
|
return modules_.count(id) != 0;
|
|
|
|
}
|
|
|
|
|
2014-07-27 14:12:09 -05:00
|
|
|
void add(RTLIL::Module *module);
|
2020-04-20 10:06:53 -05:00
|
|
|
void add(RTLIL::Binding *binding);
|
|
|
|
|
2014-07-27 14:12:09 -05:00
|
|
|
RTLIL::Module *addModule(RTLIL::IdString name);
|
|
|
|
void remove(RTLIL::Module *module);
|
2015-06-29 18:37:59 -05:00
|
|
|
void rename(RTLIL::Module *module, RTLIL::IdString new_name);
|
2014-07-27 03:40:31 -05:00
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
void scratchpad_unset(const std::string &varname);
|
2014-08-30 12:37:12 -05:00
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
void scratchpad_set_int(const std::string &varname, int value);
|
|
|
|
void scratchpad_set_bool(const std::string &varname, bool value);
|
2020-03-27 14:21:09 -05:00
|
|
|
void scratchpad_set_string(const std::string &varname, std::string value);
|
2014-08-30 12:37:12 -05:00
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
int scratchpad_get_int(const std::string &varname, int default_value = 0) const;
|
|
|
|
bool scratchpad_get_bool(const std::string &varname, bool default_value = false) const;
|
|
|
|
std::string scratchpad_get_string(const std::string &varname, const std::string &default_value = std::string()) const;
|
2014-08-30 12:37:12 -05:00
|
|
|
|
2015-01-23 17:13:27 -06:00
|
|
|
void sort();
|
2013-01-05 04:13:26 -06:00
|
|
|
void check();
|
|
|
|
void optimize();
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-06-02 10:53:30 -05:00
|
|
|
bool selected_module(RTLIL::IdString mod_name) const;
|
|
|
|
bool selected_whole_module(RTLIL::IdString mod_name) const;
|
|
|
|
bool selected_member(RTLIL::IdString mod_name, RTLIL::IdString memb_name) const;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2014-07-27 04:03:56 -05:00
|
|
|
bool selected_module(RTLIL::Module *mod) const;
|
|
|
|
bool selected_whole_module(RTLIL::Module *mod) const;
|
|
|
|
|
2015-02-24 15:31:30 -06:00
|
|
|
RTLIL::Selection &selection() {
|
|
|
|
return selection_stack.back();
|
|
|
|
}
|
|
|
|
|
|
|
|
const RTLIL::Selection &selection() const {
|
|
|
|
return selection_stack.back();
|
|
|
|
}
|
|
|
|
|
2013-10-27 03:30:58 -05:00
|
|
|
bool full_selection() const {
|
|
|
|
return selection_stack.back().full_selection;
|
|
|
|
}
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-06-02 10:53:30 -05:00
|
|
|
template<typename T1> bool selected(T1 *module) const {
|
2013-01-05 04:13:26 -06:00
|
|
|
return selected_module(module->name);
|
|
|
|
}
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-06-02 10:53:30 -05:00
|
|
|
template<typename T1, typename T2> bool selected(T1 *module, T2 *member) const {
|
2013-01-05 04:13:26 -06:00
|
|
|
return selected_member(module->name, member->name);
|
|
|
|
}
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2020-06-17 15:28:56 -05:00
|
|
|
template<typename T1> void select(T1 *module) {
|
|
|
|
if (selection_stack.size() > 0) {
|
|
|
|
RTLIL::Selection &sel = selection_stack.back();
|
|
|
|
sel.select(module);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-03 13:53:24 -06:00
|
|
|
template<typename T1, typename T2> void select(T1 *module, T2 *member) {
|
|
|
|
if (selection_stack.size() > 0) {
|
|
|
|
RTLIL::Selection &sel = selection_stack.back();
|
2013-08-09 08:20:22 -05:00
|
|
|
sel.select(module, member);
|
2013-03-03 13:53:24 -06:00
|
|
|
}
|
|
|
|
}
|
2014-07-31 07:11:39 -05:00
|
|
|
|
2018-07-09 08:48:06 -05:00
|
|
|
|
2014-07-31 07:11:39 -05:00
|
|
|
std::vector<RTLIL::Module*> selected_modules() const;
|
|
|
|
std::vector<RTLIL::Module*> selected_whole_modules() const;
|
2021-03-17 07:06:09 -05:00
|
|
|
std::vector<RTLIL::Module*> selected_whole_modules_warn(bool include_wb = false) const;
|
2018-08-13 08:18:46 -05:00
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
static std::map<unsigned int, RTLIL::Design*> *get_all_designs(void);
|
|
|
|
#endif
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::Module : public RTLIL::AttrObject
|
2014-07-25 08:05:18 -05:00
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
2014-07-26 14:16:05 -05:00
|
|
|
protected:
|
|
|
|
void add(RTLIL::Wire *wire);
|
|
|
|
void add(RTLIL::Cell *cell);
|
2021-07-11 16:57:53 -05:00
|
|
|
void add(RTLIL::Process *process);
|
2014-07-26 14:16:05 -05:00
|
|
|
|
|
|
|
public:
|
2014-07-31 07:11:39 -05:00
|
|
|
RTLIL::Design *design;
|
2014-12-28 10:51:16 -06:00
|
|
|
pool<RTLIL::Monitor*> monitors;
|
2014-07-31 07:34:12 -05:00
|
|
|
|
2014-07-27 03:13:22 -05:00
|
|
|
int refcount_wires_;
|
|
|
|
int refcount_cells_;
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, RTLIL::Wire*> wires_;
|
|
|
|
dict<RTLIL::IdString, RTLIL::Cell*> cells_;
|
2020-04-20 10:06:53 -05:00
|
|
|
|
|
|
|
std::vector<RTLIL::SigSig> connections_;
|
|
|
|
std::vector<RTLIL::Binding*> bindings_;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::IdString name;
|
2020-04-16 08:51:03 -05:00
|
|
|
idict<RTLIL::IdString> avail_parameters;
|
|
|
|
dict<RTLIL::IdString, RTLIL::Const> parameter_default_values;
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, RTLIL::Memory*> memories;
|
|
|
|
dict<RTLIL::IdString, RTLIL::Process*> processes;
|
2014-07-26 04:58:03 -05:00
|
|
|
|
2014-07-27 03:13:22 -05:00
|
|
|
Module();
|
2013-01-05 04:13:26 -06:00
|
|
|
virtual ~Module();
|
2020-03-18 13:21:53 -05:00
|
|
|
virtual RTLIL::IdString derive(RTLIL::Design *design, const dict<RTLIL::IdString, RTLIL::Const> ¶meters, bool mayfail = false);
|
|
|
|
virtual RTLIL::IdString derive(RTLIL::Design *design, const dict<RTLIL::IdString, RTLIL::Const> ¶meters, const dict<RTLIL::IdString, RTLIL::Module*> &interfaces, const dict<RTLIL::IdString, RTLIL::IdString> &modports, bool mayfail = false);
|
2013-01-05 04:13:26 -06:00
|
|
|
virtual size_t count_id(RTLIL::IdString id);
|
2020-03-18 13:21:53 -05:00
|
|
|
virtual void reprocess_module(RTLIL::Design *design, const dict<RTLIL::IdString, RTLIL::Module *> &local_interfaces);
|
2015-01-23 17:13:27 -06:00
|
|
|
|
|
|
|
virtual void sort();
|
2013-01-05 04:13:26 -06:00
|
|
|
virtual void check();
|
|
|
|
virtual void optimize();
|
2019-04-05 10:31:49 -05:00
|
|
|
virtual void makeblackbox();
|
2014-07-26 04:58:03 -05:00
|
|
|
|
2014-07-26 07:31:47 -05:00
|
|
|
void connect(const RTLIL::SigSig &conn);
|
2014-07-26 04:58:03 -05:00
|
|
|
void connect(const RTLIL::SigSpec &lhs, const RTLIL::SigSpec &rhs);
|
2014-07-31 07:34:12 -05:00
|
|
|
void new_connections(const std::vector<RTLIL::SigSig> &new_conn);
|
2014-07-26 08:57:27 -05:00
|
|
|
const std::vector<RTLIL::SigSig> &connections() const;
|
2014-08-14 09:13:42 -05:00
|
|
|
|
|
|
|
std::vector<RTLIL::IdString> ports;
|
2013-06-18 10:11:13 -05:00
|
|
|
void fixup_ports();
|
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2013-07-27 07:27:51 -05:00
|
|
|
void cloneInto(RTLIL::Module *new_mod) const;
|
|
|
|
virtual RTLIL::Module *clone() const;
|
2013-10-24 03:59:27 -05:00
|
|
|
|
2014-07-31 07:11:39 -05:00
|
|
|
bool has_memories() const;
|
|
|
|
bool has_processes() const;
|
|
|
|
|
|
|
|
bool has_memories_warn() const;
|
|
|
|
bool has_processes_warn() const;
|
|
|
|
|
|
|
|
std::vector<RTLIL::Wire*> selected_wires() const;
|
|
|
|
std::vector<RTLIL::Cell*> selected_cells() const;
|
|
|
|
|
2014-08-05 07:47:03 -05:00
|
|
|
template<typename T> bool selected(T *member) const {
|
|
|
|
return design->selected_member(name, member->name);
|
|
|
|
}
|
|
|
|
|
2020-05-26 09:19:39 -05:00
|
|
|
RTLIL::Wire* wire(RTLIL::IdString id) {
|
|
|
|
auto it = wires_.find(id);
|
|
|
|
return it == wires_.end() ? nullptr : it->second;
|
|
|
|
}
|
|
|
|
RTLIL::Cell* cell(RTLIL::IdString id) {
|
|
|
|
auto it = cells_.find(id);
|
|
|
|
return it == cells_.end() ? nullptr : it->second;
|
|
|
|
}
|
2014-07-27 04:03:56 -05:00
|
|
|
|
2021-03-24 10:23:23 -05:00
|
|
|
const RTLIL::Wire* wire(RTLIL::IdString id) const{
|
|
|
|
auto it = wires_.find(id);
|
|
|
|
return it == wires_.end() ? nullptr : it->second;
|
|
|
|
}
|
|
|
|
const RTLIL::Cell* cell(RTLIL::IdString id) const {
|
|
|
|
auto it = cells_.find(id);
|
|
|
|
return it == cells_.end() ? nullptr : it->second;
|
|
|
|
}
|
|
|
|
|
2014-07-27 03:13:22 -05:00
|
|
|
RTLIL::ObjRange<RTLIL::Wire*> wires() { return RTLIL::ObjRange<RTLIL::Wire*>(&wires_, &refcount_wires_); }
|
|
|
|
RTLIL::ObjRange<RTLIL::Cell*> cells() { return RTLIL::ObjRange<RTLIL::Cell*>(&cells_, &refcount_cells_); }
|
|
|
|
|
2020-04-20 10:06:53 -05:00
|
|
|
void add(RTLIL::Binding *binding);
|
|
|
|
|
2014-07-26 13:12:50 -05:00
|
|
|
// Removing wires is expensive. If you have to remove wires, remove them all at once.
|
2014-12-28 10:51:16 -06:00
|
|
|
void remove(const pool<RTLIL::Wire*> &wires);
|
2014-07-21 05:35:06 -05:00
|
|
|
void remove(RTLIL::Cell *cell);
|
2021-07-11 16:57:53 -05:00
|
|
|
void remove(RTLIL::Process *process);
|
2014-07-21 05:35:06 -05:00
|
|
|
|
2014-07-25 08:05:18 -05:00
|
|
|
void rename(RTLIL::Wire *wire, RTLIL::IdString new_name);
|
|
|
|
void rename(RTLIL::Cell *cell, RTLIL::IdString new_name);
|
|
|
|
void rename(RTLIL::IdString old_name, RTLIL::IdString new_name);
|
|
|
|
|
2014-08-05 07:47:03 -05:00
|
|
|
void swap_names(RTLIL::Wire *w1, RTLIL::Wire *w2);
|
|
|
|
void swap_names(RTLIL::Cell *c1, RTLIL::Cell *c2);
|
|
|
|
|
2014-08-16 16:50:36 -05:00
|
|
|
RTLIL::IdString uniquify(RTLIL::IdString name);
|
|
|
|
RTLIL::IdString uniquify(RTLIL::IdString name, int &index);
|
|
|
|
|
2014-07-21 05:02:55 -05:00
|
|
|
RTLIL::Wire *addWire(RTLIL::IdString name, int width = 1);
|
2014-07-26 14:16:05 -05:00
|
|
|
RTLIL::Wire *addWire(RTLIL::IdString name, const RTLIL::Wire *other);
|
|
|
|
|
2014-07-21 05:02:55 -05:00
|
|
|
RTLIL::Cell *addCell(RTLIL::IdString name, RTLIL::IdString type);
|
2014-07-25 17:38:44 -05:00
|
|
|
RTLIL::Cell *addCell(RTLIL::IdString name, const RTLIL::Cell *other);
|
2014-07-21 05:02:55 -05:00
|
|
|
|
2020-06-03 09:35:27 -05:00
|
|
|
RTLIL::Memory *addMemory(RTLIL::IdString name, const RTLIL::Memory *other);
|
|
|
|
|
2021-07-11 16:57:53 -05:00
|
|
|
RTLIL::Process *addProcess(RTLIL::IdString name);
|
2020-06-09 04:55:48 -05:00
|
|
|
RTLIL::Process *addProcess(RTLIL::IdString name, const RTLIL::Process *other);
|
|
|
|
|
2014-07-18 03:27:06 -05:00
|
|
|
// The add* methods create a cell and return the created cell. All signals must exist in advance.
|
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addNot (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addPos (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNeg (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addAnd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addOr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addXor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addXnor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addReduceAnd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addReduceOr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addReduceXor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addReduceXnor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addReduceBool (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addShl (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addShr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSshl (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSshr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addShift (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addShiftx (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addLt (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addLe (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addEq (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNe (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addEqx (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNex (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addGe (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addGt (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addAdd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSub (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addMul (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
2020-05-02 04:24:19 -05:00
|
|
|
// truncating division
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addDiv (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
2020-05-02 04:24:19 -05:00
|
|
|
// truncating modulo
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addMod (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
2020-04-21 05:51:58 -05:00
|
|
|
RTLIL::Cell* addDivFloor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
2020-04-08 12:30:47 -05:00
|
|
|
RTLIL::Cell* addModFloor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addPow (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool a_signed = false, bool b_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addLogicNot (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addLogicAnd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addLogicOr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addMux (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_s, const RTLIL::SigSpec &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addPmux (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_s, const RTLIL::SigSpec &sig_y, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addSlice (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, RTLIL::Const offset, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addConcat (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addLut (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_y, RTLIL::Const lut, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addTribuf (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAssert (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAssume (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addLive (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addFair (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addCover (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_en, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addEquiv (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_y, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::Cell* addSr (RTLIL::IdString name, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr, const RTLIL::SigSpec &sig_q, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addFf (RTLIL::IdString name, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addDff (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addDffe (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool en_polarity = true, const std::string &src = "");
|
2020-05-19 04:13:49 -05:00
|
|
|
RTLIL::Cell* addDffsr (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr, RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addDffsre (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr, RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool en_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-05-19 04:13:49 -05:00
|
|
|
RTLIL::Cell* addAdff (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const arst_value, bool clk_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addAdffe (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const arst_value, bool clk_polarity = true, bool en_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdff (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const srst_value, bool clk_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdffe (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const srst_value, bool clk_polarity = true, bool en_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdffce (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const srst_value, bool clk_polarity = true, bool en_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addDlatch (RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool en_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addAdlatch (RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, RTLIL::Const arst_value, bool en_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
2020-05-19 04:13:49 -05:00
|
|
|
RTLIL::Cell* addDlatchsr (RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr, RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool en_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
|
|
|
|
RTLIL::Cell* addBufGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAndGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNandGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addOrGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addXorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addXnorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAndnotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addOrnotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addMuxGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_s, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addNmuxGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_s, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAoi3Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addOai3Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addAoi4Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_d, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addOai4Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_d, const RTLIL::SigBit &sig_y, const std::string &src = "");
|
|
|
|
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addSrGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr,
|
|
|
|
const RTLIL::SigSpec &sig_q, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addFfGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addDffGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addDffeGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool en_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addDffsrGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr,
|
|
|
|
RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addDffsreGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr,
|
|
|
|
RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool clk_polarity = true, bool en_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addAdffGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
2017-09-09 03:16:48 -05:00
|
|
|
bool arst_value = false, bool clk_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addAdffeGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
|
|
|
bool arst_value = false, bool clk_polarity = true, bool en_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdffGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
|
|
|
bool srst_value = false, bool clk_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdffeGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
|
|
|
bool srst_value = false, bool clk_polarity = true, bool en_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
|
|
|
RTLIL::Cell* addSdffceGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_clk, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_srst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
|
|
|
bool srst_value = false, bool clk_polarity = true, bool en_polarity = true, bool srst_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addDlatchGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q, bool en_polarity = true, const std::string &src = "");
|
2020-06-23 08:39:25 -05:00
|
|
|
RTLIL::Cell* addAdlatchGate(RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_arst, const RTLIL::SigSpec &sig_d, const RTLIL::SigSpec &sig_q,
|
|
|
|
bool arst_value = false, bool en_polarity = true, bool arst_polarity = true, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::Cell* addDlatchsrGate (RTLIL::IdString name, const RTLIL::SigSpec &sig_en, const RTLIL::SigSpec &sig_set, const RTLIL::SigSpec &sig_clr,
|
|
|
|
RTLIL::SigSpec sig_d, const RTLIL::SigSpec &sig_q, bool en_polarity = true, bool set_polarity = true, bool clr_polarity = true, const std::string &src = "");
|
2014-07-18 03:27:06 -05:00
|
|
|
|
|
|
|
// The methods without the add* prefix create a cell and an output signal. They return the newly created output signal.
|
|
|
|
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::SigSpec Not (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Pos (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Neg (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec And (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Or (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Xor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Xnor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec ReduceAnd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec ReduceOr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec ReduceXor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec ReduceXnor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec ReduceBool (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec Shl (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Shr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Sshl (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Sshr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Shift (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Shiftx (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec Lt (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Le (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Eq (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Ne (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Eqx (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Nex (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Ge (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Gt (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec Add (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Sub (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Mul (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
2020-05-02 04:24:19 -05:00
|
|
|
// truncating division
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::SigSpec Div (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
2020-05-02 04:24:19 -05:00
|
|
|
// truncating modulo
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::SigSpec Mod (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
2020-04-21 05:51:58 -05:00
|
|
|
RTLIL::SigSpec DivFloor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
2020-04-08 12:30:47 -05:00
|
|
|
RTLIL::SigSpec ModFloor (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
2020-03-18 13:21:53 -05:00
|
|
|
RTLIL::SigSpec Pow (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool a_signed = false, bool b_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec LogicNot (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec LogicAnd (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec LogicOr (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, bool is_signed = false, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigSpec Mux (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_s, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Pmux (RTLIL::IdString name, const RTLIL::SigSpec &sig_a, const RTLIL::SigSpec &sig_b, const RTLIL::SigSpec &sig_s, const std::string &src = "");
|
|
|
|
|
|
|
|
RTLIL::SigBit BufGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const std::string &src = "");
|
|
|
|
RTLIL::SigBit NotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const std::string &src = "");
|
|
|
|
RTLIL::SigBit AndGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit NandGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit OrGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit NorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit XorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit XnorGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit AndnotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit OrnotGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const std::string &src = "");
|
|
|
|
RTLIL::SigBit MuxGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_s, const std::string &src = "");
|
|
|
|
RTLIL::SigBit NmuxGate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_s, const std::string &src = "");
|
|
|
|
RTLIL::SigBit Aoi3Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const std::string &src = "");
|
|
|
|
RTLIL::SigBit Oai3Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const std::string &src = "");
|
|
|
|
RTLIL::SigBit Aoi4Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_d, const std::string &src = "");
|
|
|
|
RTLIL::SigBit Oai4Gate (RTLIL::IdString name, const RTLIL::SigBit &sig_a, const RTLIL::SigBit &sig_b, const RTLIL::SigBit &sig_c, const RTLIL::SigBit &sig_d, const std::string &src = "");
|
2017-09-09 03:16:48 -05:00
|
|
|
|
|
|
|
RTLIL::SigSpec Anyconst (RTLIL::IdString name, int width = 1, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Anyseq (RTLIL::IdString name, int width = 1, const std::string &src = "");
|
2018-02-23 06:14:47 -06:00
|
|
|
RTLIL::SigSpec Allconst (RTLIL::IdString name, int width = 1, const std::string &src = "");
|
|
|
|
RTLIL::SigSpec Allseq (RTLIL::IdString name, int width = 1, const std::string &src = "");
|
2017-09-09 03:16:48 -05:00
|
|
|
RTLIL::SigSpec Initstate (RTLIL::IdString name, const std::string &src = "");
|
2018-07-09 08:48:06 -05:00
|
|
|
|
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
static std::map<unsigned int, RTLIL::Module*> *get_all_modules(void);
|
|
|
|
#endif
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::Wire : public RTLIL::AttrObject
|
2014-07-26 04:58:03 -05:00
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
2014-07-26 14:16:05 -05:00
|
|
|
protected:
|
2014-07-26 04:58:03 -05:00
|
|
|
// use module->addWire() and module->remove() to create or destroy wires
|
|
|
|
friend struct RTLIL::Module;
|
|
|
|
Wire();
|
2018-07-10 01:52:36 -05:00
|
|
|
~Wire();
|
2014-07-26 04:58:03 -05:00
|
|
|
|
|
|
|
public:
|
|
|
|
// do not simply copy wires
|
2014-07-26 14:16:05 -05:00
|
|
|
Wire(RTLIL::Wire &other) = delete;
|
|
|
|
void operator=(RTLIL::Wire &other) = delete;
|
2014-07-26 04:58:03 -05:00
|
|
|
|
2014-07-31 07:11:39 -05:00
|
|
|
RTLIL::Module *module;
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::IdString name;
|
|
|
|
int width, start_offset, port_id;
|
2020-04-27 11:44:24 -05:00
|
|
|
bool port_input, port_output, upto, is_signed;
|
2018-07-09 08:48:06 -05:00
|
|
|
|
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
static std::map<unsigned int, RTLIL::Wire*> *get_all_wires(void);
|
|
|
|
#endif
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::Memory : public RTLIL::AttrObject
|
2014-07-26 04:58:03 -05:00
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
2014-07-26 04:58:03 -05:00
|
|
|
Memory();
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::IdString name;
|
|
|
|
int width, start_offset, size;
|
2018-08-13 08:18:46 -05:00
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
~Memory();
|
|
|
|
static std::map<unsigned int, RTLIL::Memory*> *get_all_memorys(void);
|
|
|
|
#endif
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::Cell : public RTLIL::AttrObject
|
2014-07-25 08:05:18 -05:00
|
|
|
{
|
2014-12-28 10:51:16 -06:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
2014-07-25 08:05:18 -05:00
|
|
|
protected:
|
2014-07-26 04:58:03 -05:00
|
|
|
// use module->addCell() and module->remove() to create or destroy cells
|
2014-07-25 08:05:18 -05:00
|
|
|
friend struct RTLIL::Module;
|
2014-08-22 09:09:13 -05:00
|
|
|
Cell();
|
2018-07-10 01:52:36 -05:00
|
|
|
~Cell();
|
2014-07-25 08:05:18 -05:00
|
|
|
|
|
|
|
public:
|
2014-07-26 04:58:03 -05:00
|
|
|
// do not simply copy cells
|
2014-07-25 08:05:18 -05:00
|
|
|
Cell(RTLIL::Cell &other) = delete;
|
|
|
|
void operator=(RTLIL::Cell &other) = delete;
|
|
|
|
|
2014-07-31 07:11:39 -05:00
|
|
|
RTLIL::Module *module;
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::IdString name;
|
|
|
|
RTLIL::IdString type;
|
2014-12-26 03:53:21 -06:00
|
|
|
dict<RTLIL::IdString, RTLIL::SigSpec> connections_;
|
|
|
|
dict<RTLIL::IdString, RTLIL::Const> parameters;
|
2014-07-26 05:22:58 -05:00
|
|
|
|
|
|
|
// access cell ports
|
2014-07-31 09:38:54 -05:00
|
|
|
bool hasPort(RTLIL::IdString portname) const;
|
|
|
|
void unsetPort(RTLIL::IdString portname);
|
|
|
|
void setPort(RTLIL::IdString portname, RTLIL::SigSpec signal);
|
|
|
|
const RTLIL::SigSpec &getPort(RTLIL::IdString portname) const;
|
2014-12-26 03:53:21 -06:00
|
|
|
const dict<RTLIL::IdString, RTLIL::SigSpec> &connections() const;
|
2014-07-26 05:22:58 -05:00
|
|
|
|
2015-02-07 04:40:19 -06:00
|
|
|
// information about cell ports
|
|
|
|
bool known() const;
|
|
|
|
bool input(RTLIL::IdString portname) const;
|
|
|
|
bool output(RTLIL::IdString portname) const;
|
|
|
|
|
2014-07-31 09:38:54 -05:00
|
|
|
// access cell parameters
|
2014-08-31 10:07:07 -05:00
|
|
|
bool hasParam(RTLIL::IdString paramname) const;
|
|
|
|
void unsetParam(RTLIL::IdString paramname);
|
|
|
|
void setParam(RTLIL::IdString paramname, RTLIL::Const value);
|
|
|
|
const RTLIL::Const &getParam(RTLIL::IdString paramname) const;
|
2014-07-31 09:38:54 -05:00
|
|
|
|
2015-01-23 17:13:27 -06:00
|
|
|
void sort();
|
2014-07-21 05:02:55 -05:00
|
|
|
void check();
|
2014-07-26 05:22:58 -05:00
|
|
|
void fixup_parameters(bool set_a_signed = false, bool set_b_signed = false);
|
2013-06-18 10:11:13 -05:00
|
|
|
|
2014-09-29 05:51:54 -05:00
|
|
|
bool has_keep_attr() const {
|
2019-08-10 05:24:16 -05:00
|
|
|
return get_bool_attribute(ID::keep) || (module && module->design && module->design->module(type) &&
|
|
|
|
module->design->module(type)->get_bool_attribute(ID::keep));
|
2014-09-29 05:51:54 -05:00
|
|
|
}
|
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2018-07-09 08:48:06 -05:00
|
|
|
|
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
static std::map<unsigned int, RTLIL::Cell*> *get_all_cells(void);
|
|
|
|
#endif
|
2021-05-22 12:14:13 -05:00
|
|
|
|
|
|
|
bool has_memid() const;
|
|
|
|
bool is_mem_cell() const;
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
Allow attributes on individual switch cases in RTLIL.
The parser changes are slightly awkward. Consider the following IL:
process $0
<point 1>
switch \foo
<point 2>
case 1'1
assign \bar \baz
<point 3>
...
case
end
end
Before this commit, attributes are valid in <point 1>, and <point 3>
iff it is immediately followed by a `switch`. (They are essentially
attached to the switch.) But, after this commit, and because switch
cases do not have an ending delimiter, <point 3> becomes ambiguous:
the attribute could attach to either the following `case`, or to
the following `switch`. This isn't expressible in LALR(1) and results
in a reduce/reduce conflict.
To address this, attributes inside processes are now valid anywhere
inside the process: in <point 1> and <point 3> a part of case body,
and in <point 2> as a separate rule. As a consequence, attributes
can now precede `assign`s, which is made illegal in the same way it
is illegal to attach attributes to `connect`.
Attributes are tracked separately from the parser state, so this
does not affect collection of attributes at all, other than allowing
them on `case`s. The grammar change serves purely to allow attributes
in more syntactic places.
2019-07-08 06:34:58 -05:00
|
|
|
struct RTLIL::CaseRule : public RTLIL::AttrObject
|
2014-07-26 19:00:04 -05:00
|
|
|
{
|
2013-01-05 04:13:26 -06:00
|
|
|
std::vector<RTLIL::SigSpec> compare;
|
|
|
|
std::vector<RTLIL::SigSig> actions;
|
|
|
|
std::vector<RTLIL::SwitchRule*> switches;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
~CaseRule();
|
2013-06-18 10:11:13 -05:00
|
|
|
|
2018-12-23 03:04:23 -06:00
|
|
|
bool empty() const;
|
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2013-07-27 07:27:51 -05:00
|
|
|
RTLIL::CaseRule *clone() const;
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::SwitchRule : public RTLIL::AttrObject
|
2014-07-26 19:00:04 -05:00
|
|
|
{
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::SigSpec signal;
|
|
|
|
std::vector<RTLIL::CaseRule*> cases;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
~SwitchRule();
|
2013-06-18 10:11:13 -05:00
|
|
|
|
2018-12-23 03:04:23 -06:00
|
|
|
bool empty() const;
|
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2013-07-27 07:27:51 -05:00
|
|
|
RTLIL::SwitchRule *clone() const;
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2021-02-22 17:21:46 -06:00
|
|
|
struct RTLIL::MemWriteAction : RTLIL::AttrObject
|
|
|
|
{
|
|
|
|
RTLIL::IdString memid;
|
|
|
|
RTLIL::SigSpec address;
|
|
|
|
RTLIL::SigSpec data;
|
|
|
|
RTLIL::SigSpec enable;
|
|
|
|
RTLIL::Const priority_mask;
|
|
|
|
};
|
|
|
|
|
2014-07-26 19:00:04 -05:00
|
|
|
struct RTLIL::SyncRule
|
|
|
|
{
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::SyncType type;
|
|
|
|
RTLIL::SigSpec signal;
|
|
|
|
std::vector<RTLIL::SigSig> actions;
|
2021-02-22 17:21:46 -06:00
|
|
|
std::vector<RTLIL::MemWriteAction> mem_write_actions;
|
2013-06-18 10:11:13 -05:00
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2013-07-27 07:27:51 -05:00
|
|
|
RTLIL::SyncRule *clone() const;
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2015-04-24 15:04:05 -05:00
|
|
|
struct RTLIL::Process : public RTLIL::AttrObject
|
2014-07-26 19:00:04 -05:00
|
|
|
{
|
2021-07-11 16:57:53 -05:00
|
|
|
unsigned int hashidx_;
|
|
|
|
unsigned int hash() const { return hashidx_; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
// use module->addProcess() and module->remove() to create or destroy processes
|
|
|
|
friend struct RTLIL::Module;
|
|
|
|
Process();
|
|
|
|
~Process();
|
|
|
|
|
|
|
|
public:
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::IdString name;
|
2021-07-11 16:57:53 -05:00
|
|
|
RTLIL::Module *module;
|
2013-01-05 04:13:26 -06:00
|
|
|
RTLIL::CaseRule root_case;
|
|
|
|
std::vector<RTLIL::SyncRule*> syncs;
|
2014-07-26 19:00:04 -05:00
|
|
|
|
2017-05-28 04:59:05 -05:00
|
|
|
template<typename T> void rewrite_sigspecs(T &functor);
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T> void rewrite_sigspecs2(T &functor);
|
2013-07-27 07:27:51 -05:00
|
|
|
RTLIL::Process *clone() const;
|
2013-01-05 04:13:26 -06:00
|
|
|
};
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
|
|
|
|
inline RTLIL::SigBit::SigBit() : wire(NULL), data(RTLIL::State::S0) { }
|
|
|
|
inline RTLIL::SigBit::SigBit(RTLIL::State bit) : wire(NULL), data(bit) { }
|
2019-08-07 13:14:03 -05:00
|
|
|
inline RTLIL::SigBit::SigBit(bool bit) : wire(NULL), data(bit ? State::S1 : State::S0) { }
|
2014-12-26 03:53:21 -06:00
|
|
|
inline RTLIL::SigBit::SigBit(RTLIL::Wire *wire) : wire(wire), offset(0) { log_assert(wire && wire->width == 1); }
|
|
|
|
inline RTLIL::SigBit::SigBit(RTLIL::Wire *wire, int offset) : wire(wire), offset(offset) { log_assert(wire != nullptr); }
|
|
|
|
inline RTLIL::SigBit::SigBit(const RTLIL::SigChunk &chunk) : wire(chunk.wire) { log_assert(chunk.width == 1); if (wire) offset = chunk.offset; else data = chunk.data[0]; }
|
|
|
|
inline RTLIL::SigBit::SigBit(const RTLIL::SigChunk &chunk, int index) : wire(chunk.wire) { if (wire) offset = chunk.offset + index; else data = chunk.data[index]; }
|
|
|
|
|
|
|
|
inline bool RTLIL::SigBit::operator<(const RTLIL::SigBit &other) const {
|
|
|
|
if (wire == other.wire)
|
|
|
|
return wire ? (offset < other.offset) : (data < other.data);
|
|
|
|
if (wire != nullptr && other.wire != nullptr)
|
|
|
|
return wire->name < other.wire->name;
|
2019-03-11 14:12:28 -05:00
|
|
|
return (wire != nullptr) < (other.wire != nullptr);
|
2014-12-26 03:53:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool RTLIL::SigBit::operator==(const RTLIL::SigBit &other) const {
|
|
|
|
return (wire == other.wire) && (wire ? (offset == other.offset) : (data == other.data));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool RTLIL::SigBit::operator!=(const RTLIL::SigBit &other) const {
|
|
|
|
return (wire != other.wire) || (wire ? (offset != other.offset) : (data != other.data));
|
|
|
|
}
|
|
|
|
|
2014-12-26 14:35:22 -06:00
|
|
|
inline unsigned int RTLIL::SigBit::hash() const {
|
|
|
|
if (wire)
|
2014-12-27 05:02:57 -06:00
|
|
|
return mkhash_add(wire->name.hash(), offset);
|
2014-12-26 14:35:22 -06:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2014-12-26 03:53:21 -06:00
|
|
|
inline RTLIL::SigBit &RTLIL::SigSpecIterator::operator*() const {
|
|
|
|
return (*sig_p)[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const RTLIL::SigBit &RTLIL::SigSpecConstIterator::operator*() const {
|
|
|
|
return (*sig_p)[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline RTLIL::SigBit::SigBit(const RTLIL::SigSpec &sig) {
|
|
|
|
log_assert(sig.size() == 1 && sig.chunks().size() == 1);
|
|
|
|
*this = SigBit(sig.chunks().front());
|
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::Module::rewrite_sigspecs(T &functor)
|
2013-06-18 12:54:33 -05:00
|
|
|
{
|
2014-07-26 18:51:45 -05:00
|
|
|
for (auto &it : cells_)
|
2013-06-18 12:54:33 -05:00
|
|
|
it.second->rewrite_sigspecs(functor);
|
|
|
|
for (auto &it : processes)
|
|
|
|
it.second->rewrite_sigspecs(functor);
|
2014-07-26 04:58:03 -05:00
|
|
|
for (auto &it : connections_) {
|
2013-06-18 12:54:33 -05:00
|
|
|
functor(it.first);
|
|
|
|
functor(it.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::Module::rewrite_sigspecs2(T &functor)
|
|
|
|
{
|
|
|
|
for (auto &it : cells_)
|
|
|
|
it.second->rewrite_sigspecs2(functor);
|
|
|
|
for (auto &it : processes)
|
|
|
|
it.second->rewrite_sigspecs2(functor);
|
|
|
|
for (auto &it : connections_) {
|
|
|
|
functor(it.first, it.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::Cell::rewrite_sigspecs(T &functor) {
|
2014-07-26 04:58:03 -05:00
|
|
|
for (auto &it : connections_)
|
2013-06-18 12:54:33 -05:00
|
|
|
functor(it.second);
|
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::Cell::rewrite_sigspecs2(T &functor) {
|
|
|
|
for (auto &it : connections_)
|
|
|
|
functor(it.second);
|
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::CaseRule::rewrite_sigspecs(T &functor) {
|
2013-06-18 12:54:33 -05:00
|
|
|
for (auto &it : compare)
|
|
|
|
functor(it);
|
|
|
|
for (auto &it : actions) {
|
|
|
|
functor(it.first);
|
|
|
|
functor(it.second);
|
|
|
|
}
|
|
|
|
for (auto it : switches)
|
|
|
|
it->rewrite_sigspecs(functor);
|
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::CaseRule::rewrite_sigspecs2(T &functor) {
|
|
|
|
for (auto &it : compare)
|
|
|
|
functor(it);
|
|
|
|
for (auto &it : actions) {
|
|
|
|
functor(it.first, it.second);
|
|
|
|
}
|
|
|
|
for (auto it : switches)
|
|
|
|
it->rewrite_sigspecs2(functor);
|
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::SwitchRule::rewrite_sigspecs(T &functor)
|
2013-06-18 12:54:33 -05:00
|
|
|
{
|
|
|
|
functor(signal);
|
|
|
|
for (auto it : cases)
|
|
|
|
it->rewrite_sigspecs(functor);
|
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::SwitchRule::rewrite_sigspecs2(T &functor)
|
|
|
|
{
|
|
|
|
functor(signal);
|
|
|
|
for (auto it : cases)
|
|
|
|
it->rewrite_sigspecs2(functor);
|
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::SyncRule::rewrite_sigspecs(T &functor)
|
2013-06-18 12:54:33 -05:00
|
|
|
{
|
|
|
|
functor(signal);
|
|
|
|
for (auto &it : actions) {
|
|
|
|
functor(it.first);
|
|
|
|
functor(it.second);
|
|
|
|
}
|
2021-02-22 17:21:46 -06:00
|
|
|
for (auto &it : mem_write_actions) {
|
|
|
|
functor(it.address);
|
|
|
|
functor(it.data);
|
|
|
|
functor(it.enable);
|
|
|
|
}
|
2013-06-18 12:54:33 -05:00
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::SyncRule::rewrite_sigspecs2(T &functor)
|
|
|
|
{
|
|
|
|
functor(signal);
|
|
|
|
for (auto &it : actions) {
|
|
|
|
functor(it.first, it.second);
|
|
|
|
}
|
2021-02-22 17:21:46 -06:00
|
|
|
for (auto &it : mem_write_actions) {
|
|
|
|
functor(it.address);
|
|
|
|
functor(it.data);
|
|
|
|
functor(it.enable);
|
|
|
|
}
|
2019-05-15 09:01:00 -05:00
|
|
|
}
|
|
|
|
|
2013-06-18 12:54:33 -05:00
|
|
|
template<typename T>
|
2017-05-28 04:59:05 -05:00
|
|
|
void RTLIL::Process::rewrite_sigspecs(T &functor)
|
2013-06-18 12:54:33 -05:00
|
|
|
{
|
|
|
|
root_case.rewrite_sigspecs(functor);
|
|
|
|
for (auto it : syncs)
|
|
|
|
it->rewrite_sigspecs(functor);
|
|
|
|
}
|
|
|
|
|
2019-05-15 09:01:00 -05:00
|
|
|
template<typename T>
|
|
|
|
void RTLIL::Process::rewrite_sigspecs2(T &functor)
|
|
|
|
{
|
|
|
|
root_case.rewrite_sigspecs2(functor);
|
|
|
|
for (auto it : syncs)
|
|
|
|
it->rewrite_sigspecs2(functor);
|
|
|
|
}
|
|
|
|
|
2014-07-31 06:19:47 -05:00
|
|
|
YOSYS_NAMESPACE_END
|
|
|
|
|
2013-01-05 04:13:26 -06:00
|
|
|
#endif
|