2018-07-09 16:48:23 +03:00
|
|
|
module Risc
|
|
|
|
|
|
|
|
# A BlockCompiler is much like a Mehtodcompiler, exept for blocks
|
|
|
|
#
|
2018-07-10 22:03:32 +03:00
|
|
|
class BlockCompiler < CallableCompiler
|
2018-07-09 16:48:23 +03:00
|
|
|
|
|
|
|
attr_reader :block , :risc_instructions , :constants
|
2018-07-30 10:26:11 +03:00
|
|
|
alias :block :callable
|
2018-07-09 16:48:23 +03:00
|
|
|
|
|
|
|
def initialize( block , method)
|
|
|
|
@method = method
|
2018-07-30 10:26:11 +03:00
|
|
|
super(block)
|
2018-07-10 22:03:32 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def source_name
|
|
|
|
"#{@method.self_type.name}.init"
|
2018-07-09 16:48:23 +03:00
|
|
|
end
|
|
|
|
|
2018-07-17 10:37:33 +03:00
|
|
|
# resolve the type of the slot, by inferring from it's name, using the type
|
|
|
|
# scope related slots are resolved by the compiler by method/block
|
|
|
|
#
|
|
|
|
# This mainly calls super, and only for :caller adds extra info
|
|
|
|
# Using the info, means assuming that the block is not passed around (FIXME in 2020)
|
|
|
|
def slot_type( slot , type)
|
|
|
|
new_type = super
|
|
|
|
if slot == :caller
|
|
|
|
extra_info = { type_frame: @method.frame_type ,
|
|
|
|
type_arguments: @method.arguments_type ,
|
|
|
|
type_self: @method.self_type}
|
|
|
|
end
|
|
|
|
return new_type , extra_info
|
|
|
|
end
|
2018-07-09 17:53:56 +03:00
|
|
|
# determine how given name need to be accsessed.
|
|
|
|
# For blocks the options are args or frame
|
|
|
|
# or then the methods arg or frame
|
|
|
|
def slot_type_for(name)
|
2018-07-30 10:26:11 +03:00
|
|
|
if @callable.arguments_type.variable_index(name)
|
2018-07-16 12:03:40 +03:00
|
|
|
slot_def = [:arguments]
|
2018-07-30 10:26:11 +03:00
|
|
|
elsif @callable.frame_type.variable_index(name)
|
2018-07-09 17:53:56 +03:00
|
|
|
slot_def = [:frame]
|
|
|
|
elsif @method.arguments_type.variable_index(name)
|
2018-07-31 18:00:42 +03:00
|
|
|
slot_def = [:caller , :caller ,:arguments ]
|
2018-07-18 10:13:19 +03:00
|
|
|
elsif @method.frame_type.variable_index(name)
|
2018-07-31 18:00:42 +03:00
|
|
|
slot_def = [:caller ,:caller , :frame ]
|
2018-07-09 17:53:56 +03:00
|
|
|
elsif
|
|
|
|
raise "no variable #{name} , need to resolve at runtime"
|
|
|
|
end
|
|
|
|
slot_def << name
|
|
|
|
end
|
|
|
|
|
2018-07-16 12:03:40 +03:00
|
|
|
# return the frame type, ie the blocks frame type
|
|
|
|
def frame_type
|
2018-07-30 10:26:11 +03:00
|
|
|
@callable.frame_type
|
2018-07-16 12:03:40 +03:00
|
|
|
end
|
|
|
|
# return the frame type, ie the blocks arguments type
|
|
|
|
def arg_type
|
2018-07-30 10:26:11 +03:00
|
|
|
@callable.arguments_type
|
2018-07-16 12:03:40 +03:00
|
|
|
end
|
|
|
|
# return the frame type, ie the blocks self_type
|
|
|
|
def receiver_type
|
2018-07-30 10:26:11 +03:00
|
|
|
@callable.self_type
|
2018-07-10 22:03:32 +03:00
|
|
|
end
|
|
|
|
|
2018-07-09 16:48:23 +03:00
|
|
|
end
|
|
|
|
end
|