2020-05-05 13:31:11 -05:00
|
|
|
/********************************************************************
|
|
|
|
* This file includes functions that convert time/resistance/capacitance
|
|
|
|
* units to string or vice versa
|
|
|
|
*******************************************************************/
|
|
|
|
#include <cmath>
|
|
|
|
|
|
|
|
/* Headers from vtrutil library */
|
|
|
|
#include "vtr_assert.h"
|
|
|
|
#include "vtr_log.h"
|
|
|
|
|
|
|
|
/* Headers from openfpgautil library */
|
2022-10-06 19:08:50 -05:00
|
|
|
#include "openfpga_scale.h"
|
2020-05-05 13:31:11 -05:00
|
|
|
|
|
|
|
namespace openfpga {
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
/* A small ratio for float number comparison
|
|
|
|
* If the float number B is in the range of the referance A +/- epsilon
|
2020-05-05 13:31:11 -05:00
|
|
|
* we regard A == B
|
2022-10-06 19:08:50 -05:00
|
|
|
* A - A * EPSILON <= B <= A + A * EPSILON
|
2020-05-05 13:31:11 -05:00
|
|
|
*/
|
|
|
|
#define EPSILON_RATIO 1e-3
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
bool same_float_number(const float& a, const float& b, const float& epsilon) {
|
2020-05-05 13:31:11 -05:00
|
|
|
/* Always use a positive epsilon */
|
2022-10-06 19:08:50 -05:00
|
|
|
if ((a - a * std::abs(epsilon) <= b) && (b <= a + a * std::abs(epsilon))) {
|
2020-05-05 13:31:11 -05:00
|
|
|
return true;
|
|
|
|
}
|
2022-10-06 19:08:50 -05:00
|
|
|
|
2020-05-05 13:31:11 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Convert numeric unit to string:
|
|
|
|
* - 1e12 -> T
|
|
|
|
* - 1e9 -> B
|
|
|
|
* - 1e6 -> M
|
|
|
|
* - 1e3 -> k
|
2022-10-06 19:08:50 -05:00
|
|
|
* - 1. ->
|
2020-05-05 13:31:11 -05:00
|
|
|
* - 1e-3 -> m
|
2022-10-06 19:08:50 -05:00
|
|
|
* - 1e-6 -> u
|
2020-05-05 13:31:11 -05:00
|
|
|
* - 1e-9 -> n
|
|
|
|
* - 1e-12 -> p
|
2022-10-06 19:08:50 -05:00
|
|
|
* - 1e-15 -> f
|
|
|
|
* - 1e-18 -> a
|
2020-05-05 13:31:11 -05:00
|
|
|
*******************************************************************/
|
|
|
|
std::string unit_to_string(const float& unit) {
|
|
|
|
if (true == same_float_number(unit, 1., EPSILON_RATIO)) {
|
|
|
|
return std::string();
|
2022-10-06 19:08:50 -05:00
|
|
|
/* Larger than 1 unit */
|
2020-05-05 13:31:11 -05:00
|
|
|
} else if (true == same_float_number(unit, 1e3, EPSILON_RATIO)) {
|
|
|
|
return std::string("k");
|
|
|
|
} else if (true == same_float_number(unit, 1e6, EPSILON_RATIO)) {
|
|
|
|
return std::string("M");
|
|
|
|
} else if (true == same_float_number(unit, 1e9, EPSILON_RATIO)) {
|
|
|
|
return std::string("B");
|
|
|
|
} else if (true == same_float_number(unit, 1e12, EPSILON_RATIO)) {
|
|
|
|
return std::string("T");
|
2022-10-06 19:08:50 -05:00
|
|
|
/* Less than 1 unit */
|
2020-05-05 13:31:11 -05:00
|
|
|
} else if (true == same_float_number(unit, 1e-3, EPSILON_RATIO)) {
|
|
|
|
return std::string("m");
|
|
|
|
} else if (true == same_float_number(unit, 1e-6, EPSILON_RATIO)) {
|
|
|
|
return std::string("u");
|
|
|
|
} else if (true == same_float_number(unit, 1e-9, EPSILON_RATIO)) {
|
|
|
|
return std::string("n");
|
|
|
|
} else if (true == same_float_number(unit, 1e-12, EPSILON_RATIO)) {
|
|
|
|
return std::string("p");
|
|
|
|
} else if (true == same_float_number(unit, 1e-15, EPSILON_RATIO)) {
|
|
|
|
return std::string("f");
|
|
|
|
} else if (true == same_float_number(unit, 1e-18, EPSILON_RATIO)) {
|
|
|
|
return std::string("a");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invalid unit report error */
|
|
|
|
VTR_LOGF_ERROR(__FILE__, __LINE__,
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid unit %g!\nAcceptable units are "
|
|
|
|
"[1e12|1e9|1e6|1e3|1|1e-3|1e-6|1e-9|1e-12|1e-15|1e-18]\n",
|
|
|
|
unit);
|
2020-05-05 13:31:11 -05:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Convert numeric time unit to string
|
|
|
|
* e.g. 1e-12 -> ps
|
|
|
|
*******************************************************************/
|
2021-10-06 13:55:57 -05:00
|
|
|
std::string time_unit_to_string(const float& unit, const std::string& postfix) {
|
2020-05-05 13:31:11 -05:00
|
|
|
/* For larger than 1 unit, we do not accept */
|
|
|
|
if (1e6 < unit) {
|
|
|
|
VTR_LOGF_ERROR(__FILE__, __LINE__,
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid time unit %g!\nAcceptable units are "
|
|
|
|
"[1e6|1e3|1|1e-3|1e-6|1e-9|1e-12|1e-15|1e-18]\n",
|
|
|
|
unit);
|
2020-05-05 13:31:11 -05:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2021-10-06 13:55:57 -05:00
|
|
|
return unit_to_string(unit) + postfix;
|
2020-05-05 13:31:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Convert string unit to numeric:
|
|
|
|
* - T -> 1e12
|
2022-10-06 19:08:50 -05:00
|
|
|
* - B -> 1e9
|
2020-05-05 13:31:11 -05:00
|
|
|
* - M -> 1e6
|
|
|
|
* - k -> 1e3
|
2022-10-06 19:08:50 -05:00
|
|
|
* - "" -> 1.
|
2020-05-05 13:31:11 -05:00
|
|
|
* - m -> 1e-3
|
2022-10-06 19:08:50 -05:00
|
|
|
* - u -> 1e-6
|
2020-05-05 13:31:11 -05:00
|
|
|
* - n -> 1e-9
|
|
|
|
* - p -> 1e-12
|
|
|
|
* - f -> 1e-15
|
|
|
|
* - a -> 1e-18
|
|
|
|
*******************************************************************/
|
|
|
|
float string_to_unit(const std::string& scale) {
|
|
|
|
if (true == scale.empty()) {
|
|
|
|
return 1.;
|
2022-10-06 19:08:50 -05:00
|
|
|
/* Larger than 1 unit */
|
2020-05-05 13:31:11 -05:00
|
|
|
} else if (std::string("T") == scale) {
|
|
|
|
return 1e12;
|
|
|
|
} else if (std::string("B") == scale) {
|
|
|
|
return 1e9;
|
|
|
|
} else if (std::string("M") == scale) {
|
|
|
|
return 1e6;
|
|
|
|
} else if (std::string("k") == scale) {
|
|
|
|
return 1e3;
|
2022-10-06 19:08:50 -05:00
|
|
|
/* Less than 1 unit */
|
2020-05-05 13:31:11 -05:00
|
|
|
} else if (std::string("m") == scale) {
|
|
|
|
return 1e-3;
|
|
|
|
} else if (std::string("u") == scale) {
|
|
|
|
return 1e-6;
|
|
|
|
} else if (std::string("n") == scale) {
|
|
|
|
return 1e-9;
|
|
|
|
} else if (std::string("p") == scale) {
|
|
|
|
return 1e-12;
|
|
|
|
} else if (std::string("f") == scale) {
|
|
|
|
return 1e-15;
|
|
|
|
} else if (std::string("a") == scale) {
|
|
|
|
return 1e-18;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invalid unit report error */
|
2022-10-06 19:08:50 -05:00
|
|
|
VTR_LOGF_ERROR(
|
|
|
|
__FILE__, __LINE__,
|
|
|
|
"Invalid unit %s!\nAcceptable units are [a|f|p|n|u|k|M|B|T] or empty\n",
|
|
|
|
scale.c_str());
|
2020-05-05 13:31:11 -05:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************
|
|
|
|
* Convert string time unit to numeric
|
|
|
|
* e.g. ps -> 1e-12
|
|
|
|
*******************************************************************/
|
|
|
|
float string_to_time_unit(const std::string& scale) {
|
2022-10-06 19:08:50 -05:00
|
|
|
if ((1 != scale.length()) && (2 != scale.length())) {
|
|
|
|
VTR_LOGF_ERROR(
|
|
|
|
__FILE__, __LINE__,
|
|
|
|
"Time unit (='%s') must contain only one or two characters!\n",
|
|
|
|
scale.c_str());
|
|
|
|
}
|
2020-05-05 13:31:11 -05:00
|
|
|
/* The last character must be 's' */
|
|
|
|
if ('s' != scale.back()) {
|
2022-10-06 19:08:50 -05:00
|
|
|
VTR_LOGF_ERROR(__FILE__, __LINE__, "Time unit (='%s') must end with 's'!\n",
|
2020-09-14 19:55:21 -05:00
|
|
|
scale.c_str());
|
2020-05-05 13:31:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
float unit = 1.;
|
2022-10-06 19:08:50 -05:00
|
|
|
VTR_ASSERT((1 == scale.length()) || (2 == scale.length()));
|
2020-05-05 13:31:11 -05:00
|
|
|
if (2 == scale.length()) {
|
|
|
|
unit = string_to_unit(scale.substr(0, 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For larger than 1 unit, we do not accept */
|
|
|
|
if (1e6 < unit) {
|
|
|
|
VTR_LOGF_ERROR(__FILE__, __LINE__,
|
2022-10-06 19:08:50 -05:00
|
|
|
"Invalid time unit %g!\nAcceptable units are "
|
|
|
|
"[1e6|1e3|1|1e-3|1e-6|1e-9|1e-12|1e-15|1e-18]\n",
|
|
|
|
unit);
|
2020-05-05 13:31:11 -05:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return unit;
|
|
|
|
}
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
} // namespace openfpga
|