2018-07-19 13:47:29 +02:00
|
|
|
|
2019-10-03 23:36:49 +02:00
|
|
|
module Sol
|
2017-04-01 20:28:57 +02:00
|
|
|
class IfStatement < Statement
|
2018-03-12 13:26:44 +01:00
|
|
|
|
2017-04-06 15:06:51 +02:00
|
|
|
attr_reader :condition , :if_true , :if_false
|
2017-04-02 18:12:42 +02:00
|
|
|
|
2017-08-30 16:21:13 +02:00
|
|
|
def initialize( cond , if_true , if_false = nil)
|
2017-04-02 18:12:42 +02:00
|
|
|
@condition = cond
|
2017-04-06 15:06:51 +02:00
|
|
|
@if_true = if_true
|
|
|
|
@if_false = if_false
|
|
|
|
end
|
|
|
|
|
2019-10-03 19:55:41 +02:00
|
|
|
def to_slot( compiler )
|
|
|
|
true_label = SlotMachine::Label.new( self , "true_label_#{object_id.to_s(16)}")
|
|
|
|
false_label = SlotMachine::Label.new( self , "false_label_#{object_id.to_s(16)}")
|
|
|
|
merge_label = SlotMachine::Label.new( self , "merge_label_#{object_id.to_s(16)}")
|
2018-03-16 14:35:22 +01:00
|
|
|
|
2019-08-16 19:39:08 +02:00
|
|
|
if @condition.is_a?(CallStatement)
|
2019-10-03 19:55:41 +02:00
|
|
|
head = @condition.to_slot(compiler)
|
2019-08-25 13:40:59 +02:00
|
|
|
head << check_slot(compiler , false_label)
|
2019-08-16 17:42:57 +02:00
|
|
|
else
|
2019-08-25 13:40:59 +02:00
|
|
|
head = check_slot(compiler , false_label)
|
2019-08-16 17:42:57 +02:00
|
|
|
end
|
2018-03-16 14:35:22 +01:00
|
|
|
head << true_label
|
2019-10-03 19:55:41 +02:00
|
|
|
head << if_true.to_slot(compiler) if @if_true
|
|
|
|
head << SlotMachine::Jump.new(merge_label) if @if_false
|
2018-03-16 14:35:22 +01:00
|
|
|
head << false_label
|
2019-10-03 19:55:41 +02:00
|
|
|
head << if_false.to_slot(compiler) if @if_false
|
2019-08-14 21:24:35 +02:00
|
|
|
head << merge_label if @if_false
|
|
|
|
head
|
2018-03-15 16:03:38 +01:00
|
|
|
end
|
2017-08-30 16:21:13 +02:00
|
|
|
|
2019-10-03 19:55:41 +02:00
|
|
|
# create the slot lazily, so to_slot gets called first
|
2019-08-25 13:40:59 +02:00
|
|
|
def check_slot(compiler , false_label)
|
2020-02-17 08:29:45 +01:00
|
|
|
SlotMachine::TruthCheck.new(@condition.to_slotted(compiler) , false_label)
|
2019-08-25 13:40:59 +02:00
|
|
|
end
|
|
|
|
|
2018-03-15 16:10:21 +01:00
|
|
|
def each(&block)
|
|
|
|
block.call(condition)
|
2019-08-14 21:24:35 +02:00
|
|
|
@if_true.each(&block) if @if_true
|
2018-03-15 16:10:21 +01:00
|
|
|
@if_false.each(&block) if @if_false
|
2017-04-08 11:10:42 +02:00
|
|
|
end
|
|
|
|
|
2017-04-02 18:12:42 +02:00
|
|
|
def has_false?
|
|
|
|
@if_false != nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def has_true?
|
|
|
|
@if_true != nil
|
|
|
|
end
|
2017-08-30 17:23:54 +02:00
|
|
|
|
2018-07-03 21:18:19 +02:00
|
|
|
def to_s(depth = 0)
|
2019-09-19 19:48:21 +02:00
|
|
|
parts = "if (#{@condition.to_s(0)})\n"
|
|
|
|
parts += " #{@if_true}\n" if @if_true
|
2019-10-02 16:42:24 +02:00
|
|
|
parts += "else\n" if(@if_false)
|
|
|
|
parts += " #{@if_false}\n" if(@if_false)
|
2019-09-19 19:48:21 +02:00
|
|
|
parts += "end\n"
|
|
|
|
at_depth(depth , parts )
|
2018-07-03 21:18:19 +02:00
|
|
|
end
|
2017-04-01 20:28:57 +02:00
|
|
|
end
|
|
|
|
end
|