rubyx/lib/slot_machine/block_compiler.rb

53 lines
1.7 KiB
Ruby
Raw Normal View History

2019-10-03 21:07:55 +03:00
module SlotMachine
2020-02-26 19:01:01 +02:00
# A BlockCompiler is much like a MethodCompiler, except for blocks
# Blocks are much like methods (hence the base classes) , just the
# scope for accessing variables is different. (This only applies to static
# blocks, not ones being passed around. That is undone because it needs runtime
# code to resolve variables)
#
class BlockCompiler < CallableCompiler
2019-10-03 21:07:55 +03:00
attr_reader :block , :slot_instructions
alias :block :callable
def initialize( block , method)
@method = method
super(block)
end
def source_name
"#{@method.self_type.name}.init"
end
2020-02-26 19:01:01 +02:00
# Create the risc equivalent, which is a also a (Risc) BlockCompiler
# sintructions are converted from slot to risc in instructions_to_risc (in base class)
def to_risc
2019-10-03 21:07:55 +03:00
risc_compiler = Risc::BlockCompiler.new(@callable , @method , slot_instructions)
instructions_to_risc(risc_compiler)
#recursive blocks not done
risc_compiler
end
# 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)
2019-08-22 22:56:44 +03:00
if index = @callable.arguments_type.variable_index(name)
slot_def = ["arg#{index}".to_sym]
elsif index = @callable.frame_type.variable_index(name)
slot_def = ["local#{index}".to_sym]
2019-08-22 22:56:44 +03:00
elsif index = @method.arguments_type.variable_index(name)
slot_def = [:caller , :caller , "arg#{index}".to_sym]
elsif index = @method.frame_type.variable_index(name)
slot_def = [:caller ,:caller , "local#{index}".to_sym ]
elsif
raise "no variable #{name} , need to resolve at runtime"
end
return slot_def
end
end
end