1 # This file is part of NIT ( http://www.nitlanguage.org ).
3 # Copyright 2013 Lucas Bajolet <lucas.bajolet@gmail.com>
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
9 # http://www.apache.org/licenses/LICENSE-2.0
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
17 # Debugging of a nit program using the NaiveInterpreter
20 intrude import naive_interpreter
22 redef class ToolContext
24 var opt_debugger_mode
: OptionBool = new OptionBool("Launches the target program with the debugger attached to it", "-d")
29 self.option_context
.add_option
(self.opt_debugger_mode
)
33 redef class ModelBuilder
34 # Execute the program from the entry point (Sys::main) of the `mainmodule'
35 # `arguments' are the command-line arguments in order
37 # 1. the AST is fully loaded.
38 # 2. the model is fully built.
39 # 3. the instructions are fully analysed.
40 fun run_debugger
(mainmodule
: MModule, arguments
: Array[String])
43 self.toolcontext
.info
("*** START INTERPRETING ***", 1)
45 var interpreter
= new Debugger(self, mainmodule
, arguments
)
47 init_naive_interpreter
(interpreter
, mainmodule
)
50 self.toolcontext
.info
("*** END INTERPRETING: {time1-time0} ***", 2)
54 # The class extending NaiveInterpreter by adding debugging methods
56 super NaiveInterpreter
58 # Keeps the frame count in memory to find when to stop
59 # and launch the command prompt after a step out call
60 var step_stack_count
= 1
62 # Triggers a step over an instruction in a nit program
63 var stop_after_step_over_trigger
= true
65 # Main loop, every call to a debug command is done here
66 redef fun stmt
(n
: nullable AExpr)
68 if n
== null then return
70 var frame
= self.frame
71 var old
= frame
.current_node
72 frame
.current_node
= n
77 frame
.current_node
= old
80 private fun steps_fun_call
(n
: AExpr)
82 if self.stop_after_step_over_trigger
then
83 if self.frames
.length
<= self.step_stack_count
then
84 n
.debug
("Execute stmt {n.to_s}")
85 while process_debug_command
(gets
) do end
90 #######################################################################
91 ## Processing commands functions ##
92 #######################################################################
94 # Takes a user command as a parameter
96 # Returns a boolean value, representing whether or not to
97 # continue reading commands from the console input
98 fun process_debug_command
(command
:String): Bool
103 # Kills the current program
104 if command
== "kill" then
107 else if command
== "n" then
109 # Continues execution until the end
110 else if command
== "c" then
113 var parts_of_command
= command
.split_with
(' ')
114 # Shows the value of a variable in the current frame
115 if parts_of_command
[0] == "p" or parts_of_command
[0] == "print" then
116 print_command
(parts_of_command
)
122 # Sets the flags to step-over an instruction in the current file
125 self.step_stack_count
= frames
.length
126 self.stop_after_step_over_trigger
= true
130 # Sets the flags to continue execution
131 fun continue_exec
: Bool
133 self.stop_after_step_over_trigger
= false
137 # Prints the demanded variable in the command
139 # The name of the variable in in position 1 of the array 'parts_of_command'
140 fun print_command
(parts_of_command
: Array[String])
142 if parts_of_command
[1] == "*" then
143 var map_of_instances
= frame
.map
145 var keys
= map_of_instances
.iterator
147 print
"Variables collection : \n"
149 for instance
in map_of_instances
.keys
do
150 print
"Variable {instance.to_s}, Instance {map_of_instances[instance].to_s}"
153 print
"\nEnd of current instruction \n"
155 var instance
= seek_variable
(parts_of_command
[1], frame
)
159 print_instance
(instance
)
164 #######################################################################
165 ## Print functions ##
166 #######################################################################
168 # Prints an object instance and its attributes if it has some
170 # If it is a primitive type, its value is directly printed
171 fun print_instance
(instance
: Instance)
173 if instance
isa MutableInstance then
174 var attributes
= instance
.attributes
175 print
"Object : {instance}"
177 for current_attribute
in attributes
.keys
do
178 print
"Attribute : {current_attribute.to_s} \nValeur : {attributes[current_attribute].to_s}"
181 print
"Found variable {instance}"
185 #######################################################################
186 ## Variable Exploration functions ##
187 #######################################################################
189 # Seeks a variable from the current frame called 'variable_path', can introspect complex objects using function get_variable_in_mutable_instance
190 private fun seek_variable
(variable_path
: String, frame
: Frame): nullable Instance
192 var full_variable
= variable_path
.split_with
(".")
194 var full_variable_iterator
= full_variable
.iterator
196 var first_instance
= get_variable_in_frame
(full_variable_iterator
.item
, frame
)
198 if first_instance
== null then return null
200 if full_variable
.length
<= 1 then return first_instance
202 full_variable_iterator
.next
204 if not (first_instance
isa MutableInstance and full_variable_iterator
.is_ok
) then return null
206 return get_variable_in_mutable_instance
(first_instance
, full_variable_iterator
)
209 # Gets a variable 'variable_name' contained in the frame 'frame'
210 private fun get_variable_in_frame
(variable_name
: String, frame
: Frame): nullable Instance
212 if variable_name
== "self" then
213 if frame
.arguments
.length
>= 1 then return frame
.arguments
.first
216 var map_of_instances
= frame
.map
218 for key
in map_of_instances
.keys
do
219 if key
.to_s
== variable_name
then
220 return map_of_instances
[key
]
227 # Gets an attribute 'attribute_name' contained in variable 'variable'
228 fun get_attribute_in_mutable_instance
(variable
: MutableInstance, attribute_name
: String): nullable MAttribute
230 var map_of_attributes
= variable
.attributes
232 for key
in map_of_attributes
.keys
do
233 if key
.to_s
.substring_from
(1) == attribute_name
then
241 # Recursive function, returns the variable described by 'total_chain'
242 fun get_variable_in_mutable_instance
(variable
: MutableInstance, iterator
: Iterator[String]): nullable Instance
244 var attribute
= get_attribute_in_mutable_instance
(variable
, iterator
.item
)
246 if attribute
== null then return null
250 if iterator
.is_ok
then
251 var new_variable
= variable
.attributes
[attribute
]
252 if new_variable
isa MutableInstance then
253 return get_variable_in_mutable_instance
(new_variable
, iterator
)
258 return variable
.attributes
[attribute
]