2019-10-03 20:07:55 +02:00
|
|
|
module SlotMachine
|
2017-09-11 13:22:33 +02:00
|
|
|
|
2019-10-03 20:07:55 +02:00
|
|
|
# As reminder: a statically resolved call (the simplest one) becomes three SlotMachine Instructions.
|
2018-03-21 07:50:55 +01:00
|
|
|
# Ie: MessageSetup,ArgumentTransfer,SimpleCall
|
2017-09-11 13:22:33 +02:00
|
|
|
#
|
2018-03-21 07:50:55 +01:00
|
|
|
# MessageSetup does Setup before a call can be made, acquiring and filling the message
|
2018-11-14 11:41:13 +01:00
|
|
|
# basically. Only after MessageSetup is the next_message safe to use.
|
2017-09-11 13:22:33 +02:00
|
|
|
#
|
2018-11-14 11:41:13 +01:00
|
|
|
# The Factory (instane kept by Space) keeps a linked list of Messages,
|
|
|
|
# from which we take and currenty also return.
|
2018-03-21 07:50:55 +01:00
|
|
|
#
|
2018-04-05 11:24:49 +02:00
|
|
|
# Message setup set the name to the called method's name, and also set the arg and local
|
|
|
|
# types on the new message, currently for debugging but later for dynamic checking
|
|
|
|
#
|
|
|
|
# The only difference between the setup of a static call and a dynamic one is where
|
|
|
|
# the method comes from. A static, or simple call, passes the method, but a dynamic
|
|
|
|
# call passes the cache_entry that holds the resolved method.
|
|
|
|
#
|
|
|
|
# In either case, the method is loaded and name,frame and args set
|
2017-09-11 13:22:33 +02:00
|
|
|
#
|
|
|
|
class MessageSetup < Instruction
|
2018-04-05 11:24:49 +02:00
|
|
|
attr_reader :method_source
|
2017-09-11 13:22:33 +02:00
|
|
|
|
2018-04-05 11:24:49 +02:00
|
|
|
def initialize(method_source)
|
2019-09-07 16:56:06 +02:00
|
|
|
raise "no nil source" unless method_source
|
2018-04-05 11:24:49 +02:00
|
|
|
@method_source = method_source
|
2017-09-11 13:22:33 +02:00
|
|
|
end
|
2018-03-13 11:46:06 +01:00
|
|
|
|
2018-04-05 11:24:49 +02:00
|
|
|
# Move method name, frame and arguemnt types from the method to the next_message
|
|
|
|
# Get the message from Space and link it.
|
2018-03-13 11:46:06 +01:00
|
|
|
def to_risc(compiler)
|
2018-08-19 12:18:25 +02:00
|
|
|
build_with(compiler.builder(self))
|
2018-04-06 23:14:02 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
# directly called by to_risc
|
|
|
|
# but also used directly in __init
|
|
|
|
def build_with(builder)
|
2018-04-08 22:45:23 +02:00
|
|
|
case from = method_source
|
2018-07-07 08:11:09 +02:00
|
|
|
when Parfait::CallableMethod
|
2018-08-14 18:39:46 +02:00
|
|
|
builder.build { callable! << from }
|
2018-04-08 22:45:23 +02:00
|
|
|
when Parfait::CacheEntry
|
|
|
|
builder.build do
|
2018-08-14 18:39:46 +02:00
|
|
|
cache_entry! << from
|
|
|
|
callable! << cache_entry[:cached_method]
|
2018-04-08 22:45:23 +02:00
|
|
|
end
|
2018-07-24 10:35:49 +02:00
|
|
|
when Integer
|
|
|
|
builder.build do
|
2019-08-22 21:56:44 +02:00
|
|
|
callable! << message[ "arg#{from}".to_sym ]
|
2018-07-24 10:35:49 +02:00
|
|
|
end
|
2018-04-08 22:45:23 +02:00
|
|
|
else
|
2018-07-24 10:35:49 +02:00
|
|
|
raise "unknown source #{method_source.class}:#{method_source}"
|
2018-04-08 22:45:23 +02:00
|
|
|
end
|
2018-04-08 17:55:17 +02:00
|
|
|
build_message_data(builder)
|
|
|
|
return builder.built
|
2018-04-05 11:24:49 +02:00
|
|
|
end
|
2018-04-06 23:14:02 +02:00
|
|
|
|
2018-04-05 11:24:49 +02:00
|
|
|
private
|
2018-04-05 19:10:00 +02:00
|
|
|
def source
|
|
|
|
"method setup "
|
|
|
|
end
|
|
|
|
|
2018-08-12 13:47:05 +02:00
|
|
|
# set the method into the message
|
2018-04-06 21:40:58 +02:00
|
|
|
def build_message_data( builder )
|
2019-08-23 18:22:27 +02:00
|
|
|
if(reg = builder.names["next_message"])
|
|
|
|
raise "NEXT = #{reg}"
|
|
|
|
end
|
2018-04-08 17:55:17 +02:00
|
|
|
builder.build do
|
2019-08-23 18:22:27 +02:00
|
|
|
next_message! << message[:next_message]
|
|
|
|
next_message[:method] << callable
|
2018-04-06 13:21:38 +02:00
|
|
|
end
|
2018-04-06 15:08:35 +02:00
|
|
|
end
|
2017-09-11 13:22:33 +02:00
|
|
|
end
|
|
|
|
end
|