552 lines
10 KiB
OpenEdge ABL
552 lines
10 KiB
OpenEdge ABL
// OpenSTA, Static Timing Analyzer
|
|
// Copyright (c) 2024, Parallax Software, Inc.
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
%module util
|
|
|
|
%{
|
|
|
|
#include "Sta.hh"
|
|
#include "StaConfig.hh" // STA_VERSION
|
|
#include "Stats.hh"
|
|
#include "Report.hh"
|
|
#include "Error.hh"
|
|
#include "Fuzzy.hh"
|
|
#include "Units.hh"
|
|
|
|
using namespace sta;
|
|
|
|
%}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
//
|
|
// Empty class definitions to make swig happy.
|
|
// Private constructor/destructor so swig doesn't emit them.
|
|
//
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
%inline %{
|
|
|
|
float float_inf = INF;
|
|
|
|
const char *
|
|
version()
|
|
{
|
|
return STA_VERSION;
|
|
}
|
|
|
|
const char *
|
|
git_sha1()
|
|
{
|
|
return STA_GIT_SHA1;
|
|
}
|
|
|
|
// Elapsed run time (in seconds).
|
|
double
|
|
elapsed_run_time()
|
|
{
|
|
return elapsedRunTime();
|
|
}
|
|
|
|
// User run time (in seconds).
|
|
double
|
|
user_run_time()
|
|
{
|
|
return userRunTime();
|
|
}
|
|
|
|
// User run time (in seconds).
|
|
unsigned long
|
|
cputime()
|
|
{
|
|
return static_cast<unsigned long>(userRunTime() + .5);
|
|
}
|
|
|
|
// Peak memory usage in bytes.
|
|
unsigned long
|
|
memory_usage()
|
|
{
|
|
return memoryUsage();
|
|
}
|
|
|
|
int
|
|
processor_count()
|
|
{
|
|
return processorCount();
|
|
}
|
|
|
|
int
|
|
thread_count()
|
|
{
|
|
return Sta::sta()->threadCount();
|
|
}
|
|
|
|
void
|
|
set_thread_count(int count)
|
|
{
|
|
Sta::sta()->setThreadCount(count);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
void
|
|
report_error(int id,
|
|
const char *msg)
|
|
{
|
|
Report *report = Sta::sta()->report();
|
|
report->error(id, "%s", msg);
|
|
}
|
|
|
|
void
|
|
report_file_error(int id,
|
|
const char *filename,
|
|
int line,
|
|
const char *msg)
|
|
{
|
|
Report *report = Sta::sta()->report();
|
|
report->error(id, filename, line, "%s", msg);
|
|
}
|
|
|
|
void
|
|
report_warn(int id,
|
|
const char *msg)
|
|
{
|
|
Report *report = Sta::sta()->report();
|
|
report->warn(id, "%s", msg);
|
|
}
|
|
|
|
void
|
|
report_file_warn(int id,
|
|
const char *filename,
|
|
int line,
|
|
const char *msg)
|
|
{
|
|
Report *report = Sta::sta()->report();
|
|
report->fileWarn(id, filename, line, "%s", msg);
|
|
}
|
|
|
|
void
|
|
report_line(const char *msg)
|
|
{
|
|
Sta *sta = Sta::sta();
|
|
if (sta)
|
|
sta->report()->reportLineString(msg);
|
|
else
|
|
// After sta::delete_all_memory souce -echo prints the cmd file line
|
|
printf("%s\n", msg);
|
|
}
|
|
|
|
void
|
|
suppress_msg_id(int id)
|
|
{
|
|
Sta::sta()->report()->suppressMsgId(id);
|
|
}
|
|
|
|
void
|
|
unsuppress_msg_id(int id)
|
|
{
|
|
Sta::sta()->report()->unsuppressMsgId(id);
|
|
}
|
|
|
|
int
|
|
is_suppressed(int id)
|
|
{
|
|
return Sta::sta()->report()->isSuppressed(id);
|
|
}
|
|
|
|
void
|
|
fflush()
|
|
{
|
|
fflush(stdout);
|
|
fflush(stderr);
|
|
}
|
|
|
|
void
|
|
redirect_file_begin(const char *filename)
|
|
{
|
|
Sta::sta()->report()->redirectFileBegin(filename);
|
|
}
|
|
|
|
void
|
|
redirect_file_append_begin(const char *filename)
|
|
{
|
|
Sta::sta()->report()->redirectFileAppendBegin(filename);
|
|
}
|
|
|
|
void
|
|
redirect_file_end()
|
|
{
|
|
Sta::sta()->report()->redirectFileEnd();
|
|
}
|
|
|
|
void
|
|
redirect_string_begin()
|
|
{
|
|
Sta::sta()->report()->redirectStringBegin();
|
|
}
|
|
|
|
const char *
|
|
redirect_string_end()
|
|
{
|
|
return Sta::sta()->report()->redirectStringEnd();
|
|
}
|
|
|
|
void
|
|
log_begin_cmd(const char *filename)
|
|
{
|
|
Sta::sta()->report()->logBegin(filename);
|
|
}
|
|
|
|
void
|
|
log_end()
|
|
{
|
|
Sta::sta()->report()->logEnd();
|
|
}
|
|
|
|
void
|
|
set_debug(const char *what,
|
|
int level)
|
|
{
|
|
Sta::sta()->setDebugLevel(what, level);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
bool
|
|
is_object(const char *obj)
|
|
{
|
|
// _hexaddress_p_type
|
|
const char *s = obj;
|
|
char ch = *s++;
|
|
if (ch != '_')
|
|
return false;
|
|
while (*s && isxdigit(*s))
|
|
s++;
|
|
if ((s - obj - 1) == sizeof(void*) * 2
|
|
&& *s && *s++ == '_'
|
|
&& *s && *s++ == 'p'
|
|
&& *s && *s++ == '_') {
|
|
while (*s && *s != ' ')
|
|
s++;
|
|
return *s == '\0';
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// Assumes is_object is true.
|
|
const char *
|
|
object_type(const char *obj)
|
|
{
|
|
return &obj[1 + sizeof(void*) * 2 + 3];
|
|
}
|
|
|
|
bool
|
|
is_object_list(const char *list,
|
|
const char *type)
|
|
{
|
|
const char *s = list;
|
|
while (s) {
|
|
bool type_match;
|
|
const char *next;
|
|
objectListNext(s, type, type_match, next);
|
|
if (type_match)
|
|
s = next;
|
|
else
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
//
|
|
// Units
|
|
//
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// <unit>_sta_ui conversion from sta units to user interface units.
|
|
// <unit>_ui_sta conversion from user interface units to sta units.
|
|
|
|
double
|
|
time_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->timeUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
time_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->timeUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
capacitance_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->capacitanceUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
capacitance_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->capacitanceUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
resistance_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->resistanceUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
resistance_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->resistanceUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
voltage_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->voltageUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
voltage_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->voltageUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
current_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->currentUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
current_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->currentUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
power_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->powerUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
power_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->powerUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
distance_ui_sta(double value)
|
|
{
|
|
return Sta::sta()->units()->distanceUnit()->userToSta(value);
|
|
}
|
|
|
|
double
|
|
distance_sta_ui(double value)
|
|
{
|
|
return Sta::sta()->units()->distanceUnit()->staToUser(value);
|
|
}
|
|
|
|
double
|
|
area_ui_sta(double value)
|
|
{
|
|
double scale = Sta::sta()->units()->distanceUnit()->scale();
|
|
return value * scale * scale;
|
|
}
|
|
|
|
double
|
|
area_sta_ui(double value)
|
|
{
|
|
double scale = Sta::sta()->units()->distanceUnit()->scale();
|
|
return value / (scale * scale);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
void
|
|
set_cmd_unit_scale(const char *unit_name,
|
|
float scale)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
unit->setScale(scale);
|
|
}
|
|
|
|
void
|
|
set_cmd_unit_digits(const char *unit_name,
|
|
int digits)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
unit->setDigits(digits);
|
|
}
|
|
|
|
void
|
|
set_cmd_unit_suffix(const char *unit_name,
|
|
const char *suffix)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit) {
|
|
unit->setSuffix(suffix);
|
|
}
|
|
}
|
|
|
|
const char *
|
|
unit_scale_abbreviation (const char *unit_name)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
return unit->scaleAbbreviation();
|
|
else
|
|
return "";
|
|
}
|
|
|
|
const char *
|
|
unit_suffix(const char *unit_name)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
return unit->suffix();
|
|
else
|
|
return "";
|
|
}
|
|
|
|
const char *
|
|
unit_scaled_suffix(const char *unit_name)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
return unit->scaledSuffix();
|
|
else
|
|
return "";
|
|
}
|
|
|
|
float
|
|
unit_scale(const char *unit_name)
|
|
{
|
|
Unit *unit = Sta::sta()->units()->find(unit_name);
|
|
if (unit)
|
|
return unit->scale();
|
|
else
|
|
return 1.0F;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
// format_unit functions print with fixed digits and suffix.
|
|
// Pass value arg as string to support NaNs.
|
|
const char *
|
|
format_time(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->timeUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_capacitance(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->capacitanceUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_resistance(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->resistanceUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_voltage(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->voltageUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_current(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->currentUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_power(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
return Sta::sta()->units()->powerUnit()->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_distance(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
Unit *dist_unit = Sta::sta()->units()->distanceUnit();
|
|
return dist_unit->asString(value1, digits);
|
|
}
|
|
|
|
const char *
|
|
format_area(const char *value,
|
|
int digits)
|
|
{
|
|
float value1 = strtof(value, nullptr);
|
|
Unit *dist_unit = Sta::sta()->units()->distanceUnit();
|
|
return dist_unit->asString(value1 / dist_unit->scale(), digits);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
const char *
|
|
rise_short_name()
|
|
{
|
|
return RiseFall::rise()->shortName();
|
|
}
|
|
|
|
const char *
|
|
fall_short_name()
|
|
{
|
|
return RiseFall::fall()->shortName();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
bool
|
|
fuzzy_equal(float value1,
|
|
float value2)
|
|
{
|
|
return fuzzyEqual(value1, value2);
|
|
}
|
|
|
|
%} // inline
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
//
|
|
// Object Methods
|
|
//
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|