positioning wip

This commit is contained in:
Torsten Ruger 2018-06-02 16:12:01 +03:00
parent d1cc482f8f
commit aa6707337a
8 changed files with 84 additions and 91 deletions

View File

@ -85,7 +85,7 @@ module Risc
def position_all def position_all
raise "Not translated " unless @translated raise "Not translated " unless @translated
#need the initial jump at 0 and then functions #need the initial jump at 0 and then functions
Position.set(cpu_init , 0 , nil) Position::ObjectPosition.init(cpu_init , -1)
code_start = position_objects( @platform.padding ) code_start = position_objects( @platform.padding )
# and then everything code # and then everything code
position_code(code_start) position_code(code_start)
@ -101,7 +101,7 @@ module Risc
sorted.each do | objekt| sorted.each do | objekt|
next if objekt.is_a?( Parfait::BinaryCode) or objekt.is_a?( Risc::Label ) next if objekt.is_a?( Parfait::BinaryCode) or objekt.is_a?( Risc::Label )
before = at before = at
position = Position.set(objekt,at) position = Position::ObjectPosition.init(objekt,at)
previous.register_event(:position_changed , Position::ObjectListener.new(objekt)) if previous previous.register_event(:position_changed , Position::ObjectListener.new(objekt)) if previous
previous = position previous = position
at += objekt.padded_length at += objekt.padded_length

View File

@ -12,14 +12,13 @@ module Risc
attr_reader :code , :method attr_reader :code , :method
def initialize(code, pos , method) def initialize(code , method)
super(code,pos) super(code,pos)
@code = code @code = code
@method = method @method = method
raise "Method nil" unless method raise "Method nil" unless method
end end
def init(at , method) def set(at )
raise "No no" unless method.name == @method.name
next_pos = at + code.padded_length next_pos = at + code.padded_length
if code.next if code.next
Position.set(code.next , next_pos, method) Position.set(code.next , next_pos, method)
@ -32,11 +31,7 @@ module Risc
Position.set( next_meth.cpu_instructions, next_cpu_pos , next_meth.binary) Position.set( next_meth.cpu_instructions, next_cpu_pos , next_meth.binary)
end end
end end
def reset_to(pos , ignored)
super(pos, ignored)
init(pos , ignored)
Position.log.debug "ResetCode (#{pos.to_s(16)}) #{code}"
end
# insert a jump to the next instruction, at the last instruction # insert a jump to the next instruction, at the last instruction
# thus hopping over the object header # thus hopping over the object header
def set_jump(at) def set_jump(at)

View File

@ -16,9 +16,9 @@ module Risc
# #
class InstructionPosition < ObjectPosition class InstructionPosition < ObjectPosition
attr_reader :instruction , :binary attr_reader :instruction , :binary
def initialize(instruction, pos , binary) def initialize(instruction , binary)
raise "not set #{binary}" if pos != 0 and !binary.is_a?(Parfait::BinaryCode) pos = 0
super(instruction,pos) super(instruction)
@instruction = instruction @instruction = instruction
@binary = binary @binary = binary
end end
@ -47,6 +47,25 @@ module Risc
init(pos , binary) init(pos , binary)
Position.log.debug "ResetInstruction (#{pos.to_s(16)}) #{instruction}" Position.log.debug "ResetInstruction (#{pos.to_s(16)}) #{instruction}"
end end
# initialize the dependency graph for instructions
#
# starting from the given instruction, create InstructionPositions
# for it and the whole chain
#
# Set the next created to be dependent on the previous
def self.init( instruction , code)
while(instruction)
position = InstructionPosition.new(instruction , code)
nekst = instruction.next
if nekst
listener = InstructionListener.new( nekst )
position.register_event(:position_changed , listener)
end
instruction = nekst
end
position
end
end end
end end
end end

View File

