2018-05-28 14:46:26 +03:00
|
|
|
require_relative "fake_memory"
|
|
|
|
|
|
|
|
|
2019-09-09 20:26:54 +03:00
|
|
|
module Parfait
|
2019-09-22 19:10:47 +03:00
|
|
|
class Object
|
|
|
|
# redefine the runtime version
|
|
|
|
def self.new( *args )
|
|
|
|
object = self.allocate
|
|
|
|
Parfait.set_type_for(object)
|
|
|
|
object.send :initialize , *args
|
|
|
|
object
|
|
|
|
end
|
|
|
|
# Setter fo the boot process, only at runtime.
|
|
|
|
# only one space exists and it is generated at compile time, not runtime
|
|
|
|
def self.set_object_space( space )
|
|
|
|
@object_space = space
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.set_type_for(object)
|
|
|
|
return unless(Parfait.object_space)
|
2019-09-23 00:07:30 +03:00
|
|
|
return if(object.is_a?(Symbol))
|
|
|
|
return if(object.is_a?(::Integer))
|
2019-09-22 19:10:47 +03:00
|
|
|
name = object.class.name.split("::").last.to_sym
|
|
|
|
# have to grab the class, because we are in the ruby class not the parfait one
|
|
|
|
cl = Parfait.object_space.get_class_by_name( name )
|
|
|
|
# and have to set the type before we let the object do anything. otherwise boom
|
|
|
|
raise "No such class #{name} for #{object}" unless cl
|
|
|
|
object.set_type cl.instance_type
|
|
|
|
end
|
|
|
|
|
|
|
|
class Space < Object
|
|
|
|
|
|
|
|
# Space can only ever be creared at compile time, not runtime
|
|
|
|
def initialize( )
|
|
|
|
@classes = Dictionary.new
|
|
|
|
@types = Dictionary.new
|
|
|
|
@factories = Dictionary.new
|
|
|
|
@true_object = Parfait::TrueClass.new
|
|
|
|
@false_object = Parfait::FalseClass.new
|
|
|
|
@nil_object = Parfait::NilClass.new
|
|
|
|
end
|
|
|
|
def init_mem(pages)
|
|
|
|
[:Integer , :ReturnAddress , :Message].each do |fact_name|
|
2019-09-23 00:07:30 +03:00
|
|
|
for_type = @classes[fact_name].instance_type
|
2019-09-22 19:10:47 +03:00
|
|
|
page_size = pages[fact_name] || 1024
|
|
|
|
factory = Factory.new( for_type , page_size )
|
|
|
|
factory.get_more
|
2019-09-23 00:07:30 +03:00
|
|
|
@factories[ fact_name ] = factory
|
2019-09-22 19:10:47 +03:00
|
|
|
end
|
|
|
|
init_message_chain( factories[ :Message ].reserve )
|
|
|
|
init_message_chain( factories[ :Message ].next_object )
|
|
|
|
end
|
|
|
|
def init_message_chain( message )
|
|
|
|
prev = nil
|
|
|
|
while(message)
|
|
|
|
message.initialize
|
|
|
|
message._set_caller(prev) if prev
|
|
|
|
prev = message
|
|
|
|
message = message.next_message
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2019-09-09 20:26:54 +03:00
|
|
|
class DataObject < Object
|
2018-05-28 14:46:26 +03:00
|
|
|
def self.allocate
|
|
|
|
r = super
|
2018-08-11 19:13:09 +03:00
|
|
|
r.instance_variable_set(:@memory , Risc::FakeMemory.new(r ,self.type_length , self.memory_size))
|
2018-05-28 14:46:26 +03:00
|
|
|
r
|
|
|
|
end
|
|
|
|
# 0 -based index
|
|
|
|
def get_internal_word(index)
|
2019-09-10 14:49:02 +03:00
|
|
|
return super(index) if index < self.class.type_length
|
2019-09-09 20:26:54 +03:00
|
|
|
@memory[ index ]
|
2018-05-28 14:46:26 +03:00
|
|
|
end
|
|
|
|
|
2018-05-28 18:20:09 +03:00
|
|
|
# 0 -based index
|
2018-05-28 14:46:26 +03:00
|
|
|
def set_internal_word(index , value)
|
2019-09-09 20:26:54 +03:00
|
|
|
return super(index,value) if index < self.class.type_length
|
|
|
|
@memory[ index ] = value
|
2018-05-28 14:46:26 +03:00
|
|
|
end
|
2019-09-09 20:26:54 +03:00
|
|
|
end
|
2018-08-11 19:13:09 +03:00
|
|
|
|
2019-09-09 20:26:54 +03:00
|
|
|
class Object
|
|
|
|
|
|
|
|
# 0 -based index
|
|
|
|
def get_internal_word(index)
|
|
|
|
return @type if index == Parfait::TYPE_INDEX
|
2019-09-23 20:42:46 +03:00
|
|
|
name = self.type.names[index]
|
2019-09-09 20:26:54 +03:00
|
|
|
return nil unless name
|
|
|
|
instance_eval("@#{name}")
|
|
|
|
end
|
|
|
|
|
|
|
|
# 0 -based index
|
|
|
|
def set_internal_word(index , value)
|
2019-09-23 20:42:46 +03:00
|
|
|
name = self.type.names[index] #unless name.is_a?(Symbol)
|
2019-09-22 19:10:47 +03:00
|
|
|
raise "not sym for #{index} in #{self}:#{self.type}:#{name.class}" unless name.is_a?(Symbol)
|
2019-09-09 20:26:54 +03:00
|
|
|
instance_eval("@#{name}=value" )
|
|
|
|
value
|
2018-08-11 19:13:09 +03:00
|
|
|
end
|
|
|
|
|
2019-09-23 00:07:30 +03:00
|
|
|
end
|
2018-08-12 13:09:34 +03:00
|
|
|
|
2018-06-29 14:26:25 +03:00
|
|
|
# new list from ruby array to be precise
|
|
|
|
def self.new_list array
|
|
|
|
list = Parfait::List.new
|
|
|
|
list.set_length array.length
|
|
|
|
index = 0
|
|
|
|
while index < array.length do
|
|
|
|
list.set(index , array[index])
|
|
|
|
index = index + 1
|
|
|
|
end
|
|
|
|
list
|
|
|
|
end
|
|
|
|
|
2020-03-13 15:50:50 +02:00
|
|
|
def self.new_word_max( string )
|
|
|
|
return self.new_word(string) if string.length < 20
|
|
|
|
self.new_word(string[0 ... 20])
|
|
|
|
end
|
|
|
|
|
2018-06-29 14:26:25 +03:00
|
|
|
# Word from string
|
|
|
|
def self.new_word( string )
|
2020-03-13 15:50:50 +02:00
|
|
|
# puts "NEW #{string.length}=#{string}"
|
2018-06-29 14:26:25 +03:00
|
|
|
string = string.to_s if string.is_a? Symbol
|
|
|
|
word = Word.new( string.length )
|
|
|
|
string.codepoints.each_with_index do |code , index |
|
|
|
|
word.set_char(index , code)
|
|
|
|
end
|
|
|
|
word
|
|
|
|
end
|
|
|
|
|
2018-05-28 14:46:26 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
class Symbol
|
|
|
|
|
|
|
|
def has_type?
|
|
|
|
true
|
|
|
|
end
|
|
|
|
def get_type
|
|
|
|
l = Parfait.object_space.classes[:Word].instance_type
|
|
|
|
#puts "LL #{l.class}"
|
|
|
|
l
|
|
|
|
end
|
2020-03-25 12:43:57 +02:00
|
|
|
alias :ct_type :get_type
|
|
|
|
def value
|
|
|
|
self
|
|
|
|
end
|
2018-05-28 14:46:26 +03:00
|
|
|
def padded_length
|
2019-08-17 21:07:07 +03:00
|
|
|
Parfait::Object.padded( to_s.length + 4)
|
2018-05-28 14:46:26 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|