diff options
Diffstat (limited to 'kernel')
| -rw-r--r-- | kernel/constids.inc | 1 | ||||
| -rw-r--r-- | kernel/rtlil.cc | 29 | ||||
| -rw-r--r-- | kernel/rtlil.h | 3 | ||||
| -rw-r--r-- | kernel/timinginfo.h | 48 | 
4 files changed, 64 insertions, 17 deletions
| diff --git a/kernel/constids.inc b/kernel/constids.inc index 8ccb60089..566b76217 100644 --- a/kernel/constids.inc +++ b/kernel/constids.inc @@ -179,6 +179,7 @@ X(SRC_WIDTH)  X(SRST)  X(SRST_POLARITY)  X(SRST_VALUE) +X(sta_arrival)  X(STATE_BITS)  X(STATE_NUM)  X(STATE_NUM_LOG2) diff --git a/kernel/rtlil.cc b/kernel/rtlil.cc index 88153a380..cd0f5ab12 100644 --- a/kernel/rtlil.cc +++ b/kernel/rtlil.cc @@ -480,6 +480,35 @@ vector<string> RTLIL::AttrObject::get_hdlname_attribute() const  	return split_tokens(get_string_attribute(ID::hdlname), " ");  } +void RTLIL::AttrObject::set_intvec_attribute(RTLIL::IdString id, const vector<int> &data) +{ +	std::stringstream attrval; +	for (auto &i : data) { +		if (attrval.tellp() > 0) +			attrval << " "; +		attrval << i; +	} +	attributes[id] = RTLIL::Const(attrval.str()); +} + +vector<int> RTLIL::AttrObject::get_intvec_attribute(RTLIL::IdString id) const +{ +	vector<int> data; +	auto it = attributes.find(id); +	if (it != attributes.end()) +		for (const auto &s : split_tokens(attributes.at(id).decode_string())) { +			char *end = nullptr; +			errno = 0; +			long value = strtol(s.c_str(), &end, 10); +			if (end != s.c_str() + s.size()) +				log_cmd_error("Literal for intvec attribute has invalid format"); +			if (errno == ERANGE || value < INT_MIN || value > INT_MAX) +				log_cmd_error("Literal for intvec attribute is out of range"); +			data.push_back(value); +		} +	return data; +} +  bool RTLIL::Selection::selected_module(RTLIL::IdString mod_name) const  {  	if (full_selection) diff --git a/kernel/rtlil.h b/kernel/rtlil.h index 68481b81c..073110f16 100644 --- a/kernel/rtlil.h +++ b/kernel/rtlil.h @@ -718,6 +718,9 @@ struct RTLIL::AttrObject  	void set_hdlname_attribute(const vector<string> &hierarchy);  	vector<string> get_hdlname_attribute() const; + +	void set_intvec_attribute(RTLIL::IdString id, const vector<int> &data); +	vector<int> get_intvec_attribute(RTLIL::IdString id) const;  };  struct RTLIL::SigChunk diff --git a/kernel/timinginfo.h b/kernel/timinginfo.h index 9d88ac027..e7e4eab6e 100644 --- a/kernel/timinginfo.h +++ b/kernel/timinginfo.h @@ -49,9 +49,9 @@ struct TimingInfo  	struct ModuleTiming  	{ -		RTLIL::IdString type;  		dict<BitBit, int> comb; -		dict<NameBit, int> arrival, required; +		dict<NameBit, std::pair<int,NameBit>> arrival, required; +		bool has_inputs;  	};  	dict<RTLIL::IdString, ModuleTiming> data; @@ -120,11 +120,10 @@ struct TimingInfo  				}  			}  			else if (cell->type == ID($specify3)) { -				auto src = cell->getPort(ID::SRC); +				auto src = cell->getPort(ID::SRC).as_bit();  				auto dst = cell->getPort(ID::DST); -				for (const auto &c : src.chunks()) -					if (!c.wire->port_input) -						log_error("Module '%s' contains specify cell '%s' where SRC '%s' is not a module input.\n", log_id(module), log_id(cell), log_signal(src)); +				if (!src.wire || !src.wire->port_input) +					log_error("Module '%s' contains specify cell '%s' where SRC '%s' is not a module input.\n", log_id(module), log_id(cell), log_signal(src));  				for (const auto &c : dst.chunks())  					if (!c.wire->port_output)  						log_error("Module '%s' contains specify cell '%s' where DST '%s' is not a module output.\n", log_id(module), log_id(cell), log_signal(dst)); @@ -136,34 +135,49 @@ struct TimingInfo  					max = 0;  				}  				for (const auto &d : dst) { -					auto &v = t.arrival[NameBit(d)]; -					v = std::max(v, max); +					auto r = t.arrival.insert(NameBit(d)); +					auto &v = r.first->second; +					if (r.second || v.first < max) { +						v.first = max; +						v.second = NameBit(src); +					}  				}  			}  			else if (cell->type == ID($specrule)) { -				auto type = cell->getParam(ID::TYPE).decode_string(); -				if (type != "$setup" && type != "$setuphold") +				IdString type = cell->getParam(ID::TYPE).decode_string(); +				if (type != ID($setup) && type != ID($setuphold))  					continue;  				auto src = cell->getPort(ID::SRC); -				auto dst = cell->getPort(ID::DST); +				auto dst = cell->getPort(ID::DST).as_bit();  				for (const auto &c : src.chunks()) -					if (!c.wire->port_input) +					if (!c.wire || !c.wire->port_input)  						log_error("Module '%s' contains specify cell '%s' where SRC '%s' is not a module input.\n", log_id(module), log_id(cell), log_signal(src)); -				for (const auto &c : dst.chunks()) -					if (!c.wire->port_input) -						log_error("Module '%s' contains specify cell '%s' where DST '%s' is not a module input.\n", log_id(module), log_id(cell), log_signal(dst)); +				if (!dst.wire || !dst.wire->port_input) +					log_error("Module '%s' contains specify cell '%s' where DST '%s' is not a module input.\n", log_id(module), log_id(cell), log_signal(dst));  				int max = cell->getParam(ID::T_LIMIT_MAX).as_int();  				if (max < 0) {  					log_warning("Module '%s' contains specify cell '%s' with T_LIMIT_MAX < 0 which is currently unsupported. Clamping to 0.\n", log_id(module), log_id(cell));  					max = 0;  				}  				for (const auto &s : src) { -					auto &v = t.required[NameBit(s)]; -					v = std::max(v, max); +					auto r = t.required.insert(NameBit(s)); +					auto &v = r.first->second; +					if (r.second || v.first < max) { +						v.first = max; +						v.second = NameBit(dst); +					}  				}  			}  		} +		for (auto port_name : module->ports) { +			auto wire = module->wire(port_name); +			if (wire->port_input) { +				t.has_inputs = true; +				break; +			} +		} +  		return t;  	} | 