@ -7,11 +7,13 @@ module Risc
attr_reader :at , :object attr_reader :at , :object
def initialize( object, at) # initialize with a given object, first parameter
@at = at # The object ill be the key in global position map
# Give an integer as the actual position, where -1
# which means no legal position known
def initialize(object , pos )
@at = 0
@object = object @object = object
@listeners = []
raise "not int #{self}-#{at}" unless @at.is_a?(Integer)
end end
def +(offset) def +(offset)
@ -31,13 +33,17 @@ module Risc
end end
def reset_to(pos , guaranteed_nil ) def reset_to(pos , guaranteed_nil )
return false if pos == at return false if pos == at
trigger(:position_changed , self)
if((at - pos).abs > 1000) if((at - pos).abs > 1000)
raise "position set too far off #{pos}!=#{at} for #{object}:#{object.class}" raise "position set too far off #{pos}!=#{at} for #{object}:#{object.class}"
end end
@at = pos @at = pos
trigger(:position_changed , self)
true true
end end
def self.init(object , at = -1)
position = ObjectPosition.new(object , at)
Position.set_to( position , at)
end
end end
end end
end end

View File

@ -56,52 +56,19 @@ module Risc
old.reset_to( old.at ) old.reset_to( old.at )
end end
# resetting of position used to be error, but since relink and dynamic instruction size it is ok. def self.set_to( position , to)
# in measures postest = Position.positions[position.object]
# raise "Mismatch #{position}" if postest and postest != position
# reseting to the same position as before, triggers code that propagates @reverse_cache.delete(position.at) unless position.object.is_a? Label
def self.reset(position , to , extra) testing = self.at( position.at ) if position.at >= 0
log.debug "ReSetting #{position}, to:#{to.to_s(16)}, for #{position.object.class}-#{position.object}"
position.reset_to(to, extra)
if testing = @reverse_cache[ to ]
if testing.class != position.class
raise "Mismatch (at #{to.to_s(16)}) new:#{position}:#{position.class} , was #{testing}:#{testing.class}"
end
end
unless position.object.is_a? Label
@reverse_cache.delete(to)
@reverse_cache[position.at] = position
end
log.debug "Reset #{position} (#{to.to_s(16)}) for #{position.class}"
return position
end
def self.set( object , pos , extra = nil)
old = Position.positions[object]
return self.reset(old , pos , extra) if old
log.debug "Setting #{pos.to_s(16)} for #{object.class}-#{object}"
testing = self.at( pos )
position = for_at( object , pos , extra)
if testing and testing.class != position.class if testing and testing.class != position.class
raise "Mismatch (at #{pos.to_s(16)}) was:#{position} #{position.class} #{position.object} , should #{testing}#{testing.class}" raise "Mismatch (at #{pos.to_s(16)}) was:#{position} #{position.class} #{position.object} , should #{testing}#{testing.class}"
end end
self.positions[object] = position self.positions[position.object] = position
position.init(pos , extra) @reverse_cache[to] = position unless position.object.is_a? Label
@reverse_cache[position.at] = position unless object.is_a? Label log.debug "Set #{position} (#{to.to_s(16)}) for #{position.class}"
log.debug "Set #{position} (#{pos.to_s(16)}) for #{position.class}"
position position
end end
def self.for_at(object , at , extra)
case object
when Parfait::BinaryCode
CodePosition.new(object,at , extra)
when Arm::Instruction , Risc::Instruction
InstructionPosition.new(object,at , extra)
else
ObjectPosition.new(object,at)
end
end
end end
end end
require_relative "object_position" require_relative "object_position"

View File

