blob: fe3158c3fe967d9f0c652ebd1d2e6e1a288b4edf [file] [log] [blame] [edit]
#!/usr/bin/wish
# This file provides many valuable stack inquiry procedures like
# get_file_proc_names, get_stack_var, etc..
my_source [list print.tcl]
proc get_file_proc_names { file_path { name_regex "" } } {
# Get all proc names from the file indicated by file_path and return them
# as a list.
# Description of argument(s):
# file_path The path to the file whose proc names are
# to be retrieved.
# name_regex A regular expression to be used to narrow
# the result to just the desired procs.
# The first sed command serves to eliminate curly braces from the target
# file. They are a distraction to what we are trying to do.
# TCL proc lines begin with...
# - Zero or more spaces...
# - The "proc" keyword...
# - One or more spaces...
set proc_regex "^\[ \]*proc\[ \]+"
set cmd_buf "sed -re 's/\[\\\{\\\}]//g' $file_path | egrep"
append cmd_buf " '${proc_regex}${name_regex}[ ]' | sed -re"
append cmd_buf " 's/${proc_regex}(\[^ \]+).*/\\1/g'"
return [split [eval exec bash -c {$cmd_buf}] "\n"]
}
proc get_stack_var { var_name { default {} } { init_stack_ix 1 } } {
# Starting with the caller's stack level, search upward in the call stack,
# for a variable named "${var_name}" and return its value. If the variable
# cannot be found, return ${default}.
# Description of argument(s):
# var_name The name of the variable be searched for.
# default The value to return if the the variable
# cannot be found.
for {set stack_ix $init_stack_ix} {$stack_ix <= [info level]} \
{incr stack_ix} {
upvar $stack_ix $var_name var_ref
if { [info exists var_ref] } { return $var_ref }
}
return $default
}
proc get_stack_var_level { var_name { init_stack_ix 1 } { fail_on_err 1 } } {
# Starting with the caller's stack level, search upward in the call stack,
# for a variable named "${var_name}" and return its associated stack level.
# If the variable cannot be found, return -1.
# Description of argument(s):
# var_name The name of the variable be searched for.
# init_stack_ix The level of the stack where the search
# should start. The default is 1 which is
# the caller's stack level.
# fail_on_err Indicates that if the variable cannot be
# found on the stack, this proc should write
# to stderr and exit with a non-zero return
# code.
for {set stack_ix $init_stack_ix} {$stack_ix <= [info level]} \
{incr stack_ix} {
upvar $stack_ix $var_name var_ref
set stack_level [expr $stack_ix - $init_stack_ix]
if { [info exists var_ref] } { return $stack_level }
}
if { $fail_on_err } {
append message "Programmer error - Couldn't find variable \"${var_name}\""
append message " on the stack."
print_error_report $message
exit 1
}
return -1
}
proc get_stack_proc_name { { level -1 } { include_args 0 } } {
# Get the name of the procedure at the indicated call stack level and
# return it.
# Description of argument(s):
# level The call stack level: 0 would mean this
# procedure's level (i.e.
# get_stack_proc_name's level), -1 would
# indicate the caller's level, etc.
# include_args Indicates whether proc arg values should
# be included in the result.
# Set default.
set_var_default level -1
if { $include_args } {
set cmd_buf "set proc_name \[info level $level\]"
} else {
set cmd_buf "set proc_name \[lindex \[info level $level\] 0\]"
}
if { [ catch $cmd_buf result ] } {
# The command failed most likely due to being called from "main".
set proc_name "main"
}
return $proc_name
}
proc get_call_stack { { stack_top_ix -1 } { include_args 0 } } {
# Return the call stack as a list of procedure names.
# Example:
# set call_stack [get_call_stack 0]
# call_stack: get_call_stack calc_wrap_stack_ix_adjust sprint_var
# sprint_vars print_vars
# Description of argument(s):
# stack_top_ix The index to the bottom of the stack to be
# returned. 0 means include the entire
# stack. 1 means include the entire stack
# with the exception of this procedure
# itself, etc.
# include_args Indicates whether proc args should be
# included in the result.
set_var_default stack_top_ix -1
# Get the current stack size.
set stack_size [info level]
# Calculate stack_bottom_ix. Example: if stack_size is 5, stack_bottom_ix
# is -4.
set stack_bottom_ix [expr 1 - $stack_size]
for {set stack_ix $stack_top_ix} {$stack_ix >= $stack_bottom_ix} \
{incr stack_ix -1} {
if { $include_args } {
set proc_name [info level $stack_ix]
} else {
set proc_name [lindex [info level $stack_ix] 0]
}
lappend call_stack $proc_name
}
return $call_stack
}