split compiled_method into method and compiled_method_info

This commit is contained in:
Torsten Ruger 2015-05-20 16:43:26 +03:00
parent dd2a5e367f
commit d6d0f4f43a
8 changed files with 147 additions and 103 deletions

70
lib/parfait/method.rb Normal file
View File

@ -0,0 +1,70 @@
# 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
def initialize name , arg_names
@name = name
@arg_names = arg_names
@locals = []
@tmps = []
end
attr_reader :name , :arg_names
attr_accessor :for_class
# 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
name = name.to_sym
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
@arg_names.index_of name.to_sym
end
# determine if method has a local variable or tmp (anonymous local) by given name
def has_local name
name = name.to_sym
index = @locals.index(name)
index = @tmps.index(name) unless index
index
end
def ensure_local name
index = has_local name
return index if index
@locals << name
@locals.length
end
def get_var name
var = has_var name
raise "no var #{name} in method #{self.name} , #{@locals} #{@arg_names}" unless var
var
end
end
end

View File

@ -15,15 +15,21 @@ module Parfait
class Module < Object class Module < Object
def initialize name , superclass def initialize name , superclass
@instance_methods = [] @instance_methods = []
@name = name.to_sym @name = name
@super_class = superclass @super_class = superclass
@meta_class = Virtual::MetaClass.new(self) @meta_class = Virtual::MetaClass.new(self)
end end
def add_instance_method method def add_instance_method method
raise "not a method #{method.class} #{method.inspect}" unless method.is_a? Virtual::CompiledMethod raise "not a method #{method.class} #{method.inspect}" unless method.is_a? Method
raise "syserr " unless method.name.is_a? Symbol raise "syserr " unless method.name.is_a? Symbol
method.for_class = self
@instance_methods << method @instance_methods << method
method
end
def create_instance_method name , arg_names
add_instance_method Method.new( name , arg_names )
end end
# this needs to be done during booting as we can't have all the classes and superclassses # this needs to be done during booting as we can't have all the classes and superclassses

View File

@ -2,7 +2,7 @@ require "virtual/machine"
require "virtual/compiler" require "virtual/compiler"
require "virtual/instruction" require "virtual/instruction"
require "virtual/compiled_method" require "virtual/compiled_method_info"
require "virtual/slots/slot" require "virtual/slots/slot"
require "virtual/type" require "virtual/type"
require "virtual/constants" require "virtual/constants"
@ -13,4 +13,4 @@ require "virtual/passes/enter_implementation"
require "virtual/passes/frame_implementation" require "virtual/passes/frame_implementation"
Sof::Volotile.add(Virtual::Block , [:method]) Sof::Volotile.add(Virtual::Block , [:method])
Sof::Volotile.add(Virtual::CompiledMethod , [:current]) Sof::Volotile.add(Virtual::CompiledMethodInfo , [:current])

View File

@ -49,30 +49,10 @@ module Parfait
end end
end end
class Parfait::List class List
def to_sof_node(writer , level , ref ) def to_sof_node(writer , level , ref )
Sof.array_to_sof_node(self , writer , level , ref ) Sof.array_to_sof_node(self , writer , level , ref )
end end
end
class Dictionary
def to_sof_node(writer , level , ref)
Sof.hash_to_sof_node( self , writer , level , ref)
end
end
Word.class_eval do
def to_s
string = ""
index = 1
while( index <= self.length)
string[index - 1] = get_char(index).chr
index = index + 1
end
string
end
end
List.class_eval do
def to_a def to_a
array = [] array = []
index = 1 index = 1
@ -83,6 +63,26 @@ module Parfait
array array
end end
end end
class Dictionary
def to_sof_node(writer , level , ref)
Sof.hash_to_sof_node( self , writer , level , ref)
end
end
class Method
attr_accessor :info
end
class Word
def to_s
string = ""
index = 1
while( index <= self.length)
string[index - 1] = get_char(index).chr
index = index + 1
end
string
end
end
end end
module Virtual module Virtual

View File

