xref: /openbmc/openbmc-test-automation/lib/call_stack.tcl (revision ff3879e015e38f8161b5ad019f2eee569ba992f0)
1#!/usr/bin/wish
2
3# This file provides many valuable stack inquiry procedures like get_file_proc_names, get_stack_var, etc..
4
5my_source [list print.tcl]
6
7
8proc get_file_proc_names { file_path { name_regex "" } } {
9
10  # Get all proc names from the file indicated by file_path and return them as a list.
11
12  # Description of argument(s):
13  # file_path                       The path to the file whose proc names are to be retrieved.
14  # name_regex                      A regular expression to be used to narrow the result to just the desired
15  #                                 procs.
16
17  # The first sed command serves to eliminate curly braces from the target file.  They are a distraction to
18  # what we are trying to do.
19  # TCL proc lines begin with...
20  # - Zero or more spaces...
21  # - The "proc" keyword...
22  # - One or more spaces...
23  set proc_regex "^\[ \]*proc\[ \]+"
24  set cmd_buf "sed -re 's/\[\\\{\\\}]//g' $file_path | egrep"
25  append cmd_buf " '${proc_regex}${name_regex}[ ]' | sed -re"
26  append cmd_buf " 's/${proc_regex}(\[^ \]+).*/\\1/g'"
27  return [split [eval exec bash -c {$cmd_buf}] "\n"]
28
29}
30
31
32proc get_stack_var { var_name { default {} } { init_stack_ix 1 } } {
33
34  # Starting with the caller's stack level, search upward in the call stack, for a variable named
35  # "${var_name}" and return its value.  If the variable cannot be found, return ${default}.
36
37  # Description of argument(s):
38  # var_name                        The name of the variable be searched for.
39  # default                         The value to return if the the variable cannot be found.
40
41  for {set stack_ix $init_stack_ix} {$stack_ix <= [info level]} \
42      {incr stack_ix} {
43    upvar $stack_ix $var_name var_ref
44    if { [info exists var_ref] } { return $var_ref }
45  }
46
47  return $default
48
49}
50
51
52proc get_stack_var_level { var_name { init_stack_ix 1 } { fail_on_err 1 } } {
53
54  # Starting with the caller's stack level, search upward in the call stack, for a variable named
55  # "${var_name}" and return its associated stack level.  If the variable cannot be found, return -1.
56
57  # Description of argument(s):
58  # var_name                        The name of the variable be searched for.
59  # init_stack_ix                   The level of the stack where the search should start.  The default is 1
60  #                                 which is the caller's stack level.
61  # fail_on_err                     Indicates that if the variable cannot be found on the stack, this proc
62  #                                 should write to stderr and exit with a non-zero return code.
63
64  for {set stack_ix $init_stack_ix} {$stack_ix <= [info level]} \
65      {incr stack_ix} {
66    upvar $stack_ix $var_name var_ref
67    set stack_level [expr $stack_ix - $init_stack_ix]
68    if { [info exists var_ref] } { return $stack_level }
69  }
70
71  if { $fail_on_err } {
72    append message "Programmer error - Couldn't find variable \"${var_name}\""
73    append message " on the stack."
74    print_error_report $message
75    exit 1
76  }
77
78  return -1
79
80}
81
82
83proc get_stack_proc_name { { level -1 } { include_args 0 } } {
84
85  # Get the name of the procedure at the indicated call stack level and return it.
86
87  # Description of argument(s):
88  # level                           The call stack level: 0 would mean this procedure's level (i.e.
89  #                                 get_stack_proc_name's level), -1 would indicate the caller's level, etc.
90  # include_args                    Indicates whether proc arg values should be included in the result.
91
92  # Set default.
93  set_var_default level -1
94
95  if { $include_args } {
96    set cmd_buf "set proc_name \[info level $level\]"
97  } else {
98    set cmd_buf "set proc_name \[lindex \[info level $level\] 0\]"
99  }
100
101  if { [ catch $cmd_buf result ] } {
102    # The command failed most likely due to being called from "main".
103    set proc_name "main"
104  }
105
106  return $proc_name
107
108}
109
110
111proc get_call_stack { { stack_top_ix -1 } { include_args 0 } } {
112
113  # Return the call stack as a list of procedure names.
114
115  # Example:
116  # set call_stack [get_call_stack 0]
117  # call_stack: get_call_stack calc_wrap_stack_ix_adjust sprint_var sprint_vars print_vars
118
119  # Description of argument(s):
120  # stack_top_ix                    The index to the bottom of the stack to be returned.  0 means include the
121  #                                 entire stack.  1 means include the entire stack with the exception of
122  #                                 this procedure itself, etc.
123  # include_args                    Indicates whether proc args should be included in the result.
124
125  set_var_default stack_top_ix -1
126
127  # Get the current stack size.
128  set stack_size [info level]
129  # Calculate stack_bottom_ix.  Example:  if stack_size is 5, stack_bottom_ix is -4.
130  set stack_bottom_ix [expr 1 - $stack_size]
131  for {set stack_ix $stack_top_ix} {$stack_ix >= $stack_bottom_ix} \
132      {incr stack_ix -1} {
133    if { $include_args } {
134      set proc_name [info level $stack_ix]
135    } else {
136      set proc_name [lindex [info level $stack_ix] 0]
137    }
138    lappend call_stack $proc_name
139  }
140
141  return $call_stack
142
143}
144