Skip to content

Commit a652d66

Browse files
authored
generalize next_until! to take a frame and add maybe_next_until! (#269)
* generalize next_until! to take a frame and add maybe_next_until! * Update src/commands.jl Co-Authored-By: KristofferC <[email protected]>
1 parent b4f91ad commit a652d66

File tree

2 files changed

+26
-23
lines changed

2 files changed

+26
-23
lines changed

docs/src/dev_reference.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -38,13 +38,14 @@ JuliaInterpreter.finish_and_return!
3838
JuliaInterpreter.finish_stack!
3939
JuliaInterpreter.get_return
4040
JuliaInterpreter.next_until!
41+
JuliaInterpreter.maybe_next_until!
4142
JuliaInterpreter.through_methoddef_or_done!
4243
JuliaInterpreter.evaluate_call!
4344
JuliaInterpreter.evaluate_foreigncall
4445
JuliaInterpreter.maybe_evaluate_builtin
46+
JuliaInterpreter.next_call!
4547
JuliaInterpreter.maybe_next_call!
4648
JuliaInterpreter.next_line!
47-
JuliaInterpreter.next_call!
4849
JuliaInterpreter.maybe_reset_frame!
4950
JuliaInterpreter.maybe_step_through_wrapper!
5051
JuliaInterpreter.maybe_step_through_kwprep!

src/commands.jl

+24-22
Original file line numberDiff line numberDiff line change
@@ -86,13 +86,13 @@ finish_stack!(frame::Frame, istoplevel::Bool=false) = finish_stack!(finish_and_r
8686
pc = next_until!(predicate, frame, istoplevel=false)
8787
8888
Execute the current statement. Then step through statements of `frame` until the next
89-
statement satifies `predicate(stmt)`. `pc` will be the index of the statement at which
89+
statement satisfies `predicate(frame)`. `pc` will be the index of the statement at which
9090
evaluation terminates, `nothing` (if the frame reached a `return`), or a `BreakpointRef`.
9191
"""
9292
function next_until!(@nospecialize(predicate), @nospecialize(recurse), frame::Frame, istoplevel::Bool=false)
9393
pc = step_expr!(recurse, frame, istoplevel)
9494
while pc !== nothing && !isa(pc, BreakpointRef)
95-
if predicate(pc_expr(frame, pc)) || shouldbreak(frame, pc)
95+
if predicate(frame) || shouldbreak(frame, pc)
9696
return pc
9797
end
9898
pc = step_expr!(recurse, frame, istoplevel)
@@ -102,6 +102,20 @@ end
102102
next_until!(predicate, frame::Frame, istoplevel::Bool=false) =
103103
next_until!(predicate, finish_and_return!, frame, istoplevel)
104104

105+
"""
106+
pc = maybe_next_until!(predicate, recurse, frame, istoplevel=false)
107+
pc = maybe_next_until!(predicate, frame, istoplevel=false)
108+
109+
Like [`next_until!`](@ref) except checks `predicate` before executing the current statment.
110+
111+
"""
112+
function maybe_next_until!(@nospecialize(predicate), @nospecialize(recurse), frame::Frame, istoplevel::Bool=false)
113+
predicate(frame) && return frame.pc
114+
return next_until!(predicate, recurse, frame, istoplevel)
115+
end
116+
maybe_next_until!(@nospecialize(predicate), frame::Frame, istoplevel::Bool=false) =
117+
maybe_next_until!(predicate, finish_and_return!, frame, istoplevel)
118+
105119
"""
106120
pc = next_call!(recurse, frame, istoplevel=false)
107121
pc = next_call!(frame, istoplevel=false)
@@ -110,7 +124,7 @@ Execute the current statement. Continue stepping through `frame` until the next
110124
`:return` or `:call` expression.
111125
"""
112126
next_call!(@nospecialize(recurse), frame::Frame, istoplevel::Bool=false) =
113-
next_until!(is_call_or_return, recurse, frame, istoplevel)
127+
next_until!(frame -> is_call_or_return(pc_expr(frame)), recurse, frame, istoplevel)
114128
next_call!(frame::Frame, istoplevel::Bool=false) = next_call!(finish_and_return!, frame, istoplevel)
115129

116130
"""
@@ -120,13 +134,9 @@ next_call!(frame::Frame, istoplevel::Bool=false) = next_call!(finish_and_return!
120134
Return the current program counter of `frame` if it is a `:return` or `:call` expression.
121135
Otherwise, step through the statements of `frame` until the next `:return` or `:call` expression.
122136
"""
123-
function maybe_next_call!(@nospecialize(recurse), frame::Frame, istoplevel::Bool=false)
124-
pc = frame.pc
125-
is_call_or_return(pc_expr(frame, pc)) && return pc
126-
return next_call!(recurse, frame, istoplevel)
127-
end
128-
maybe_next_call!(frame::Frame, istoplevel::Bool=false) =
129-
maybe_next_call!(finish_and_return!, frame, istoplevel)
137+
maybe_next_call!(@nospecialize(recurse), frame::Frame, istoplevel::Bool=false) =
138+
maybe_next_until!(frame -> is_call_or_return(pc_expr(frame)), recurse, frame, istoplevel)
139+
maybe_next_call!(frame::Frame, istoplevel::Bool=false) = maybe_next_call!(finish_and_return!, frame, istoplevel)
130140

131141
"""
132142
pc = through_methoddef_or_done!(recurse, frame)
@@ -136,7 +146,7 @@ Runs `frame` at top level until it either finishes (e.g., hits a `return` statem
136146
or defines a new method.
137147
"""
138148
function through_methoddef_or_done!(@nospecialize(recurse), frame::Frame)
139-
predicate(stmt) = isexpr(stmt, :method, 3) || isexpr(stmt, :thunk)
149+
predicate(frame) = (stmt = pc_expr(frame); isexpr(stmt, :method, 3) || isexpr(stmt, :thunk))
140150
pc = next_until!(predicate, recurse, frame, true)
141151
(pc === nothing || isa(pc, BreakpointRef)) && return pc
142152
return step_expr!(recurse, frame, true) # define the method and return
@@ -162,17 +172,9 @@ to obtain the new execution frame.
162172
function next_line!(@nospecialize(recurse), frame::Frame, istoplevel::Bool=false)
163173
pc = frame.pc
164174
initialline, initialfile = linenumber(frame, pc), getfile(frame, pc)
165-
first = true
166-
while linenumber(frame, pc) == initialline && getfile(frame, pc) == initialfile
167-
# If this is a return node, interrupt execution
168-
expr = pc_expr(frame, pc)
169-
(!first && isexpr(expr, :return)) && return pc
170-
first = false
171-
172-
pc = step_expr!(recurse, frame, istoplevel)
173-
(pc === nothing || isa(pc, BreakpointRef)) && return pc
174-
shouldbreak(frame, pc) && return BreakpointRef(frame.framecode, pc)
175-
end
175+
predicate(frame) = isexpr(pc_expr(frame), :return) || (linenumber(frame) != initialline || getfile(frame) != initialfile)
176+
pc = next_until!(predicate, frame, istoplevel)
177+
(pc === nothing || isa(pc, BreakpointRef)) && return pc
176178
maybe_step_through_kwprep!(recurse, frame, istoplevel)
177179
maybe_next_call!(recurse, frame, istoplevel)
178180
end

0 commit comments

Comments
 (0)