2017-09-11 14:22:33 +03:00
|
|
|
module Mom
|
|
|
|
|
2018-03-21 12:20:55 +05:30
|
|
|
# As reminder: a statically resolved call (the simplest one) becomes three Mom Instructions.
|
|
|
|
# Ie: MessageSetup,ArgumentTransfer,SimpleCall
|
2017-09-11 14:22:33 +03:00
|
|
|
#
|
2018-03-21 12:20:55 +05:30
|
|
|
# MessageSetup does Setup before a call can be made, acquiring and filling the message
|
2018-04-05 12:24:49 +03:00
|
|
|
# basically.Only after MessageSetup is the next_message safe to use.
|
2017-09-11 14:22:33 +03:00
|
|
|
#
|
2018-04-05 12:24:49 +03:00
|
|
|
# The space keeps a linked list of Messages, from which we take and currenty also return.
|
2018-03-21 12:20:55 +05:30
|
|
|
#
|
2018-04-05 12:24:49 +03: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 14:22:33 +03:00
|
|
|
#
|
|
|
|
class MessageSetup < Instruction
|
2018-04-05 12:24:49 +03:00
|
|
|
attr_reader :method_source
|
2017-09-11 14:22:33 +03:00
|
|
|
|
2018-04-05 12:24:49 +03:00
|
|
|
def initialize(method_source)
|
|
|
|
@method_source = method_source
|
2017-09-11 14:22:33 +03:00
|
|
|
end
|
2018-03-13 16:16:06 +05:30
|
|
|
|
2018-04-05 12:24:49 +03: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 16:16:06 +05:30
|
|
|
def to_risc(compiler)
|
2018-04-18 19:12:30 +03:00
|
|
|
builder = compiler.builder(false, self)
|
2018-04-08 23:45:23 +03:00
|
|
|
build_with(builder)
|
2018-04-07 00:14:02 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
# directly called by to_risc
|
|
|
|
# but also used directly in __init
|
|
|
|
def build_with(builder)
|
2018-04-08 23:45:23 +03:00
|
|
|
case from = method_source
|
|
|
|
when Parfait::TypedMethod
|
|
|
|
builder.build { typed_method << from }
|
|
|
|
when Parfait::CacheEntry
|
|
|
|
builder.build do
|
|
|
|
cache_entry << from
|
|
|
|
typed_method << cache_entry[:cached_method]
|
|
|
|
end
|
|
|
|
else
|
|
|
|
raise "unknown source #{method_source}"
|
|
|
|
end
|
2018-04-08 18:55:17 +03:00
|
|
|
build_message_data(builder)
|
|
|
|
return builder.built
|
2018-04-05 12:24:49 +03:00
|
|
|
end
|
2018-04-07 00:14:02 +03:00
|
|
|
|
2018-04-05 12:24:49 +03:00
|
|
|
private
|
2018-04-05 20:10:00 +03:00
|
|
|
def source
|
|
|
|
"method setup "
|
|
|
|
end
|
|
|
|
|
2018-04-05 12:24:49 +03:00
|
|
|
# get the next message from space and unlink it there
|
2018-04-06 22:54:54 +03:00
|
|
|
# also put it into next_message of current message (and reverse)
|
|
|
|
# set name and type data in the message, from the method loaded
|
2018-04-06 22:40:58 +03:00
|
|
|
def build_message_data( builder )
|
2018-04-08 18:55:17 +03:00
|
|
|
builder.build do
|
2018-04-06 20:58:58 +03:00
|
|
|
space << Parfait.object_space
|
2018-04-06 21:05:26 +03:00
|
|
|
next_message << space[:first_message]
|
2018-04-06 22:40:58 +03:00
|
|
|
message[:next_message] << next_message
|
|
|
|
next_message[:caller] << message
|
|
|
|
|
|
|
|
type << typed_method[:arguments_type]
|
2018-04-06 23:11:08 +03:00
|
|
|
named_list << next_message[:arguments]
|
2018-04-06 22:40:58 +03:00
|
|
|
named_list[:type] << type
|
|
|
|
|
2018-04-06 23:11:08 +03:00
|
|
|
type << typed_method[:frame_type]
|
2018-04-06 22:40:58 +03:00
|
|
|
named_list << next_message[:frame]
|
|
|
|
named_list[:type] << type
|
|
|
|
|
2018-04-06 23:11:08 +03:00
|
|
|
name << typed_method[:name]
|
|
|
|
next_message[:name] << name
|
2018-04-06 22:40:58 +03:00
|
|
|
|
|
|
|
#store next.next back into space
|
2018-04-06 23:11:08 +03:00
|
|
|
#FIXME in a multithreaded future this should be done soon after getting
|
|
|
|
# the first_message, using lock free compare and swap. Now saving regs
|
2018-04-06 22:40:58 +03:00
|
|
|
next_message << next_message[:next_message]
|
|
|
|
space[:first_message] << next_message
|
2018-04-06 14:21:38 +03:00
|
|
|
end
|
2018-04-06 16:08:35 +03:00
|
|
|
end
|
2017-09-11 14:22:33 +03:00
|
|
|
end
|
|
|
|
end
|