blob: a1c4a7a78bff5fbbcf641efeeca80dac967608d7 [file] [log] [blame]
/*
* Copyright 2020-2022 F4PGA Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* SPDX-License-Identifier: Apache-2.0
*/
#include "clocks.h"
#include "kernel/register.h"
#include "propagation.h"
#include <cassert>
#include <cmath>
#include <regex>
void Clock::Add(const std::string &name, RTLIL::Wire *wire, float period, float rising_edge, float falling_edge, ClockType type)
{
wire->set_string_attribute(RTLIL::escape_id("CLOCK_SIGNAL"), "yes");
wire->set_bool_attribute(RTLIL::escape_id("IS_GENERATED"), type == GENERATED);
wire->set_bool_attribute(RTLIL::escape_id("IS_EXPLICIT"), type == EXPLICIT);
wire->set_bool_attribute(RTLIL::escape_id("IS_PROPAGATED"), type == PROPAGATED);
wire->set_string_attribute(RTLIL::escape_id("CLASS"), "clock");
wire->set_string_attribute(RTLIL::escape_id("NAME"), name);
wire->set_string_attribute(RTLIL::escape_id("SOURCE_WIRES"), Clock::WireName(wire));
wire->set_string_attribute(RTLIL::escape_id("PERIOD"), std::to_string(period));
std::string waveform(std::to_string(rising_edge) + " " + std::to_string(falling_edge));
wire->set_string_attribute(RTLIL::escape_id("WAVEFORM"), waveform);
}
void Clock::Add(const std::string &name, std::vector<RTLIL::Wire *> wires, float period, float rising_edge, float falling_edge, ClockType type)
{
std::for_each(wires.begin(), wires.end(), [&](RTLIL::Wire *wire) { Add(name, wire, period, rising_edge, falling_edge, type); });
}
void Clock::Add(RTLIL::Wire *wire, float period, float rising_edge, float falling_edge, ClockType type)
{
Add(Clock::WireName(wire), wire, period, rising_edge, falling_edge, type);
}
float Clock::Period(RTLIL::Wire *clock_wire)
{
if (!clock_wire->has_attribute(RTLIL::escape_id("PERIOD"))) {
log_cmd_error("PERIOD has not been specified on wire '%s'.\n", WireName(clock_wire).c_str());
}
float period(0);
std::string period_str;
try {
period_str = clock_wire->get_string_attribute(RTLIL::escape_id("PERIOD"));
period = std::stof(period_str);
} catch (const std::invalid_argument &e) {
log_cmd_error("Incorrect value '%s' specifed on PERIOD attribute for wire "
"'%s'.\nPERIOD needs to be a float value.\n",
period_str.c_str(), WireName(clock_wire).c_str());
}
return period;
}
std::pair<float, float> Clock::Waveform(RTLIL::Wire *clock_wire)
{
if (!clock_wire->has_attribute(RTLIL::escape_id("WAVEFORM"))) {
float period(Period(clock_wire));
if (!period) {
log_cmd_error("Neither PERIOD nor WAVEFORM has been specified for wire %s\n", WireName(clock_wire).c_str());
return std::make_pair(0, 0);
}
float falling_edge = period / 2;
log_warning("Waveform has not been specified on wire '%s'.\nDefault value {0 %f} "
"will be used\n",
WireName(clock_wire).c_str(), falling_edge);
return std::make_pair(0, falling_edge);
}
float rising_edge(0);
float falling_edge(0);
std::string waveform(clock_wire->get_string_attribute(RTLIL::escape_id("WAVEFORM")));
if (std::sscanf(waveform.c_str(), "%f %f", &rising_edge, &falling_edge) != 2) {
log_cmd_error("Incorrect value '%s' specifed on WAVEFORM attribute for wire "
"'%s'.\nWAVEFORM needs to be specified in form of '<rising_edge> "
"<falling_edge>' where the edge values are floats.\n",
waveform.c_str(), WireName(clock_wire).c_str());
}
return std::make_pair(rising_edge, falling_edge);
}
float Clock::RisingEdge(RTLIL::Wire *clock_wire) { return Waveform(clock_wire).first; }
float Clock::FallingEdge(RTLIL::Wire *clock_wire) { return Waveform(clock_wire).second; }
std::string Clock::Name(RTLIL::Wire *clock_wire)
{
if (clock_wire->has_attribute(RTLIL::escape_id("NAME"))) {
return clock_wire->get_string_attribute(RTLIL::escape_id("NAME"));
}
return WireName(clock_wire);
}
std::string Clock::WireName(RTLIL::Wire *clock_wire)
{
if (!clock_wire) {
return std::string();
}
return AddEscaping(RTLIL::unescape_id(clock_wire->name));
}
std::string Clock::SourceWireName(RTLIL::Wire *clock_wire)
{
if (clock_wire->has_attribute(RTLIL::escape_id("SOURCE_WIRES"))) {
return clock_wire->get_string_attribute(RTLIL::escape_id("SOURCE_WIRES"));
}
return Name(clock_wire);
}
bool Clock::GetClockWireBoolAttribute(RTLIL::Wire *wire, const std::string &attribute_name)
{
if (wire->has_attribute(RTLIL::escape_id(attribute_name))) {
return wire->get_bool_attribute(RTLIL::escape_id(attribute_name));
}
return false;
}
const std::map<std::string, RTLIL::Wire *> Clocks::GetClocks(RTLIL::Design *design)
{
std::map<std::string, RTLIL::Wire *> clock_wires;
RTLIL::Module *top_module = design->top_module();
for (auto &wire_obj : top_module->wires_) {
auto &wire = wire_obj.second;
if (wire->has_attribute(RTLIL::escape_id("CLOCK_SIGNAL"))) {
if (wire->get_string_attribute(RTLIL::escape_id("CLOCK_SIGNAL")) == "yes") {
clock_wires.insert(std::make_pair(Clock::WireName(wire), wire));
}
}
}
return clock_wires;
}
void Clocks::UpdateAbc9DelayTarget(RTLIL::Design *design)
{
std::map<std::string, RTLIL::Wire *> clock_wires = Clocks::GetClocks(design);
for (auto &clock_wire : clock_wires) {
auto &wire = clock_wire.second;
float period = Clock::Period(wire);
// Set the ABC9 delay to the shortest clock period in the design.
//
// By convention, delays in Yosys are in picoseconds, but ABC9 has
// no information on interconnect delay, so target half the specified
// clock period to give timing slack; otherwise ABC9 may produce a
// mapping that cannot meet the specified clock.
int abc9_delay = design->scratchpad_get_int("abc9.D", INT32_MAX);
int period_ps = period * 1000.0 / 2.0;
design->scratchpad_set_int("abc9.D", std::min(abc9_delay, period_ps));
}
}