OpenSTA/tcl/Util.tcl

388 lines
11 KiB
Tcl

# 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/>.
# The sta namespace is used for all commands defined by the sta.
# Use define_cmd_args to define command arguments for the help
# command, and export the command name to the global namespace.
# Global variables must be defined as
# set global_var init_value
# File local variables must be defined as
# variable sta_var init_value
namespace eval sta {
# Parse arg_var for keyword/values pairs and flags.
# $key_var(key) -> key_value
# $flag_var(flag) -> 1 if the flag is present
# Keys and flags are removed from arg_var in the caller.
proc parse_key_args { cmd arg_var key_var keys {flag_var ""} {flags {}} \
{unknown_key_is_error 1} } {
upvar 1 $arg_var args
upvar 1 $key_var key_value
upvar 1 $flag_var flag_present
set args_rtn {}
while { $args != "" } {
set arg [lindex $args 0]
if { [is_keyword_arg $arg] } {
set key_index [lsearch -exact $keys $arg]
if { $key_index >= 0 } {
set key $arg
if { [llength $args] == 1 } {
sta_error 560 "$cmd $key missing value."
}
set key_value($key) [lindex $args 1]
set args [lrange $args 1 end]
} else {
set flag_index [lsearch -exact $flags $arg]
if { $flag_index >= 0 } {
set flag_present($arg) 1
} else {
# No exact keyword/flag match found.
# Try finding a keyword/flag that begins with
# the same substring.
set wild_arg "${arg}*"
set key_index [lsearch -glob $keys $wild_arg]
if { $key_index >= 0 } {
set key [lindex $keys $key_index]
if { [llength $args] == 1 } {
sta_error 561 "$cmd $key missing value."
}
set key_value($key) [lindex $args 1]
set args [lrange $args 1 end]
} else {
set flag_index [lsearch -glob $flags $wild_arg]
if { $flag_index >= 0 } {
set flag [lindex $flags $flag_index]
set flag_present($flag) 1
} elseif { $unknown_key_is_error } {
sta_error 562 "$cmd $arg is not a known keyword or flag."
} else {
lappend args_rtn $arg
}
}
}
}
} else {
lappend args_rtn $arg
}
set args [lrange $args 1 end]
}
set args $args_rtn
}
# Check for keyword args in arg_var.
proc check_for_key_args { cmd arg_var } {
upvar 1 $arg_var args
set args_rtn {}
while { $args != "" } {
set arg [lindex $args 0]
if { [is_keyword_arg $arg] } {
sta_error 563 "$cmd $arg is not a known keyword or flag."
} else {
lappend args_rtn $arg
}
set args [lrange $args 1 end]
}
set args $args_rtn
}
proc is_keyword_arg { arg } {
if { [string length $arg] >= 2 \
&& [string index $arg 0] == "-" \
&& [string is alpha [string index $arg 1]] } {
return 1
} else {
return 0
}
}
################################################################
# Define a procedure that checks the args for redirection using unix
# shell redirection syntax.
# The value of the last expression in the body is returned.
proc proc_redirect { proc_name body } {
set proc_body [concat "proc $proc_name { args } {" \
"global errorCode errorInfo;" \
"set redirect \[parse_redirect_args args\];" \
"set code \[catch {" $body "} ret \];" \
"if {\$redirect} { redirect_file_end };" \
"if {\$code == 1} {return -code \$code -errorcode \$errorCode -errorinfo \$errorInfo \$ret} else {return \$ret} }" ]
eval $proc_body
}
proc parse_redirect_args { arg_var } {
upvar 1 $arg_var args
set argc [llength $args]
if { $argc >= 1 } {
set last_arg [lindex $args [expr $argc - 1]]
# arg >>filename
if { [string range $last_arg 0 1] == ">>" } {
set redirect_file [file nativename [string range $last_arg 2 end]]
set args [lrange $args 0 [expr $argc - 2]]
redirect_file_append_begin $redirect_file
return 1
# arg >filename
} elseif { [string range $last_arg 0 0] == ">" } {
set redirect_file [file nativename [string range $last_arg 1 end]]
set args [lrange $args 0 [expr $argc - 2]]
redirect_file_begin $redirect_file
return 1
}
}
if { $argc >= 2 } {
set next_last_arg [lindex $args [expr $argc - 2]]
# arg > filename
if { $next_last_arg == ">" } {
set redirect_file [file nativename [lindex $args end]]
set args [lrange $args 0 [expr $argc - 3]]
redirect_file_begin $redirect_file
return 1
# arg >> filename
} elseif { $next_last_arg == ">>" } {
set redirect_file [file nativename [lindex $args end]]
set args [lrange $args 0 [expr $argc - 3]]
redirect_file_append_begin $redirect_file
return 1
}
}
return 0
}
################################################################
proc define_cmd_args { cmd arglist } {
variable cmd_args
set cmd_args($cmd) $arglist
namespace export $cmd
}
# Hidden commands are exported to the global namespace but are not
# shown by the "help" command.
proc define_hidden_cmd_args { cmd arglist } {
namespace export $cmd
}
################################################################
proc sta_warn { msg_id msg } {
variable sdc_file
variable sdc_line
if { [info exists sdc_file] } {
report_file_warn $msg_id [file tail $sdc_file] $sdc_line $msg
} else {
report_warn $msg_id $msg
}
}
proc sta_error { msg_id msg } {
variable sdc_file
variable sdc_line
if { ! [is_suppressed $msg_id] } {
if { [info exists sdc_file] } {
error "Error: [file tail $sdc_file] line $sdc_line, $msg"
} else {
error "Error: $msg"
}
}
}
proc sta_warn_error { msg_id warn_error msg } {
if { $warn_error == "warn" } {
sta_warn $msg_id $msg
} else {
sta_error $msg_id $msg
}
}
define_cmd_args "suppress_msg" msg_ids
proc suppress_msg { args } {
foreach msg_id $args {
check_integer "msg_id" $msg_id
suppress_msg_id $msg_id
}
}
define_cmd_args "unsuppress_msg" msg_ids
proc unsuppress_msg { args } {
foreach msg_id $args {
check_integer "msg_id" $msg_id
unsuppress_msg_id $msg_id
}
}
# Defined by StaTcl.i
define_cmd_args "elapsed_run_time" {}
define_cmd_args "user_run_time" {}
# Write run time statistics to filename.
proc write_stats { filename } {
if { ![catch {open $filename w} stream] } {
puts $stream "[elapsed_run_time] [user_run_time] [memory_usage]"
close $stream
}
}
################################################################
# Begin/end logging all output to a file.
define_cmd_args "log_begin" { filename }
proc log_begin { filename } {
log_begin_cmd [file nativename $filename]
}
# Defined by StaTcl.i
define_cmd_args "log_end" {}
# set_debug is NOT in the global namespace
# because it isn't intended for nosy users.
################################################################
proc check_argc_eq0 { cmd arglist } {
if { $arglist != {} } {
sta_error 564 "$cmd positional arguments not supported."
}
}
proc check_argc_eq1 { cmd arglist } {
if { [llength $arglist] != 1 } {
sta_error 565 "$cmd requires one positional argument."
}
}
proc check_argc_eq0or1 { cmd arglist } {
set argc [llength $arglist]
if { $argc != 0 && $argc != 1 } {
sta_error 566 "$cmd requires zero or one positional arguments."
}
}
proc check_argc_eq2 { cmd arglist } {
if { [llength $arglist] != 2 } {
sta_error 567 "$cmd requires two positional arguments."
}
}
proc check_argc_eq1or2 { cmd arglist } {
set argc [llength $arglist]
if { $argc != 1 && $argc != 2 } {
sta_error 568 "$cmd requires one or two positional arguments."
}
}
proc check_argc_eq3 { cmd arglist } {
if { [llength $arglist] != 3 } {
sta_error 569 "$cmd requires three positional arguments."
}
}
proc check_argc_eq4 { cmd arglist } {
if { [llength $arglist] != 4 } {
sta_error 570 "$cmd requires four positional arguments."
}
}
################################################################
proc check_float { cmd_arg arg } {
if {![string is double $arg]} {
sta_error 571 "$cmd_arg '$arg' is not a float."
}
}
proc check_positive_float { cmd_arg arg } {
if {!([string is double $arg] && $arg >= 0.0)} {
sta_error 572 "$cmd_arg '$arg' is not a positive float."
}
}
proc check_integer { cmd_arg arg } {
if {!([string is integer $arg])} {
sta_error 573 "$cmd_arg '$arg' is not an integer."
}
}
proc check_positive_integer { cmd_arg arg } {
if {!([string is integer $arg] && $arg >= 0)} {
sta_error 574 "$cmd_arg '$arg' is not a positive integer."
}
}
proc check_cardinal { cmd_arg arg } {
if {!([string is integer $arg] && $arg >= 1)} {
sta_error 575 "$cmd_arg '$arg' is not an integer greater than or equal to one."
}
}
proc check_percent { cmd_arg arg } {
if {!([string is double $arg] && $arg >= 0.0 && $arg <= 100.0)} {
sta_error 576 "$cmd_arg '$arg' is not between 0 and 100."
}
}
# sta namespace end
}
################################################################
# Bus signal names like foo[2] or bar[31:0] use brackets that
# look like "eval" to TCL. Catch the numeric "function" with the
# namespace's unknown handler and return the value instead of an error.
proc sta_unknown { args } {
global errorCode errorInfo
set name [lindex $args 0]
if { [llength $args] == 1 && [is_bus_subscript $args] } {
return "\[$args\]"
}
# Command name abbreviation support.
set ret [catch {set cmds [info commands $name*]} msg]
if {[string equal $name "::"]} {
set name ""
}
if { $ret != 0 } {
return -code $ret -errorcode $errorCode \
"Error in unknown while checking if \"$name\" is a unique command abbreviation: $msg."
}
if { [llength $cmds] == 1 } {
return [uplevel 1 [lreplace $args 0 0 $cmds]]
}
if { [llength $cmds] > 1 } {
if {[string equal $name ""]} {
return -code error "Empty command name \"\""
} else {
return -code error \
"Ambiguous command name \"$name\": [lsort $cmds]."
}
}
return [uplevel 1 [::unknown {*}$args]]
}
proc is_bus_subscript { subscript } {
return [expr [string is integer $subscript] \
|| [string match $subscript "*"] \
|| [regexp {[0-9]+:[0-9]} $subscript]]
}
namespace unknown sta_unknown