2018-03-15 06:54:14 +01:00
|
|
|
require_relative "normalizer"
|
2017-04-01 20:28:57 +02:00
|
|
|
module Vool
|
|
|
|
class IfStatement < Statement
|
2018-03-15 06:54:14 +01:00
|
|
|
include Normalizer
|
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
|
|
|
|
|
2018-03-15 08:16:56 +01:00
|
|
|
def normalize
|
|
|
|
cond , rest = *normalize_name(@condition)
|
|
|
|
fals = @if_false ? @if_false.normalize : nil
|
|
|
|
me = IfStatement.new(cond , @if_true.normalize, fals)
|
2018-03-15 06:54:14 +01:00
|
|
|
return me unless rest
|
|
|
|
rest << me
|
2018-03-16 14:35:22 +01:00
|
|
|
rest
|
2018-03-15 06:54:14 +01:00
|
|
|
end
|
|
|
|
|
2018-07-05 13:02:38 +02:00
|
|
|
def to_mom( compiler )
|
|
|
|
if_false ? full_if(compiler) : simple_if(compiler)
|
2018-04-19 09:34:15 +02:00
|
|
|
end
|
|
|
|
|
2018-07-05 13:02:38 +02:00
|
|
|
def simple_if(compiler)
|
2018-05-01 18:20:16 +02:00
|
|
|
true_label = Mom::Label.new( "true_label_#{object_id.to_s(16)}")
|
|
|
|
merge_label = Mom::Label.new( "merge_label_#{object_id.to_s(16)}")
|
2018-04-19 09:34:15 +02:00
|
|
|
|
2018-07-05 13:02:38 +02:00
|
|
|
head = Mom::TruthCheck.new(condition.slot_definition(compiler) , merge_label)
|
2018-04-19 09:34:15 +02:00
|
|
|
head << true_label
|
2018-07-05 13:02:38 +02:00
|
|
|
head << if_true.to_mom(compiler)
|
2018-04-19 09:34:15 +02:00
|
|
|
head << merge_label
|
|
|
|
end
|
|
|
|
|
2018-07-05 13:02:38 +02:00
|
|
|
def full_if(compiler)
|
2018-05-01 18:20:16 +02:00
|
|
|
true_label = Mom::Label.new( "true_label_#{object_id.to_s(16)}")
|
|
|
|
false_label = Mom::Label.new( "false_label_#{object_id.to_s(16)}")
|
|
|
|
merge_label = Mom::Label.new( "merge_label_#{object_id.to_s(16)}")
|
2018-03-16 14:35:22 +01:00
|
|
|
|
2018-07-05 13:02:38 +02:00
|
|
|
head = Mom::TruthCheck.new(condition.slot_definition(compiler) , false_label)
|
2018-03-16 14:35:22 +01:00
|
|
|
head << true_label
|
2018-07-05 13:02:38 +02:00
|
|
|
head << if_true.to_mom(compiler)
|
2018-04-19 09:34:15 +02:00
|
|
|
head << Mom::Jump.new(merge_label)
|
2018-03-16 14:35:22 +01:00
|
|
|
head << false_label
|
2018-07-05 13:02:38 +02:00
|
|
|
head << if_false.to_mom(compiler)
|
2018-04-19 09:34:15 +02:00
|
|
|
head << merge_label
|
2018-03-15 16:03:38 +01:00
|
|
|
end
|
2017-08-30 16:21:13 +02:00
|
|
|
|
2018-03-15 16:10:21 +01:00
|
|
|
def each(&block)
|
|
|
|
block.call(condition)
|
|
|
|
@if_true.each(&block)
|
|
|
|
@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)
|
|
|
|
parts = ["if (#{@condition})" , @body.to_s(depth + 1) ]
|
|
|
|
parts += ["else" , "@if_false.to_s(depth + 1)"] if(@false)
|
|
|
|
parts << "end"
|
|
|
|
at_depth(depth , *parts )
|
|
|
|
end
|
2017-04-01 20:28:57 +02:00
|
|
|
end
|
|
|
|
end
|