@ -3,32 +3,35 @@ require_relative "helper"
module Risc module Risc
module Position module Position
# tests that require a boot and test propagation # tests that require a boot and test propagation
class TestPositionBasic < MiniTest::Test class TestInstructionPosition < MiniTest::Test
def setup def setup
Risc.machine.boot Risc.machine.boot
@binary = Parfait::BinaryCode.new(1) @binary = Parfait::BinaryCode.new(1)
Position.set(@binary , 0 , Parfait.object_space.get_main) #Position.set(@binary , 0 , Parfait.object_space.get_main)
@label = Risc.label("hi","ho") @label = Risc.label("hi","ho")
end end
def test_set_instr def test_init
assert InstructionPosition.init(@label , nil)
end
def pest_set_instr
pos = Position.set( @label , 8 , @binary) pos = Position.set( @label , 8 , @binary)
assert_equal InstructionPosition , pos.class assert_equal InstructionPosition , pos.class
end end
def test_label_set_int def pest_label_set_int
Position.set( @label , 8 , @binary) Position.set( @label , 8 , @binary)
assert_equal 8 , @label.address.value assert_equal 8 , @label.address.value
end end
def test_label_reset_int def pest_label_reset_int
Position.set( @label , 8 , @binary) Position.set( @label , 8 , @binary)
Position.set( @label , 18 , @binary) Position.set( @label , 18 , @binary)
assert_equal 18 , @label.address.value assert_equal 18 , @label.address.value
end end
def test_ins_propagates def pest_ins_propagates
@label.set_next Arm::ArmMachine.b( @label) @label.set_next Arm::ArmMachine.b( @label)
Position.set( @label , 8 , @binary) Position.set( @label , 8 , @binary)
assert_equal 8 , Position.get(@label.next).at assert_equal 8 , Position.get(@label.next).at
end end
def test_ins_propagates_again def pest_ins_propagates_again
second = Arm::ArmMachine.b( @label) second = Arm::ArmMachine.b( @label)
@label.set_next(second) @label.set_next(second)
Position.set( @label , 8 , @binary) Position.set( @label , 8 , @binary)
@ -36,7 +39,7 @@ module Risc
Position.set( @label , 8 , @binary) Position.set( @label , 8 , @binary)
assert_equal 8 , Position.get(@label.next).at assert_equal 8 , Position.get(@label.next).at
end end
def test_label_at def pest_label_at
branch = Branch.new("b" , @label) branch = Branch.new("b" , @label)
Position.set(@label , 8 , @binary) Position.set(@label , 8 , @binary)
Position.set(branch , 8 , @binary) Position.set(branch , 8 , @binary)
@ -44,7 +47,7 @@ module Risc
assert_equal InstructionPosition , at_4.class assert_equal InstructionPosition , at_4.class
assert_equal Branch , at_4.instruction.class assert_equal Branch , at_4.instruction.class
end end
def test_label_at_reverse def pest_label_at_reverse
branch = Branch.new("b" , @label) branch = Branch.new("b" , @label)
Position.set(branch , 8 , @binary) Position.set(branch , 8 , @binary)
Position.set(@label , 8 , @binary) Position.set(@label , 8 , @binary)
@ -52,7 +55,7 @@ module Risc
assert_equal InstructionPosition , at_4.class assert_equal InstructionPosition , at_4.class
assert_equal Branch , at_4.instruction.class assert_equal Branch , at_4.instruction.class
end end
def test_reset_false_type def pest_reset_false_type
assert_raises {Position.set(@label , 0 , @binary)} assert_raises {Position.set(@label , 0 , @binary)}
end end
end end

View File