@ -1,14 +1,13 @@
require_relative "block" require_relative "block"
module Virtual module Virtual
# static description of a method # the static info of a method (with its compiled code, argument names etc ) is part of the
# name # runtime, ie found in Parfait::Method
# arg_names (with defaults)
# the info we create here is injected int the method and used only at compile-time
# receiver # receiver
# code
# return arg (usually mystery, but for coded ones can be more specific) # return arg (usually mystery, but for coded ones can be more specific)
# known local variable names
# temp variables (numbered)
# #
# Methods are one step up from to VM::Blocks. Where Blocks can be jumped to, Methods can be called. # Methods are one step up from to VM::Blocks. Where Blocks can be jumped to, Methods can be called.
@ -28,27 +27,36 @@ module Virtual
# These (eg if/while) blocks may themselves have linear blocks ,but the last of these # These (eg if/while) blocks may themselves have linear blocks ,but the last of these
# MUST have an uncoditional branch. And remember, all roads lead to return. # MUST have an uncoditional branch. And remember, all roads lead to return.
class CompiledMethod < Virtual::Object class CompiledMethodInfo
#return the main function (the top level) into which code is compiled # return the main function (the top level) into which code is compiled
def CompiledMethod.main # this just create a "main" with create_method , see there
CompiledMethod.new(:main , [] ) def self.main
self.create_method( "Object" , :main , [] )
end end
def initialize name , arg_names , receiver = Virtual::Self.new , return_type = Virtual::Mystery
@name = name.to_sym # create method does two things
@class_name = "Object" # first it creates the parfait method, for the given class, with given argument names
@arg_names = arg_names # second, it creates CompiledMethodInfo and attaches it to the method
@locals = [] #
@tmps = [] # compile code then works with the method, but adds code tot the info
@receiver = receiver def self.create_method( class_name , method_name , args)
@return_type = return_type class_name = Virtual.new_word(class_name) if class_name.is_a? String
method_name = Virtual.new_word(method_name) if method_name.is_a? String
clazz = Machine.instance.space.get_class_by_name class_name
raise "No such class #{class_name}" unless clazz
method = clazz.create_instance_method(method_name , Virtual.new_list(args))
method.info = CompiledMethodInfo.new
method
end
def initialize receiver = Virtual::Self.new , return_type = Virtual::Mystery
# first block we have to create with .new , as new_block assumes a current # first block we have to create with .new , as new_block assumes a current
enter = Block.new( "enter" , self ).add_code(MethodEnter.new()) enter = Block.new( "enter" , self ).add_code(MethodEnter.new())
@blocks = [enter] @blocks = [enter]
@current = enter @current = enter
new_block("return").add_code(MethodReturn.new) new_block("return").add_code(MethodReturn.new)
end end
attr_reader :name , :arg_names , :receiver , :blocks attr_reader :receiver , :blocks
attr_accessor :return_type , :current , :class_name attr_accessor :return_type , :current
# add an instruction after the current (insertion point) # add an instruction after the current (insertion point)
# the added instruction will become the new insertion point # the added instruction will become the new insertion point
@ -105,52 +113,12 @@ module Virtual
return new_b return new_b
end end
# 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
name = name.to_sym
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
@arg_names.index name.to_sym
end
# determine if method has a local variable or tmp (anonymous local) by given name
def has_local name
name = name.to_sym
index = @locals.index(name)
index = @tmps.index(name) unless index
index
end
def ensure_local name
index = has_local name
return index if index
@locals << name
@locals.length
end
def get_var name
var = has_var name
raise "no var #{name} in method #{self.name} , #{@locals} #{@arg_names}" unless var
var
end
def get_tmp def get_tmp
name = "__tmp__#{@tmps.length}" name = "__tmp__#{@tmps.length}"
@tmps << name @tmps << name
Ast::NameExpression.new(name) Ast::NameExpression.new(name)
end end
def old_layout
Virtual::Object.layout
end
# sugar to create instructions easily. # sugar to create instructions easily.
# any method will be passed on to the RegisterMachine and the result added to the insertion block # any method will be passed on to the RegisterMachine and the result added to the insertion block
# With this trick we can write what looks like assembler, # With this trick we can write what looks like assembler,

View File

