Fix meta_class, sis class instance variables and class methods

after some serious recursive thinking it now actually makes sense.
The key was to change the actual type of the class that the meta_class manages
For objects it's (still) ok just to change the instance_type, but since the class object exists and has type, when adding instance variables, that actual type has to change
This commit is contained in:
Torsten Rüger 2019-09-23 20:42:46 +03:00
parent 7b40bb9106
commit 66728f09f4
8 changed files with 60 additions and 34 deletions

View File

@ -1,8 +1,17 @@
# Behaviour is something that has methods, basically class and modules superclass
# instance_methods is the attribute in the including class that has the methods
module Parfait module Parfait
# Behaviour is the old smalltalk name for the duperclass of class and meta_class
#
# Classes and meta_classes are in fact very similar, in that they manage
# - the type of instances
# - the methods for instances
#
# The main way they differ is that Classes manage type for a class of objects (ie many)
# whereas meta_class, or singleton_class manages the type of only one object (here a class)
#
# Singleton classes can manage the type/methods of any single object, and in the
# future off course they will, just not yet. Most single objects don't need that,
# only Classes and Modules _always _ do, so that's where we start.
#
class Behaviour < Object class Behaviour < Object
attr_reader :instance_type , :instance_methods attr_reader :instance_type , :instance_methods

View File

@ -43,6 +43,12 @@ module Parfait
inspect inspect
end end
# adding an instance changes the instance_type to include that variable
def add_instance_variable( name , type)
super(name,type)
@clazz.set_type(@instance_type)
end
# Nil name means no superclass, and so nil returned # Nil name means no superclass, and so nil returned
def super_class def super_class
return nil return nil

View File

@ -85,14 +85,14 @@ module Parfait
# 0 -based index # 0 -based index
def get_internal_word(index) def get_internal_word(index)
return @type if index == Parfait::TYPE_INDEX return @type if index == Parfait::TYPE_INDEX
name = Parfait.name_for_index(self , index) name = self.type.names[index]
return nil unless name return nil unless name
instance_eval("@#{name}") instance_eval("@#{name}")
end end
# 0 -based index # 0 -based index
def set_internal_word(index , value) def set_internal_word(index , value)
name = Parfait.name_for_index(self , index) name = self.type.names[index] #unless name.is_a?(Symbol)
raise "not sym for #{index} in #{self}:#{self.type}:#{name.class}" unless name.is_a?(Symbol) raise "not sym for #{index} in #{self}:#{self.type}:#{name.class}" unless name.is_a?(Symbol)
instance_eval("@#{name}=value" ) instance_eval("@#{name}=value" )
value value
@ -100,16 +100,6 @@ module Parfait
end end
def self.name_for_index(object , index)
return :type if index == 0
clazz = object.class.name.split("::").last.to_sym
cl = self.type_names[clazz]
keys = cl.keys
keys[index - 1] # -1 because type is excluded in the lists (FIX)
# FIXME Now that we use instance variables in parfait, they should be parsed
# and the type_names generated automatically
end
# new list from ruby array to be precise # new list from ruby array to be precise
def self.new_list array def self.new_list array
list = Parfait::List.new list = Parfait::List.new

View File

@ -36,15 +36,18 @@ module Parfait
assert_raises{ @try.add_instance_method(nil)} assert_raises{ @try.add_instance_method(nil)}
end end
def test_add_instance_variable_changes_type def test_add_instance_variable_changes_type
before = @space.get_class.instance_type before = @try.instance_type
@space.get_class.add_instance_variable(:counter , :Integer) @try.add_instance_variable(:counter , :Integer)
assert before != @space.get_class.instance_type assert before != @try.instance_type
end end
def test_add_instance_variable_changes_type_hash def test_add_instance_variable_changes_type_hash
before = @space.get_class.instance_type.hash before = @try.instance_type.hash
@space.get_class.add_instance_variable(:counter , :Integer) @try.add_instance_variable(:counter , :Integer)
assert before != @space.get_class.instance_type.hash assert before != @try.instance_type.hash
end
def test_add_instance_variable_changes_class_type
@try.add_instance_variable(:counter , :Integer)
assert_equal @try.clazz.type , @try.instance_type
end end
end end
end end

