rename length to mem_length and suffer the consequences

This commit is contained in:
Torsten Ruger 2014-09-17 12:04:54 +03:00
parent 79e42003b2
commit e4474e7e2c
7 changed files with 27 additions and 30 deletions

View File

@ -17,7 +17,7 @@ module Arm
@is_load = opcode.to_s[0] == "l" ? 1 : 0 #L (load) flag @is_load = opcode.to_s[0] == "l" ? 1 : 0 #L (load) flag
end end
def assemble(io , assembler ) def assemble(io )
# don't overwrite instance variables, to make assembly repeatable # don't overwrite instance variables, to make assembly repeatable
rn = @rn rn = @rn
operand = @operand operand = @operand

View File

@ -72,7 +72,10 @@ module Arm
immediate = 1 immediate = 1
raise "hmm" raise "hmm"
else else
raise "cannot fit numeric literal argument in operand #{right.inspect}" operand = right.integer / 256
immediate = 1
# raise "cannot fit numeric literal argument in operand #{right.inspect}"
end end
elsif (right.is_a?(Symbol) or right.is_a?(Virtual::Integer)) elsif (right.is_a?(Symbol) or right.is_a?(Virtual::Integer))
operand = reg_code(right) #integer means the register the integer is in (otherwise constant) operand = reg_code(right) #integer means the register the integer is in (otherwise constant)

View File

@ -14,7 +14,7 @@ module Elf
Elf::Constants::SHF_WRITE | Elf::Constants::SHF_ALLOC | Elf::Constants::SHF_EXECINSTR Elf::Constants::SHF_WRITE | Elf::Constants::SHF_ALLOC | Elf::Constants::SHF_EXECINSTR
end end
def mem_length def length
@text.length @text.length
end end
def to_s def to_s

View File

@ -79,9 +79,9 @@ module Register
raise "Object(#{object.object_id}) not linked #{object.inspect}" unless @objects[object.object_id] raise "Object(#{object.object_id}) not linked #{object.inspect}" unless @objects[object.object_id]
type = type_word(variables) type = type_word(variables)
@stream.write_uint32( type ) @stream.write_uint32( type )
write_ref_for(object.layout[:names] , object ) write_ref_for(object.layout[:names] )
variables.each do |var| variables.each do |var|
write_ref_for(var , object) write_ref_for(var)
end end
pad_after( variables.length * 4 ) pad_after( variables.length * 4 )
object.position object.position
@ -90,9 +90,9 @@ module Register
def assemble_Array array def assemble_Array array
type = type_word(array) type = type_word(array)
@stream.write_uint32( type ) @stream.write_uint32( type )
write_ref_for(layout[:names],array) #ref write_ref_for(array.layout[:names]) #ref
array.each do |var| array.each do |var|
write_ref_for(var,array) write_ref_for(var)
end end
pad_after( array.length * 4 ) pad_after( array.length * 4 )
array.position array.position
@ -112,13 +112,13 @@ module Register
end end
def assemble_CompiledMethod(method) def assemble_CompiledMethod(method)
count = method.blocks.inject(0) { |c , block| c += block.length } count = method.blocks.inject(0) { |c , block| c += block.mem_length }
word = (count+7) / 32 # all object are multiple of 8 words (7 for header) word = (count+7) / 32 # all object are multiple of 8 words (7 for header)
raise "Method too long, splitting not implemented #{method.name}/#{count}" if word > 15 raise "Method too long, splitting not implemented #{method.name}/#{count}" if word > 15
# first line is integers, convention is that following lines are the same # first line is integers, convention is that following lines are the same
TYPE_LENGTH.times { word = ((word << TYPE_BITS) + TYPE_INT) } TYPE_LENGTH.times { word = ((word << TYPE_BITS) + TYPE_INT) }
@stream.write_uint32( word ) @stream.write_uint32( word )
write_ref_for(method.layout[:names] , method) #ref of layout write_ref_for(method.layout[:names]) #ref of layout
# TODO the assembly may have to move to the object to be more extensible # TODO the assembly may have to move to the object to be more extensible
method.blocks.each do |block| method.blocks.each do |block|
block.codes.each do |code| block.codes.each do |code|
@ -131,14 +131,14 @@ module Register
def assemble_String( str ) def assemble_String( str )
str = str.string if str.is_a? Virtual::StringConstant str = str.string if str.is_a? Virtual::StringConstant
str = str.to_s if str.is_a? Symbol str = str.to_s if str.is_a? Symbol
word = (str.mem_length + 7) / 32 # all object are multiple of 8 words (7 for header) word = (str.length + 7) / 32 # all object are multiple of 8 words (7 for header)
raise "String too long (implement split string!) #{word}" if word > 15 raise "String too long (implement split string!) #{word}" if word > 15
# first line is integers, convention is that following lines are the same # first line is integers, convention is that following lines are the same
TYPE_LENGTH.times { word = ((word << TYPE_BITS) + TYPE_INT) } TYPE_LENGTH.times { word = ((word << TYPE_BITS) + TYPE_INT) }
@stream.write_uint32( word ) @stream.write_uint32( word )
write_ref_for( str.layout[:names] , slot) #ref write_ref_for( str.layout[:names] ) #ref
@stream.write str @stream.write str
pad_after(str.mem_length) pad_after(str.length)
#puts "String (#{slot.mem_length}) stream #{@stream.mem_length.to_s(16)}" #puts "String (#{slot.mem_length}) stream #{@stream.mem_length.to_s(16)}"
end end
@ -181,6 +181,7 @@ module Register
add_object(clazz.instance_methods) add_object(clazz.instance_methods)
end end
def add_CompiledMethod(method) def add_CompiledMethod(method)
add_object(method.name)
end end
def add_String( str) def add_String( str)
end end
@ -194,8 +195,7 @@ module Register
# write means we write the resulting address straight into the assembler stream (ie don't return it) # write means we write the resulting address straight into the assembler stream (ie don't return it)
# object means the object of which we write the address # object means the object of which we write the address
# and we write the address into the self, given as second parameter # and we write the address into the self, given as second parameter
def write_ref_for object , self_ref def write_ref_for object
raise "Object (#{object.object_id}) not linked #{object.inspect}" unless slot
@stream.write_sint32 object.position @stream.write_sint32 object.position
end end
@ -218,7 +218,7 @@ module Register
pad.times do pad.times do
@stream.write_uint8(0) @stream.write_uint8(0)
end end
#puts "padded #{length} with #{pad} stream pos #{@stream.mem_length.to_s(16)}" puts "padded #{length} with #{pad} stream pos #{@stream.length.to_s(16)}"
end end
end end