@ -15,28 +15,28 @@ module Virtual
def self.compile_integer expression , method def self.compile_integer expression , method
int = IntegerConstant.new(expression.value) int = IntegerConstant.new(expression.value)
to = Return.new(Integer , int) to = Return.new(Integer , int)
method.add_code Set.new( to , int) method.info.add_code Set.new( to , int)
to to
end end
def self.compile_true expression , method def self.compile_true expression , method
value = TrueConstant.new value = TrueConstant.new
to = Return.new(Reference , value) to = Return.new(Reference , value)
method.add_code Set.new( to , value ) method.info.add_code Set.new( to , value )
to to
end end
def self.compile_false expression , method def self.compile_false expression , method
value = FalseConstant.new value = FalseConstant.new
to = Return.new(Reference , value) to = Return.new(Reference , value)
method.add_code Set.new( to , value ) method.info.add_code Set.new( to , value )
to to
end end
def self.compile_nil expression , method def self.compile_nil expression , method
value = NilConstant.new value = NilConstant.new
to = Return.new(Reference , value) to = Return.new(Reference , value)
method.add_code Set.new( to , value ) method.info.add_code Set.new( to , value )
to to
end end
@ -50,9 +50,9 @@ module Virtual
if method.has_var(expression.name) if method.has_var(expression.name)
# either an argument, so it's stored in message # either an argument, so it's stored in message
if( index = method.has_arg(name)) if( index = method.has_arg(name))
method.add_code MessageGet.new(name , index) method.info.add_code MessageGet.new(name , index)
else # or a local so it is in the frame else # or a local so it is in the frame
method.add_code FrameGet.new(name , index) method.info.add_code FrameGet.new(name , index)
end end
else else
call = Ast::CallSiteExpression.new(expression.name , [] ) #receiver self is implicit call = Ast::CallSiteExpression.new(expression.name , [] ) #receiver self is implicit
@ -65,7 +65,7 @@ module Virtual
clazz = Space.space.get_class_by_name name clazz = Space.space.get_class_by_name name
raise "uups #{clazz}.#{name}" unless clazz raise "uups #{clazz}.#{name}" unless clazz
to = Return.new(Reference , clazz ) to = Return.new(Reference , clazz )
method.add_code Set.new( to , clazz ) method.info.add_code Set.new( to , clazz )
to to
end end
@ -74,7 +74,7 @@ module Virtual
value = Virtual.new_word(expression.string) value = Virtual.new_word(expression.string)
to = Return.new(Reference , value) to = Return.new(Reference , value)
Machine.instance.space.add_object value Machine.instance.space.add_object value
method.add_code Set.new( to , value ) method.info.add_code Set.new( to , value )
to to
end end
@ -85,16 +85,16 @@ module Virtual
raise "oh noo, nil from where #{expression.right.inspect}" unless r raise "oh noo, nil from where #{expression.right.inspect}" unless r
index = method.has_arg(name) index = method.has_arg(name)
if index if index
method.add_code Set.new(Return.new , MessageSlot.new(index , r,type , r )) method.info.add_code Set.new(Return.new , MessageSlot.new(index , r,type , r ))
else else
index = method.ensure_local(expression.left.name) index = method.ensure_local(expression.left.name)
method.add_code Set.new(Return.new , FrameSlot.new(index , r.type , r )) method.info.add_code Set.new(Return.new , FrameSlot.new(index , r.type , r ))
end end
r r
end end
def self.compile_variable expression, method def self.compile_variable expression, method
method.add_code InstanceGet.new(expression.name) method.info.add_code InstanceGet.new(expression.name)
Return.new( Mystery ) Return.new( Mystery )
end end
end end

View File

@ -6,9 +6,9 @@ module Virtual
def self.compile_callsite expession , method def self.compile_callsite expession , method
me = Compiler.compile( expession.receiver , method ) me = Compiler.compile( expession.receiver , method )
method.add_code NewMessage.new method.info.add_code NewMessage.new
method.add_code Set.new(NewSelf.new(me.type), me) method.info.add_code Set.new(NewSelf.new(me.type), me)
method.add_code Set.new(NewName.new(), Virtual.new_word(expession.name)) method.info.add_code Set.new(NewName.new(), Virtual.new_word(expession.name))
compiled_args = [] compiled_args = []
expession.args.each_with_index do |arg , i| expession.args.each_with_index do |arg , i|
#compile in the running method, ie before passing control #compile in the running method, ie before passing control
@ -16,13 +16,13 @@ module Virtual
# move the compiled value to it's slot in the new message # move the compiled value to it's slot in the new message
to = NewMessageSlot.new(i ,val.type , val) to = NewMessageSlot.new(i ,val.type , val)
# (doing this immediately, not after the loop, so if it's a return it won't get overwritten) # (doing this immediately, not after the loop, so if it's a return it won't get overwritten)
method.add_code Set.new(to , val ) method.info.add_code Set.new(to , val )
compiled_args << to compiled_args << to
end end
method.add_code MessageSend.new(expession.name , me , compiled_args) #and pass control method.info.add_code MessageSend.new(expession.name , me , compiled_args) #and pass control
# the effect of the method is that the NewMessage Return slot will be filled, return it # the effect of the method is that the NewMessage Return slot will be filled, return it
# (this is what is moved _inside_ above loop for such expressions that are calls (or constants)) # (this is what is moved _inside_ above loop for such expressions that are calls (or constants))
Return.new( method.return_type ) Return.new( method.info.return_type )
end end
end end
end end

View File

@ -7,7 +7,7 @@ module Virtual
p.name p.name
end end
r = expression.receiver ? Compiler.compile(expression.receiver, method ) : Self.new() r = expression.receiver ? Compiler.compile(expression.receiver, method ) : Self.new()
new_method = CompiledMethod.new(expression.name , args , r ) new_method = CompiledMethodInfo.create_method(expression.name , args , r )
new_method.class_name = r.is_a?(Parfait::Class) ? r.name : method.class_name new_method.class_name = r.is_a?(Parfait::Class) ? r.name : method.class_name
clazz = Machine.instance.space.get_class_by_name(new_method.class_name) clazz = Machine.instance.space.get_class_by_name(new_method.class_name)
clazz.add_instance_method new_method clazz.add_instance_method new_method