View File

@ -1,7 +1,7 @@
require_relative "../helper" require_relative "../helper"
module Risc module Risc
class InterpreterSetters < MiniTest::Test class InterpreterGetters < MiniTest::Test
include Ticker include Ticker
def setup def setup
@ -19,7 +19,7 @@ module Risc
MAIN MAIN
super super
end end
#Space type is wrong, shold be same as meta_class.instance_type
def test_chain def test_chain
#show_main_ticks # get output of what is #show_main_ticks # get output of what is
run_input @string_input run_input @string_input

View File

@ -23,7 +23,7 @@ MAIN
super super
end end
def est_chain def test_chain
#show_main_ticks # get output of what is #show_main_ticks # get output of what is
run_input @string_input run_input @string_input
assert_equal 5 , get_return assert_equal 5 , get_return

View File

@ -12,7 +12,7 @@ module Vool
return @inst return @inst
end end
def main(arg) def main(arg)
return Space.one_plus return Space.some_inst
end end
end end
eos eos
@ -20,16 +20,34 @@ module Vool
def setup def setup
ret = RubyX::RubyXCompiler.new(RubyX.default_test_options).ruby_to_mom(class_main) ret = RubyX::RubyXCompiler.new(RubyX.default_test_options).ruby_to_mom(class_main)
@ins = ret.compilers.find{|c|c.callable.name==:some_inst}.mom_instructions.next @compiler = ret.compilers.find{|c|c.callable.name==:some_inst}
@main = ret.compilers.find{|c|c.callable.name==:main}
@ins = @compiler.mom_instructions.next
end end
def test_class_inst def test_class_inst
space_class = Parfait.object_space.get_class space_class = Parfait.object_space.get_class
assert_equal :Space , space_class.name assert_equal :Space , space_class.name
names = space_class.meta_class.instance_type.names names = space_class.meta_class.instance_type.names
assert names.index_of(:inst) , names assert names.index_of(:inst) , names
end
def test_compiler
assert_equal Mom::MethodCompiler, @compiler.class
assert_equal Parfait::Type, @compiler.callable.self_type.class
assert_equal 6, @compiler.callable.self_type.names.index_of(:inst) , @compiler.callable.self_type.names
end end
def test_array def test_array
check_array [SlotLoad, ReturnJump, Label, ReturnSequence, Label] , @ins check_array [SlotLoad, ReturnJump, Label, ReturnSequence, Label] , @ins
end
def test_main_array
check_array [MessageSetup, ArgumentTransfer, SimpleCall, SlotLoad, ReturnJump ,
Label, ReturnSequence, Label] , @main.mom_instructions.next
end
def test_main_args
args = @main.mom_instructions.next(2)
assert_equal Parfait::Class , args.receiver.known_object.class
assert_equal :Space , args.receiver.known_object.name
assert_equal :some_inst , args.receiver.known_object.type.method_names.first
assert_equal :inst , args.receiver.known_object.type.names.last
end end
def test_load_inst def test_load_inst
assert_equal SlotLoad, @ins.class assert_equal SlotLoad, @ins.class

View File

@ -40,7 +40,7 @@ module Vool
assert_equal SimpleCall, @ins.next(2).class assert_equal SimpleCall, @ins.next(2).class
assert_equal :one_plus, @ins.next(2).method.name assert_equal :one_plus, @ins.next(2).method.name
assert_equal Parfait::Type, @ins.next(2).method.self_type.class assert_equal Parfait::Type, @ins.next(2).method.self_type.class
assert_equal :Object, @ins.next(2).method.self_type.object_class.name assert_equal :Class, @ins.next(2).method.self_type.object_class.name
end end
end end
end end