rename length to mem_length and suffer the consequences
This commit is contained in:
parent
79e42003b2
commit
e4474e7e2c
@ -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
|
||||||
|
@ -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)
|
||||||
|
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user