2019-10-03 23:36:49 +02:00
|
|
|
module Sol
|
2017-04-01 20:28:57 +02:00
|
|
|
class ReturnStatement < Statement
|
2018-03-16 14:56:27 +01:00
|
|
|
|
2017-04-06 15:06:51 +02:00
|
|
|
attr_reader :return_value
|
2017-04-04 09:42:20 +02:00
|
|
|
|
|
|
|
def initialize(value)
|
|
|
|
@return_value = value
|
|
|
|
end
|
2017-04-08 11:10:42 +02:00
|
|
|
|
2018-03-16 14:56:27 +01:00
|
|
|
def each(&block)
|
2019-08-17 14:58:27 +02:00
|
|
|
block.call(self)
|
|
|
|
@return_value.each(&block)
|
2017-04-08 11:10:42 +02:00
|
|
|
end
|
2017-04-14 09:52:23 +02:00
|
|
|
|
2018-03-16 14:56:27 +01:00
|
|
|
# Since the return is normalized to only allow simple values it is simple.
|
2019-10-03 20:07:55 +02:00
|
|
|
# To return form a method in slot_machine instructions we only need to do two things:
|
2018-03-16 14:56:27 +01:00
|
|
|
# - store the given return value, this is a SlotMove
|
2017-04-14 20:01:50 +02:00
|
|
|
# - activate return sequence (reinstantiate old message and jump to return address)
|
2019-10-03 19:55:41 +02:00
|
|
|
def to_slot( compiler )
|
2019-08-16 19:39:08 +02:00
|
|
|
if @return_value.is_a?(CallStatement)
|
2019-10-03 19:55:41 +02:00
|
|
|
ret = @return_value.to_slot(compiler)
|
2019-08-25 13:40:59 +02:00
|
|
|
ret << slot_load(compiler)
|
2019-08-16 17:42:57 +02:00
|
|
|
else
|
2019-08-25 13:40:59 +02:00
|
|
|
ret = slot_load(compiler)
|
2019-08-16 17:42:57 +02:00
|
|
|
end
|
2019-10-03 19:55:41 +02:00
|
|
|
ret << SlotMachine::ReturnJump.new(self , compiler.return_label )
|
2017-04-14 09:52:23 +02:00
|
|
|
end
|
|
|
|
|
2018-07-03 21:18:19 +02:00
|
|
|
def to_s(depth = 0)
|
|
|
|
at_depth(depth , "return #{@return_value.to_s}")
|
|
|
|
end
|
2019-08-25 13:40:59 +02:00
|
|
|
|
|
|
|
def slot_load(compiler)
|
2019-10-03 19:55:41 +02:00
|
|
|
SlotMachine::SlotLoad.new( self , [:message , :return_value] ,
|
2020-02-17 08:29:45 +01:00
|
|
|
@return_value.to_slotted(compiler) )
|
2019-08-25 13:40:59 +02:00
|
|
|
end
|
2017-04-01 20:28:57 +02:00
|
|
|
end
|
|
|
|
end
|