module Bosl module Compiler # if - attr_reader :cond, :if_true, :if_false def self.compile_if expression , method condition , if_true , if_false = *expression condition = condition.first # to execute the logic as the if states it, the blocks are the other way around # so we can the jump over the else if true , # and the else joins unconditionally after the true_block 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 is = Compiler.compile(condition, method ) # TODO should/will use different branches for different conditions. # just a scetch : cond_val = cond_val.is_true?(method) unless cond_val.is_a? BranchCondition method.source.add_code Virtual::IsTrueBranch.new( true_block ) # compile the true block (as we think of it first, even it is second in sequential order) method.source.current true_block last = is if_true.each do |part| last = Compiler.compile(part,method ) raise part.inspect if last.nil? end # compile the false block method.source.current false_block if_false.each do |part| #puts "compiling in if false #{part}" last = Compiler.compile(part,method ) raise part.inspect if last.nil? end method.source.add_code Virtual::UnconditionalBranch.new( merge_block ) #puts "compiled if: end" method.source.current merge_block #TODO should return the union of the true and false types last end end end