2015-10-07 15:22:47 +03:00
|
|
|
module Phisol
|
2015-09-19 18:56:18 +03:00
|
|
|
Compiler.class_eval do
|
2015-05-04 14:22:22 +03:00
|
|
|
# if - attr_reader :cond, :if_true, :if_false
|
2014-08-13 20:05:32 +03:00
|
|
|
|
2015-10-09 18:06:00 +03:00
|
|
|
def on_if_statement statement
|
2015-10-09 17:51:14 +03:00
|
|
|
condition , if_true , if_false = *statement
|
2015-09-19 17:57:44 +03:00
|
|
|
condition = condition.first
|
2014-08-13 20:05:32 +03:00
|
|
|
# to execute the logic as the if states it, the blocks are the other way around
|
2015-05-30 12:20:39 +03:00
|
|
|
# so we can the jump over the else if true ,
|
|
|
|
# and the else joins unconditionally after the true_block
|
2015-10-06 00:27:13 +03:00
|
|
|
merge_block = @method.source.new_block "if_merge" # last one, created first
|
|
|
|
true_block = @method.source.new_block "if_true" # second, linked in after current, before merge
|
|
|
|
false_block = @method.source.new_block "if_false" # directly next in order, ie if we don't jump we land here
|
2015-05-04 14:22:22 +03:00
|
|
|
|
2015-10-07 10:02:51 +03:00
|
|
|
is = process(condition)
|
2015-05-04 14:22:22 +03:00
|
|
|
# TODO should/will use different branches for different conditions.
|
2015-05-08 15:19:30 +03:00
|
|
|
# just a scetch : cond_val = cond_val.is_true?(method) unless cond_val.is_a? BranchCondition
|
2015-10-07 10:02:51 +03:00
|
|
|
@method.source.add_code Register::IsZeroBranch.new( condition , true_block )
|
2014-08-13 20:05:32 +03:00
|
|
|
|
|
|
|
# compile the true block (as we think of it first, even it is second in sequential order)
|
2015-10-06 00:27:13 +03:00
|
|
|
@method.source.current true_block
|
2015-09-20 17:33:05 +03:00
|
|
|
|
2015-09-19 18:56:18 +03:00
|
|
|
last = process_all(if_true).last
|
2014-08-13 20:05:32 +03:00
|
|
|
|
|
|
|
# compile the false block
|
2015-10-06 00:27:13 +03:00
|
|
|
@method.source.current false_block
|
2015-09-20 17:33:05 +03:00
|
|
|
last = process_all(if_false).last if if_false
|
2015-10-09 17:51:14 +03:00
|
|
|
@method.source.add_code Register::AlwaysBranch.new(statement, merge_block )
|
2014-08-13 20:05:32 +03:00
|
|
|
|
2015-05-15 21:11:44 +03:00
|
|
|
#puts "compiled if: end"
|
2015-10-06 00:27:13 +03:00
|
|
|
@method.source.current merge_block
|
2014-08-13 20:05:32 +03:00
|
|
|
|
2014-07-16 20:16:40 +03:00
|
|
|
#TODO should return the union of the true and false types
|
2014-07-16 13:20:47 +03:00
|
|
|
last
|
2014-07-14 21:28:21 +03:00
|
|
|
end
|
2015-05-08 15:10:30 +03:00
|
|
|
end
|
2015-05-04 14:22:22 +03:00
|
|
|
end
|