2015-05-20 15:43:26 +02:00
|
|
|
# A Method (at runtime , sis in Parfait) is static object that primarily holds the executable
|
|
|
|
# code.
|
|
|
|
|
|
|
|
# For reflection also holds arguments and such
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
module Parfait
|
|
|
|
|
|
|
|
# static description of a method
|
|
|
|
# name
|
|
|
|
# arg_names
|
|
|
|
# known local variable names
|
|
|
|
# temp variables (numbered)
|
|
|
|
# executable code
|
|
|
|
|
|
|
|
# ps, the compiler injects its own info, see virtual::compiled_method_info
|
|
|
|
|
|
|
|
|
|
|
|
class Method < Object
|
|
|
|
|
2015-05-24 12:53:49 +02:00
|
|
|
def initialize clazz , name , arg_names
|
2015-05-21 20:50:17 +02:00
|
|
|
super()
|
2015-05-24 12:53:49 +02:00
|
|
|
raise "No class #{name}" unless clazz
|
2015-07-21 14:40:25 +02:00
|
|
|
self.for_class = clazz
|
|
|
|
self.name = name
|
|
|
|
self.code = BinaryCode.new name
|
2015-08-06 17:27:25 +02:00
|
|
|
raise "Wrong type, expect List not #{arg_names.class}" unless arg_names.is_a? List
|
2015-07-21 14:40:25 +02:00
|
|
|
self.arg_names = arg_names
|
|
|
|
self.locals = List.new
|
|
|
|
self.tmps = List.new
|
2015-05-20 15:43:26 +02:00
|
|
|
end
|
2015-07-21 14:40:25 +02:00
|
|
|
attributes [:name , :arg_names , :for_class , :code , :locals , :tmps]
|
2015-05-20 15:43:26 +02:00
|
|
|
|
|
|
|
|
|
|
|
# determine whether this method has a variable by the given name
|
|
|
|
# variables are locals and and arguments
|
|
|
|
# used to determine if a send must be issued
|
|
|
|
# return index of the name into the message if so
|
|
|
|
def has_var name
|
2015-06-01 16:31:35 +02:00
|
|
|
raise "uups #{name}.#{name.class}" unless name.is_a? Symbol
|
2015-05-20 15:43:26 +02:00
|
|
|
index = has_arg(name)
|
|
|
|
return index if index
|
|
|
|
has_local(name)
|
|
|
|
end
|
|
|
|
|
|
|
|
# determine whether this method has an argument by the name
|
|
|
|
def has_arg name
|
2015-06-01 16:31:35 +02:00
|
|
|
raise "uups #{name}.#{name.class}" unless name.is_a? Symbol
|
2015-07-21 14:40:25 +02:00
|
|
|
self.arg_names.index_of name
|
2015-05-20 15:43:26 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
# determine if method has a local variable or tmp (anonymous local) by given name
|
|
|
|
def has_local name
|
2015-06-01 16:31:35 +02:00
|
|
|
raise "uups #{name}.#{name.class}" unless name.is_a? Symbol
|
2015-07-21 14:40:25 +02:00
|
|
|
index = self.locals.index_of(name)
|
|
|
|
index = self.tmps.index_of(name) unless index
|
2015-05-20 15:43:26 +02:00
|
|
|
index
|
|
|
|
end
|
|
|
|
|
|
|
|
def ensure_local name
|
|
|
|
index = has_local name
|
|
|
|
return index if index
|
2015-07-21 14:40:25 +02:00
|
|
|
self.locals.push name
|
|
|
|
self.locals.get_length
|
2015-05-20 15:43:26 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def get_var name
|
|
|
|
var = has_var name
|
2015-07-21 14:40:25 +02:00
|
|
|
raise "no var #{name} in method #{self.name} , #{self.locals} #{self.arg_names}" unless var
|
2015-05-20 15:43:26 +02:00
|
|
|
var
|
|
|
|
end
|
|
|
|
|
2015-06-19 18:50:53 +02:00
|
|
|
def sof_reference_name
|
2015-07-21 14:40:25 +02:00
|
|
|
self.name
|
2015-06-19 18:50:53 +02:00
|
|
|
end
|
|
|
|
|
2015-05-20 15:43:26 +02:00
|
|
|
end
|
|
|
|
end
|