View File

@ -172,15 +172,13 @@ module Virtual
end end
# position of the function is the position of the entry block, is where we call # position of the function is the position of the entry block, is where we call
def set_position at def set_position at
super
at += 8 #for the 2 header words at += 8 #for the 2 header words
@blocks.each do |block| @blocks.each do |block|
block.set_position at block.set_position at
at = at + block.mem_length at = at + block.mem_length
end end
end end
def position
@blocks.first.position
end
end end
end end

View File

@ -53,6 +53,11 @@ module Virtual
def mem_length def mem_length
padded(1 + string.length) padded(1 + string.length)
end end
def position
return @position if @position
return @string.position if @string.position
super
end
end end
end end

View File

@ -22,7 +22,7 @@ module Virtual
end end
attr_accessor :length , :layout attr_accessor :length , :layout
def position def position
raise "position accessed but not set at #{length} for #{self.objekt}" if @position == nil raise "position accessed but not set at #{length} for #{self.inspect}" if @position == nil
@position @position
end end
def set_position pos def set_position pos
@ -75,21 +75,12 @@ module Virtual
def padded_words words def padded_words words
padded(words*4) # 4 == word length, a constant waiting for a home padded(words*4) # 4 == word length, a constant waiting for a home
end end
# pad_after is always in bytes and pads (writes 0's) up to the next 8 word boundary
def pad_after length
pad = padded(length) - length - 8 # for header, type and layout
pad.times do
@stream.write_uint8(0)
end
#puts "padded #{length} with #{pad} stream pos #{@stream.length.to_s(16)}"
end
end end
end end
Parfait::Hash.class_eval do Parfait::Hash.class_eval do
@@HASH = { :names => [:keys,:values] , :types => [Virtual::Reference,Virtual::Reference]} HASH = { :names => [:keys,:values] , :types => [Virtual::Reference,Virtual::Reference]}
def layout def layout
@@HASH HASH
end end
def set_position pos def set_position pos
@position = pos @position = pos