rubyx/lib/virtual/parfait_adapter.rb

178 lines
4.6 KiB
Ruby
Raw Normal View History

2015-05-18 10:21:23 +03:00
# Below we define functions (in different classes) that are not part of the run-time
# They are used for the boot process, ie when this codes executes in the vm that builds salama
# To stay sane, we use the same classes that we use later, but "adapt" them to work in ruby
# This affects mainly memory layout
module Virtual
def self.new_list array
2015-07-20 13:01:15 +03:00
list = Parfait::List.new
list.set_length array.length
index = 1
while index <= array.length do
list.set(index , array[index - 1])
index = index + 1
end
list
end
end
class Symbol
include Positioned
2015-06-08 12:19:53 +02:00
include Padding
2015-06-19 13:11:40 +03:00
def has_layout?
true
end
def get_layout
l = Virtual.machine.space.classes[:Word].object_layout
#puts "LL #{l.class}"
l
end
def word_length
2015-06-08 12:19:53 +02:00
padded to_s.length
2015-06-01 08:33:23 +03:00
end
# not the prettiest addition to the game, but it wasn't me who decided symbols are frozen in 2.x
def cache_positions
unless defined?(@@symbol_positions)
@@symbol_positions = {}
end
@@symbol_positions
end
def position
pos = cache_positions[self]
if pos == nil
str = "position accessed but not set, "
str += "Machine has object=#{Virtual.machine.objects.has_key?(self.object_id)} "
2015-06-01 08:33:23 +03:00
raise str + " for Symbol:#{self}"
end
pos
end
def set_position pos
# resetting of position used to be error, but since relink and dynamic instruction size it is ok.
# in measures (of 32)
old = cache_positions[self]
if old != nil and ((old - pos).abs > 32)
raise "position set again #{pos}!=#{old} for #{self}"
end
cache_positions[self] = pos
end
end
2015-05-18 10:21:23 +03:00
module Parfait
# Objects memory functions. Object memory is 1 based
# but we implement it with ruby array (0 based) and use 0 as type-word
# These are the same functions that Builtin implements at run-time
class Object
include Padding
2015-06-08 11:37:20 +02:00
include Positioned
2015-06-06 18:46:53 +02:00
def fake_init
@memory = [0,nil]
@position = nil
self # for chaining
end
2015-05-18 10:21:23 +03:00
# these internal functions are _really_ internal
# they respresent the smallest code needed to build larger functionality
# but should _never_ be used outside parfait. in fact that should be impossible
def internal_object_get_typeword
raise "failed init for #{self.class}" unless @memory
2015-05-18 10:21:23 +03:00
@memory[0]
end
def internal_object_set_typeword w
raise "failed init for #{self.class}" unless @memory
2015-05-18 10:21:23 +03:00
@memory[0] = w
end
def internal_object_length
raise "failed init for #{self.class}" unless @memory
2015-05-18 10:21:23 +03:00
@memory.length - 1 # take of type-word
end
# 1 -based index
def internal_object_get(index)
@memory[index]
end
# 1 -based index
def internal_object_set(index , value)
raise "failed init for #{self.class}" unless @memory
@memory[index] = value
value
2015-05-18 10:21:23 +03:00
end
def internal_object_grow(length)
old_length = internal_object_length()
while( old_length < length )
internal_object_set( old_length + 1, nil)
old_length = old_length + 1
end
end
2015-05-30 14:22:33 +03:00
def internal_object_shrink(length)
old_length = internal_object_length()
while( length < old_length )
@memory.delete_at(old_length)
old_length = old_length - 1
end
end
2015-05-18 10:21:23 +03:00
end
class List
2015-05-20 10:57:20 +03:00
def to_sof_node(writer , level , ref )
Sof.array_to_sof_node(self , writer , level , ref )
end
def to_a
array = []
index = 1
while( index <= self.get_length)
array[index - 1] = get(index)
index = index + 1
end
array
end
2015-05-20 10:57:20 +03:00
end
class Dictionary
def to_sof_node(writer , level , ref)
Sof.hash_to_sof_node( self , writer , level , ref)
end
2015-05-18 10:21:23 +03:00
end
class Method
2015-07-03 20:13:03 +03:00
attr_accessor :source
end
2015-05-24 15:31:44 +03:00
class Word
2015-05-24 15:31:44 +03:00
def == other
return false unless other.is_a?(String) or other.is_a?(Word)
as_string = self.to_string
2015-05-24 15:31:44 +03:00
unless other.is_a? String
other = other.to_string
2015-05-24 15:31:44 +03:00
end
as_string == other
end
2015-06-03 10:01:59 +03:00
def to_string
string = ""
index = 1
while( index <= self.length)
string[index - 1] = get_char(index).chr
index = index + 1
end
string
end
end
## sof related stuff
class Object
# 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
end
end