2018-07-24 11:35:49 +03:00
|
|
|
module Mom
|
|
|
|
|
|
|
|
# A BlockYield calls an argument block. All we need to know is the index
|
|
|
|
# of the argument, and the rest is almost as simple as a SimpleCall
|
|
|
|
|
|
|
|
class BlockYield < Instruction
|
|
|
|
attr :arg_index
|
|
|
|
|
2019-08-10 21:59:31 +03:00
|
|
|
# pass in the source (vool statement) and the index.
|
|
|
|
# The index is the argument index of the block that we call
|
|
|
|
def initialize(source , index)
|
|
|
|
super(source)
|
2018-07-24 11:35:49 +03:00
|
|
|
@arg_index = index
|
|
|
|
end
|
|
|
|
|
|
|
|
def to_s
|
|
|
|
"BlockYield[#{arg_index}] "
|
|
|
|
end
|
|
|
|
|
2019-08-10 21:59:31 +03:00
|
|
|
# almost as simple as a SimpleCall, use a dynamic_jump to get there
|
2018-07-24 11:35:49 +03:00
|
|
|
def to_risc(compiler)
|
2018-08-16 08:58:49 +03:00
|
|
|
return_label = Risc.label("block_yield", "continue_#{object_id}")
|
|
|
|
index = arg_index
|
|
|
|
compiler.build("BlockYield") do
|
|
|
|
next_message! << message[:next_message]
|
|
|
|
return_address! << return_label
|
|
|
|
next_message[:return_address] << return_address
|
|
|
|
|
|
|
|
block_reg! << message[:arguments]
|
|
|
|
block_reg << block_reg[index]
|
|
|
|
|
|
|
|
message << message[:next_message]
|
|
|
|
add_code Risc::DynamicJump.new("block_yield", block_reg )
|
|
|
|
add_code return_label
|
|
|
|
end
|
2018-07-24 11:35:49 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|