2014-07-17 00:53:19 +03:00
|
|
|
module Virtual
|
2014-07-24 21:56:31 +03:00
|
|
|
# So when an object calls a method, or sends a message, this is what it sends: a Message
|
2014-07-17 00:53:19 +03:00
|
|
|
|
|
|
|
# A message contains the sender, return and exceptional return addresses,the arguments, and a slot for the frame.
|
|
|
|
|
|
|
|
# As such it is a very run-time object, deep in the machinery as it were, and does not have meaningful
|
|
|
|
# methods you could call at compile time.
|
|
|
|
|
|
|
|
# The methods that are there, are nevertheless meant to be called at compile time and generate code, rather than
|
|
|
|
# executing it.
|
|
|
|
|
|
|
|
# The caller creates the Message and passes control to the receiver's method
|
|
|
|
|
|
|
|
# The receiver create a new Frame to hold local and temporary variables and (later) creates default values for
|
|
|
|
# arguments that were not passed
|
|
|
|
|
|
|
|
# How the actual finding of the method takes place (acording to the ruby rules) is not simple, but as there is a
|
|
|
|
# guaranteed result (be it method_missing) it does not matter to the passing mechanism described
|
|
|
|
|
|
|
|
# During compilation Message and frame objects are created to do type analysis
|
|
|
|
|
|
|
|
class Message
|
2014-09-11 19:00:14 +03:00
|
|
|
|
2014-09-11 21:26:22 +03:00
|
|
|
SELF_REG = :r0
|
|
|
|
MESSAGE_REG = :r1
|
|
|
|
FRAME_REG = :r2
|
|
|
|
NEW_MESSAGE_REG = :r3
|
|
|
|
|
|
|
|
TMP_REG = :r4
|
|
|
|
|
2014-07-17 00:53:19 +03:00
|
|
|
def initialize me , normal , exceptional
|
|
|
|
@me = me
|
|
|
|
@next_normal = normal
|
|
|
|
@next_exception = exceptional
|
2014-07-24 21:56:31 +03:00
|
|
|
@arguments = arguments
|
2014-07-17 00:53:19 +03:00
|
|
|
# a frame represents the local and temporary variables at a point in the program.
|
2014-07-24 21:56:31 +03:00
|
|
|
@frame = nil
|
2014-07-17 00:53:19 +03:00
|
|
|
end
|
2014-07-24 21:56:31 +03:00
|
|
|
attr_reader :me, :next_normal, :next_exception, :arguments , :frame
|
2014-07-17 00:53:19 +03:00
|
|
|
|
|
|
|
# dummy for the eventual
|
|
|
|
def new_frame
|
2014-07-24 21:56:31 +03:00
|
|
|
raise self.inspect
|
2014-07-17 00:53:19 +03:00
|
|
|
end
|
|
|
|
#
|
|
|
|
def compile_get method , name
|
2014-09-24 18:25:18 +03:00
|
|
|
raise "CALLED"
|
2014-07-25 20:28:38 +03:00
|
|
|
if method.has_arg(name)
|
2014-08-13 11:59:51 +03:00
|
|
|
method.add_code MessageGet.new(name)
|
2014-07-25 20:28:38 +03:00
|
|
|
else
|
2014-08-13 11:59:51 +03:00
|
|
|
method.add_code FrameGet.new(name)
|
2014-07-25 20:28:38 +03:00
|
|
|
end
|
2014-07-17 00:53:19 +03:00
|
|
|
method.get_var(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def compile_set method , name , val
|
|
|
|
method.set_var(name,val)
|
2014-07-25 20:28:38 +03:00
|
|
|
if method.has_arg(name)
|
2014-08-13 11:59:51 +03:00
|
|
|
method.add_code MessageSet.new(name , val )
|
2014-07-25 20:28:38 +03:00
|
|
|
else
|
2014-08-13 11:59:51 +03:00
|
|
|
method.add_code FrameSet.new(name , val )
|
2014-07-25 20:28:38 +03:00
|
|
|
end
|
2014-07-17 00:53:19 +03:00
|
|
|
method.get_var(name)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|