2016-02-25 21:03:11 +01: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 14:55:24 +02:00
|
|
|
|
2016-12-06 10:38:09 +01: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 14:55:24 +02:00
|
|
|
|
2016-02-25 21:03:11 +01:00
|
|
|
# Objects are arranged or layed out (in memory) according to their Type
|
|
|
|
# every object has a Type. Type objects are immutalbe and may be reused for a group/class
|
2015-05-17 13:40:02 +02:00
|
|
|
# off objects.
|
2016-02-25 21:03:11 +01:00
|
|
|
# The Type of an object may change, but then a new Type is created
|
|
|
|
# The Type also defines the class of the object
|
|
|
|
# The Type is **always** the first entry (index 1) in an object
|
2015-05-17 13:40:02 +02:00
|
|
|
|
2015-05-11 17:55:49 +02:00
|
|
|
module Parfait
|
2016-02-25 21:03:11 +01:00
|
|
|
TYPE_INDEX = 1
|
2015-04-08 19:24:50 +02:00
|
|
|
|
2015-10-26 13:33:36 +01:00
|
|
|
class Object < Value
|
2015-05-17 13:40:02 +02:00
|
|
|
|
2016-12-06 10:38:09 +01:00
|
|
|
# we define new, so we can do memory layout also at compile time.
|
|
|
|
# At compile time we fake memory by using a global array for pages
|
2015-07-20 12:01:15 +02:00
|
|
|
def self.new *args
|
|
|
|
object = self.allocate
|
2015-07-21 14:40:25 +02:00
|
|
|
#HACK, but used to do the adapter in the init, bu that is too late now
|
|
|
|
object.fake_init if object.respond_to?(:fake_init) # at compile, not run-time
|
|
|
|
# have to grab the class, because we are in the ruby class not the parfait one
|
|
|
|
cl = Space.object_space.get_class_by_name( self.name.split("::").last.to_sym)
|
2016-02-25 20:50:10 +01:00
|
|
|
# and have to set the type before we let the object do anything. otherwise boom
|
2016-02-25 21:16:13 +01:00
|
|
|
object.set_type cl.instance_type
|
2015-07-21 14:40:25 +02:00
|
|
|
|
2015-07-20 12:01:15 +02:00
|
|
|
object.send :initialize , *args
|
2015-05-18 11:35:01 +02:00
|
|
|
object
|
|
|
|
end
|
|
|
|
|
2015-11-18 10:55:29 +01:00
|
|
|
include Padding
|
|
|
|
include Positioned
|
|
|
|
|
|
|
|
def fake_init
|
|
|
|
@memory = Array.new(16)
|
|
|
|
@position = nil
|
|
|
|
self # for chaining
|
|
|
|
end
|
|
|
|
|
|
|
|
# 1 -based index
|
2015-11-18 14:36:43 +01:00
|
|
|
def get_internal_word(index)
|
2015-11-18 10:55:29 +01:00
|
|
|
@memory[index]
|
|
|
|
end
|
|
|
|
# 1 -based index
|
2015-11-18 14:36:43 +01:00
|
|
|
def set_internal_word(index , value)
|
2015-11-18 10:55:29 +01:00
|
|
|
raise "failed init for #{self.class}" unless @memory
|
|
|
|
raise "Word[#{index}] = " if((self.class == Parfait::Word) and value.nil? )
|
|
|
|
@memory[index] = value
|
|
|
|
value
|
|
|
|
end
|
|
|
|
|
2015-07-21 14:40:25 +02:00
|
|
|
def self.attributes names
|
|
|
|
names.each{|name| attribute(name) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.attribute name
|
|
|
|
define_method(name) { get_instance_variable(name) }
|
|
|
|
define_method("#{name}=".to_sym) { |value| set_instance_variable(name , value) }
|
|
|
|
end
|
|
|
|
|
2015-06-03 09:01:59 +02:00
|
|
|
def == other
|
|
|
|
self.object_id == other.object_id
|
|
|
|
end
|
|
|
|
|
2015-05-11 17:55:49 +02:00
|
|
|
# This is the crux of the object system. The class of an object is stored in the objects
|
|
|
|
# memory (as opposed to an integer that has no memory and so always has the same class)
|
|
|
|
#
|
2016-02-25 21:03:11 +01:00
|
|
|
# In Salama we store the class in the Type, and so the Type is the only fixed
|
2015-05-11 17:55:49 +02:00
|
|
|
# data that every object carries.
|
|
|
|
def get_class()
|
2016-02-25 20:50:10 +01:00
|
|
|
l = get_type()
|
|
|
|
#puts "Type #{l.class} in #{self.class} , #{self}"
|
2015-07-21 14:40:25 +02:00
|
|
|
l.object_class()
|
2015-05-11 17:55:49 +02:00
|
|
|
end
|
2015-04-08 19:24:50 +02:00
|
|
|
|
2015-05-22 21:51:36 +02:00
|
|
|
# private
|
2016-02-25 20:50:10 +01:00
|
|
|
def set_type(type)
|
|
|
|
# puts "Type was set for #{self.class}"
|
|
|
|
raise "Nil type" unless type
|
2016-02-25 21:03:11 +01:00
|
|
|
set_internal_word(TYPE_INDEX , type)
|
2015-05-22 21:51:36 +02:00
|
|
|
end
|
|
|
|
|
2016-02-25 20:50:10 +01:00
|
|
|
# so we can keep the raise in get_type
|
|
|
|
def has_type?
|
2016-02-25 21:03:11 +01:00
|
|
|
! get_internal_word(TYPE_INDEX).nil?
|
2015-05-31 12:02:29 +02:00
|
|
|
end
|
|
|
|
|
2016-02-25 20:50:10 +01:00
|
|
|
def get_type()
|
2016-02-25 21:03:11 +01:00
|
|
|
l = get_internal_word(TYPE_INDEX)
|
2016-02-25 20:50:10 +01:00
|
|
|
#puts "get type for #{self.class} returns #{l.class}"
|
|
|
|
raise "No type #{self.object_id.to_s(16)}:#{self.class} " unless l
|
2015-05-25 17:48:35 +02:00
|
|
|
return l
|
2015-05-16 11:53:10 +02:00
|
|
|
end
|
|
|
|
|
2015-11-06 23:12:17 +01:00
|
|
|
# return the metaclass
|
|
|
|
def meta
|
|
|
|
MetaClass.new self
|
|
|
|
end
|
|
|
|
|
2015-05-20 09:57:20 +02:00
|
|
|
def get_instance_variables
|
2016-02-25 20:50:10 +01:00
|
|
|
get_type().instance_names
|
2015-05-14 18:53:56 +02:00
|
|
|
end
|
|
|
|
|
2015-05-20 09:57:20 +02:00
|
|
|
def get_instance_variable name
|
2015-05-14 18:53:56 +02:00
|
|
|
index = instance_variable_defined(name)
|
2015-07-21 14:40:25 +02:00
|
|
|
#puts "getting #{name} at #{index}"
|
2015-05-14 18:53:56 +02:00
|
|
|
return nil if index == nil
|
2015-11-18 14:36:43 +01:00
|
|
|
return get_internal_word(index)
|
2015-05-14 18:53:56 +02:00
|
|
|
end
|
|
|
|
|
2015-05-20 09:57:20 +02:00
|
|
|
def set_instance_variable name , value
|
2015-05-14 18:53:56 +02:00
|
|
|
index = instance_variable_defined(name)
|
|
|
|
return nil if index == nil
|
2015-11-18 14:36:43 +01:00
|
|
|
return set_internal_word(index , value)
|
2015-05-14 18:53:56 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def instance_variable_defined name
|
2016-02-25 20:50:10 +01:00
|
|
|
get_type().variable_index(name)
|
2015-05-14 18:53:56 +02:00
|
|
|
end
|
|
|
|
|
2015-11-04 09:33:10 +01:00
|
|
|
def padded_length
|
2016-02-25 20:50:10 +01:00
|
|
|
padded_words( get_type().instance_length )
|
2015-05-24 17:05:20 +02:00
|
|
|
end
|
|
|
|
|
2015-11-18 10:55:29 +01:00
|
|
|
# parfait versions are deliberately called different, so we "relay"
|
|
|
|
# have to put the "@" on the names for sof to take them off again
|
|
|
|
def instance_variables
|
|
|
|
get_instance_variables.to_a.collect{ |n| "@#{n}".to_sym }
|
|
|
|
end
|
|
|
|
# name comes in as a ruby @var name
|
|
|
|
def instance_variable_get name
|
|
|
|
var = get_instance_variable name.to_s[1 .. -1].to_sym
|
|
|
|
#puts "getting #{name} #{var}"
|
|
|
|
var
|
|
|
|
end
|
|
|
|
|
2015-05-11 17:55:49 +02:00
|
|
|
end
|
2014-07-30 20:43:12 +02:00
|
|
|
end
|