@ -5,44 +5,47 @@ module Risc
# tests that do no require a boot and only test basic positioning # tests that do no require a boot and only test basic positioning
class TestPositionBasic < MiniTest::Test class TestPositionBasic < MiniTest::Test
def test_creation_ok def test_init
assert_equal ObjectPosition.init(self)
end
def pest_creation_ok
assert ObjectPosition.new(self,0) assert ObjectPosition.new(self,0)
end end
def test_creation_fail def pest_creation_fail
assert_raises {Position.new("0")} assert_raises {Position.new("0")}
end end
def test_add def pest_add
res = ObjectPosition.new(self,0) + 5 res = ObjectPosition.new(self,0) + 5
assert_equal 5 , res assert_equal 5 , res
end end
def test_sub def pest_sub
res = ObjectPosition.new(self,0) - 1 res = ObjectPosition.new(self,0) - 1
assert_equal -1 , res assert_equal -1 , res
end end
def test_sub_pos def pest_sub_pos
res = ObjectPosition.new(self,0) - ObjectPosition.new(self,0) res = ObjectPosition.new(self,0) - ObjectPosition.new(self,0)
assert_equal 0 , res assert_equal 0 , res
end end
def test_set def pest_set
pos = Position.set(self , 5) pos = Position.set(self , 5)
assert_equal 5 , pos.at assert_equal 5 , pos.at
end end
def tet_tos def tet_tos
assert_equal "0x10" , Position.set(self).to_s assert_equal "0x10" , Position.set(self).to_s
end end
def test_reset_ok def pest_reset_ok
pos = Position.set(self , 5) pos = Position.set(self , 5)
pos = Position.set(self , 10) pos = Position.set(self , 10)
assert_equal 10 , pos.at assert_equal 10 , pos.at
end end
def test_reset_fail def pest_reset_fail
Position.set(self , 5) Position.set(self , 5)
assert_raises{Position.set(self , 10000)} assert_raises{Position.set(self , 10000)}
end end
def test_raises_set_nil def pest_raises_set_nil
assert_raises { Position.set(self,nil)} assert_raises { Position.set(self,nil)}
end end
def test_at def pest_at
pos = Position.set(self , 5) pos = Position.set(self , 5)
pos = Position.at(5) pos = Position.at(5)
assert_equal 5 , pos.at assert_equal 5 , pos.at
@ -50,21 +53,21 @@ module Risc
end end
class TestPositionEvents < MiniTest::Test class TestPositionEvents < MiniTest::Test
def setup def setup
@position = ObjectPosition.new(self,0) @position = ObjectPosition.new(self)
end end
def test_has_register def pest_has_register
assert @position.register_event(:position_changed , self) assert @position.register_event(:position_changed , self)
end end
def test_can_unregister def pest_can_unregister
assert @position.register_event(:position_changed ,self) assert @position.register_event(:position_changed ,self)
assert @position.unregister_event(:position_changed ,self) assert @position.unregister_event(:position_changed ,self)
end end
def test_fires def pest_fires
@position.register_event(:position_changed ,self) @position.register_event(:position_changed ,self)
@position.trigger(:position_changed , @position) @position.trigger(:position_changed , @position)
assert_equal @position , @trigger assert_equal @position , @trigger
end end
def test_no_fire_after_unregister def pest_no_fire_after_unregister
assert @position.register_event(:position_changed ,self) assert @position.register_event(:position_changed ,self)
assert @position.unregister_event(:position_changed ,self) assert @position.unregister_event(:position_changed ,self)
@position.trigger(:position_changed , @position) @position.trigger(:position_changed , @position)

View File

@ -5,25 +5,25 @@ module Risc
def setup def setup
@machine = Risc.machine.boot @machine = Risc.machine.boot
end end
def test_cpu_init def pest_cpu_init
@machine.translate(:interpreter) @machine.translate(:interpreter)
@machine.position_all @machine.position_all
assert Position.get @machine.cpu_init assert Position.get @machine.cpu_init
end end
def test_cpu_label def pest_cpu_label
@machine.translate(:interpreter) @machine.translate(:interpreter)
@machine.position_all @machine.position_all
assert Position.get( @machine.cpu_init.label ) assert Position.get( @machine.cpu_init.label )
end end
def test_cpu_first_arm def pest_cpu_first_arm
@machine.translate(:arm) @machine.translate(:arm)
@machine.position_all @machine.position_all
assert Position.get( @machine.cpu_init.first ) assert Position.get( @machine.cpu_init.first )
end end
def test_has_arm_pos def pest_has_arm_pos
has_positions(:arm) has_positions(:arm)
end end
def test_has_int_pos def pest_has_int_pos
has_positions(:interpreter) has_positions(:interpreter)
end end
def has_positions(platform) def has_positions(platform)
@ -33,10 +33,10 @@ module Risc
assert Position.get(obj) assert Position.get(obj)
end end
end end
def test_has_arm_meth def pest_has_arm_meth
meth_positions(:arm) meth_positions(:arm)
end end
def test_has_int_meth def pest_has_int_meth
meth_positions(:interpreter) meth_positions(:interpreter)
end end
def meth_positions(platform) def meth_positions(platform)