2014-06-25 01:47:59 +02:00
|
|
|
module Ast
|
|
|
|
class WhileExpression < Expression
|
|
|
|
# attr_reader :condition, :body
|
2014-07-24 13:56:27 +02:00
|
|
|
def compile method , frame
|
2014-07-17 15:46:17 +02:00
|
|
|
start = Virtual::Label.new("while_start")
|
|
|
|
method.add start
|
2014-07-24 13:56:27 +02:00
|
|
|
is = condition.compile(method,frame)
|
2014-07-17 15:46:17 +02:00
|
|
|
branch = Virtual::ImplicitBranch.new "while"
|
|
|
|
merge = Virtual::Label.new(branch.name)
|
|
|
|
branch.other = merge #false jumps to end of while
|
|
|
|
method.add branch
|
|
|
|
last = is
|
|
|
|
body.each do |part|
|
2014-07-24 13:56:27 +02:00
|
|
|
last = part.compile(method,frame )
|
2014-07-17 15:46:17 +02:00
|
|
|
raise part.inspect if last.nil?
|
|
|
|
end
|
|
|
|
# unconditionally brnach to the start
|
2014-07-22 22:27:13 +02:00
|
|
|
merge.next = method.current.next
|
|
|
|
method.current.next = start
|
2014-07-17 15:46:17 +02:00
|
|
|
# here we add the end of while that the branch jumps to
|
|
|
|
#but don't link it in (not using add)
|
|
|
|
method.current = merge
|
|
|
|
last
|
2014-07-14 15:19:47 +02:00
|
|
|
end
|
|
|
|
def old
|
2014-06-25 01:47:59 +02:00
|
|
|
into = context.function
|
|
|
|
ret = into.new_block "while_end"
|
|
|
|
while_block = into.new_block "while_start"
|
|
|
|
into.insert_at while_block
|
|
|
|
|
|
|
|
puts "compiling while condition #{condition}"
|
|
|
|
cond_val = condition.compile(context)
|
|
|
|
into.b ret , condition_code: cond_val.not_operator
|
|
|
|
into.insertion_point.branch = ret
|
|
|
|
|
|
|
|
last = nil
|
|
|
|
|
|
|
|
body.each do |part|
|
|
|
|
puts "compiling in while #{part}"
|
|
|
|
last = part.compile(context)
|
|
|
|
end
|
|
|
|
into.b while_block
|
|
|
|
into.insertion_point.branch = while_block
|
|
|
|
|
|
|
|
puts "compile while end"
|
|
|
|
into.insert_at ret
|
|
|
|
return last
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
end
|