@@ -86,13 +86,13 @@ finish_stack!(frame::Frame, istoplevel::Bool=false) = finish_stack!(finish_and_r
86
86
pc = next_until!(predicate, frame, istoplevel=false)
87
87
88
88
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
90
90
evaluation terminates, `nothing` (if the frame reached a `return`), or a `BreakpointRef`.
91
91
"""
92
92
function next_until! (@nospecialize (predicate), @nospecialize (recurse), frame:: Frame , istoplevel:: Bool = false )
93
93
pc = step_expr! (recurse, frame, istoplevel)
94
94
while pc != = nothing && ! isa (pc, BreakpointRef)
95
- if predicate (pc_expr ( frame, pc) ) || shouldbreak (frame, pc)
95
+ if predicate (frame) || shouldbreak (frame, pc)
96
96
return pc
97
97
end
98
98
pc = step_expr! (recurse, frame, istoplevel)
102
102
next_until! (predicate, frame:: Frame , istoplevel:: Bool = false ) =
103
103
next_until! (predicate, finish_and_return!, frame, istoplevel)
104
104
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
+
105
119
"""
106
120
pc = next_call!(recurse, frame, istoplevel=false)
107
121
pc = next_call!(frame, istoplevel=false)
@@ -110,7 +124,7 @@ Execute the current statement. Continue stepping through `frame` until the next
110
124
`:return` or `:call` expression.
111
125
"""
112
126
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)
114
128
next_call! (frame:: Frame , istoplevel:: Bool = false ) = next_call! (finish_and_return!, frame, istoplevel)
115
129
116
130
"""
@@ -120,13 +134,9 @@ next_call!(frame::Frame, istoplevel::Bool=false) = next_call!(finish_and_return!
120
134
Return the current program counter of `frame` if it is a `:return` or `:call` expression.
121
135
Otherwise, step through the statements of `frame` until the next `:return` or `:call` expression.
122
136
"""
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)
130
140
131
141
"""
132
142
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
136
146
or defines a new method.
137
147
"""
138
148
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 ) )
140
150
pc = next_until! (predicate, recurse, frame, true )
141
151
(pc === nothing || isa (pc, BreakpointRef)) && return pc
142
152
return step_expr! (recurse, frame, true ) # define the method and return
@@ -162,17 +172,9 @@ to obtain the new execution frame.
162
172
function next_line! (@nospecialize (recurse), frame:: Frame , istoplevel:: Bool = false )
163
173
pc = frame. pc
164
174
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
176
178
maybe_step_through_kwprep! (recurse, frame, istoplevel)
177
179
maybe_next_call! (recurse, frame, istoplevel)
178
180
end
0 commit comments