2016-02-25 12:03:11 -08:00
|
|
|
# From a programmers perspective an object has hash like data (with instance variables as keys)
|
|
|
|
# and functions to work on that data.
|
|
|
|
# Only the object may access it's data directly.
|
2014-08-05 15:55:24 +03:00
|
|
|
|
2016-12-06 11:38:09 +02:00
|
|
|
# From an implementation perspective it is a chunk of memory with a type as the first
|
|
|
|
# word (instance of class Type).
|
2014-08-05 15:55:24 +03:00
|
|
|
|
2016-02-25 12:03:11 -08:00
|
|
|
# Objects are arranged or layed out (in memory) according to their Type
|
2018-08-11 19:15:34 +03:00
|
|
|
# every object has a Type. Type objects are immutable and may be reused for a group/class
|
2018-05-14 11:55:01 +03:00
|
|
|
# of objects.
|
2016-02-25 12:03:11 -08:00
|
|
|
# The Type of an object may change, but then a new Type is created
|
|
|
|
# The Type also defines the class of the object
|
2018-05-14 11:55:01 +03:00
|
|
|
# The Type is **always** the first entry (index 0) in an object
|
2015-05-17 14:40:02 +03:00
|
|
|
|
2019-09-09 11:54:45 +03:00
|
|
|
module Parfait
|
2016-12-16 01:14:09 +02:00
|
|
|
class Object
|
2019-09-09 20:26:54 +03:00
|
|
|
attr_reader :type
|
2019-09-18 22:36:56 +03:00
|
|
|
|
2019-09-10 00:18:20 +03:00
|
|
|
def self.type_length
|
|
|
|
1
|
|
|
|
end
|
|
|
|
def self.memory_size
|
|
|
|
4
|
|
|
|
end
|
2019-09-18 22:36:56 +03:00
|
|
|
# Make the object space globally available
|
|
|
|
def self.object_space
|
|
|
|
@object_space
|
|
|
|
end
|
2019-09-10 00:18:20 +03:00
|
|
|
|
2019-09-18 22:36:56 +03:00
|
|
|
def self.new
|
|
|
|
factory = @object_space.get_factory(:Object)
|
|
|
|
object = factory.get_next
|
|
|
|
object.initialize
|
|
|
|
end
|
2019-09-10 00:18:20 +03:00
|
|
|
|
2019-09-09 20:26:54 +03:00
|
|
|
def type=(t)
|
|
|
|
set_type( t )
|
|
|
|
end
|
2015-05-17 14:40:02 +03:00
|
|
|
|
2015-06-03 10:01:59 +03:00
|
|
|
def == other
|
|
|
|
self.object_id == other.object_id
|
|
|
|
end
|
|
|
|
|
2018-08-11 19:15:34 +03:00
|
|
|
# This is the core of the object system.
|
|
|
|
# The class of an object is stored in the objects memory
|
2015-05-11 18:55:49 +03:00
|
|
|
#
|
2017-01-02 00:29:20 +02:00
|
|
|
# In RubyX we store the class in the Type, and so the Type is the only fixed
|
2015-05-11 18:55:49 +03:00
|
|
|
# data that every object carries.
|
|
|
|
def get_class()
|
2016-02-25 11:50:10 -08:00
|
|
|
l = get_type()
|
|
|
|
#puts "Type #{l.class} in #{self.class} , #{self}"
|
2015-07-21 15:40:25 +03:00
|
|
|
l.object_class()
|
2015-05-11 18:55:49 +03:00
|
|
|
end
|
2015-04-08 20:24:50 +03:00
|
|
|
|
2015-05-22 22:51:36 +03:00
|
|
|
# private
|
2018-08-11 19:15:34 +03:00
|
|
|
def set_type(typ)
|
2019-02-10 21:00:25 +02:00
|
|
|
raise "not type" + typ.class.to_s + "in " + object_id.to_s(16) unless typ.is_a?(Type)
|
2019-09-09 20:26:54 +03:00
|
|
|
@type = typ
|
2015-05-22 22:51:36 +03:00
|
|
|
end
|
|
|
|
|
2016-02-25 11:50:10 -08:00
|
|
|
# so we can keep the raise in get_type
|
|
|
|
def has_type?
|
2019-09-09 20:26:54 +03:00
|
|
|
! @type.nil?
|
2015-05-31 13:02:29 +03:00
|
|
|
end
|
|
|
|
|
2016-02-25 11:50:10 -08:00
|
|
|
def get_type()
|
2019-09-23 00:07:30 +03:00
|
|
|
raise "No type " + self.object_id.to_s(16) + " : " + self.class.name+" : " + self.to_s unless @type
|
2019-09-09 20:26:54 +03:00
|
|
|
@type
|
2015-05-16 12:53:10 +03:00
|
|
|
end
|
|
|
|
|
2015-05-20 10:57:20 +03:00
|
|
|
def get_instance_variables
|
2019-09-09 20:26:54 +03:00
|
|
|
@type.names
|
2015-05-14 19:53:56 +03:00
|
|
|
end
|
|
|
|
|
2016-12-29 18:49:03 +02:00
|
|
|
def get_instance_variable( name )
|
2015-05-14 19:53:56 +03:00
|
|
|
index = instance_variable_defined(name)
|
2019-09-09 20:26:54 +03:00
|
|
|
#raise "at :#{name}:" if name.to_s[0] == "@"
|
2015-05-14 19:53:56 +03:00
|
|
|
return nil if index == nil
|
2015-11-18 15:36:43 +02:00
|
|
|
return get_internal_word(index)
|
2015-05-14 19:53:56 +03:00
|
|
|
end
|
|
|
|
|
2016-12-29 18:49:03 +02:00
|
|
|
def set_instance_variable( name , value )
|
2015-05-14 19:53:56 +03:00
|
|
|
index = instance_variable_defined(name)
|
2019-09-09 20:26:54 +03:00
|
|
|
#puts "setting #{name} at #{index}"
|
2015-05-14 19:53:56 +03:00
|
|
|
return nil if index == nil
|
2015-11-18 15:36:43 +02:00
|
|
|
return set_internal_word(index , value)
|
2015-05-14 19:53:56 +03:00
|
|
|
end
|
|
|
|
|
2016-12-29 18:49:03 +02:00
|
|
|
def instance_variable_defined( name )
|
2019-09-09 20:26:54 +03:00
|
|
|
@type.variable_index(name)
|
2015-05-14 19:53:56 +03:00
|
|
|
end
|
|
|
|
|
2019-08-17 21:07:07 +03:00
|
|
|
# objects only come in lengths of multiple of 8 words / 32 bytes
|
|
|
|
# and there is a "hidden" 1 word that is used for debug/check memory corruption
|
|
|
|
def self.padded( len )
|
|
|
|
a = 32 * (1 + ((len + 3)/32).floor )
|
|
|
|
#puts "#{a} for #{len}"
|
2019-08-17 23:29:42 +03:00
|
|
|
return a
|
2019-08-17 21:07:07 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.padded_words( words )
|
|
|
|
padded(words*4) # 4 == word length, a constant waiting for a home
|
|
|
|
end
|
|
|
|
|
2015-11-04 10:33:10 +02:00
|
|
|
def padded_length
|
2019-09-09 20:26:54 +03:00
|
|
|
Object.padded_words( @type.instance_length )
|
2015-05-24 18:05:20 +03:00
|
|
|
end
|
|
|
|
|
2015-11-18 11:55:29 +02:00
|
|
|
# parfait versions are deliberately called different, so we "relay"
|
2018-08-11 19:15:34 +03:00
|
|
|
# have to put the "" on the names for rfx to take them off again
|
2015-11-18 11:55:29 +02:00
|
|
|
def instance_variables
|
2019-02-10 21:00:25 +02:00
|
|
|
get_instance_variables.to_a.collect{ |n| n.to_s.to_sym }
|
2015-11-18 11:55:29 +02:00
|
|
|
end
|
2016-12-07 23:35:51 +02:00
|
|
|
|
2015-05-11 18:55:49 +03:00
|
|
|
end
|
2019-09-09 11:54:45 +03:00
|
|
|
end
|