2020-02-10 19:28:48 +07:00
|
|
|
module SlotMachine
|
2020-02-17 14:26:50 +07:00
|
|
|
class SlottedObject < Slotted
|
2020-02-10 19:28:48 +07:00
|
|
|
|
2020-02-15 15:26:49 +07:00
|
|
|
attr_reader :known_object
|
|
|
|
|
2020-02-10 19:28:48 +07:00
|
|
|
def initialize( object , slots)
|
2020-02-15 15:26:49 +07:00
|
|
|
super(slots)
|
|
|
|
@known_object = object
|
|
|
|
raise "Not known #{slots}" unless object
|
2020-02-10 19:28:48 +07:00
|
|
|
end
|
|
|
|
|
|
|
|
def known_name
|
|
|
|
known_object.class.short_name
|
|
|
|
end
|
|
|
|
|
|
|
|
# load the slots into a register
|
|
|
|
# the code is added to compiler
|
|
|
|
# the register returned
|
|
|
|
def to_register(compiler, source)
|
|
|
|
type = known_object.get_type
|
|
|
|
right = compiler.use_reg( type )
|
|
|
|
const = Risc.load_constant(source, known_object , right)
|
|
|
|
compiler.add_code const
|
2020-02-17 14:29:45 +07:00
|
|
|
if slots_length > 1
|
2020-02-10 19:28:48 +07:00
|
|
|
# desctructively replace the existing value to be loaded if more slots
|
2020-02-17 14:29:45 +07:00
|
|
|
compiler.add_code Risc.slot_to_reg( source , right ,slots.name, right)
|
2020-02-10 19:28:48 +07:00
|
|
|
end
|
2020-02-17 14:29:45 +07:00
|
|
|
if slots_length > 2
|
2020-02-10 19:28:48 +07:00
|
|
|
# desctructively replace the existing value to be loaded if more slots
|
2020-02-17 14:29:45 +07:00
|
|
|
index = Risc.resolve_to_index(slots.name , slots.next_slot.name ,compiler)
|
2020-02-10 19:28:48 +07:00
|
|
|
compiler.add_code Risc::SlotToReg.new( source , right ,index, right)
|
2020-02-17 14:29:45 +07:00
|
|
|
if slots_length > 3
|
|
|
|
raise "3 slots only for type #{slots}" unless slots.next_slot.next_slot.name == :type
|
2020-02-10 19:28:48 +07:00
|
|
|
compiler.add_code Risc::SlotToReg.new( source , right , Parfait::TYPE_INDEX, right)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return const.register
|
|
|
|
end
|
|
|
|
|
2020-02-11 16:03:51 +07:00
|
|
|
# Note: this is the left hand case, the right hand being to_register
|
|
|
|
# They are very similar (apart from the final reg_to_slot here) and should
|
|
|
|
# most likely be united
|
|
|
|
def reduce_and_load(const_reg , compiler , original_source )
|
|
|
|
raise "only cache" unless known_object.is_a?( Parfait::CacheEntry)
|
|
|
|
left = compiler.use_reg( :CacheEntry )
|
|
|
|
compiler.add_code Risc.load_constant(original_source, known_object , left)
|
2020-02-17 14:29:45 +07:00
|
|
|
compiler.add_code Risc.reg_to_slot(original_source, const_reg , left, slots.name)
|
2020-02-11 16:03:51 +07:00
|
|
|
end
|
2020-02-10 19:28:48 +07:00
|
|
|
end
|
|
|
|
end
|