introduce the LabeListener to move instructions along when first code position changes
This commit is contained in:
parent
ec1d38f5a6
commit
c7ad1d98ca
@ -14,7 +14,7 @@ module Risc
|
|||||||
# fire events for changed pc and register contents
|
# fire events for changed pc and register contents
|
||||||
include Util::Eventable
|
include Util::Eventable
|
||||||
include Util::Logging
|
include Util::Logging
|
||||||
log_level :debug
|
log_level :info
|
||||||
|
|
||||||
attr_reader :instruction , :clock , :pc # current instruction and pc
|
attr_reader :instruction , :clock , :pc # current instruction and pc
|
||||||
attr_reader :registers # the registers, 16 (a hash, sym -> contents)
|
attr_reader :registers # the registers, 16 (a hash, sym -> contents)
|
||||||
|
@ -120,12 +120,13 @@ module Risc
|
|||||||
Parfait.object_space.types.values.each do |type|
|
Parfait.object_space.types.values.each do |type|
|
||||||
next unless type.methods
|
next unless type.methods
|
||||||
type.methods.each_method do |method|
|
type.methods.each_method do |method|
|
||||||
Position.log.debug "Position starts for method #{method.name}"
|
#next unless method.name == :main or method.name == :__init__
|
||||||
last_code = CodeListener.init(method.binary)
|
Position.log.debug "Method start #{code_start.to_s(16)} #{method.name}"
|
||||||
last_code.set(code_start)
|
code_pos = CodeListener.init(method.binary)
|
||||||
first_position = InstructionListener.init(method.cpu_instructions, method.binary)
|
InstructionListener.init(method.cpu_instructions, method.binary)
|
||||||
first_position.set( code_start + Parfait::BinaryCode.byte_offset)
|
code_pos.position_listener( LabelListener.new(method.cpu_instructions))
|
||||||
code_start = last_code.next_slot
|
code_pos.set(code_start)
|
||||||
|
code_start = Position.get(method.binary.last_code).next_slot
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
#Position.set( first_method.cpu_instructions, code_start + Parfait::BinaryCode.byte_offset , first_method.binary)
|
#Position.set( first_method.cpu_instructions, code_start + Parfait::BinaryCode.byte_offset , first_method.binary)
|
||||||
|
@ -25,6 +25,7 @@ module Risc
|
|||||||
# Taking into account that BinaryCodes only take 13 instructions,
|
# Taking into account that BinaryCodes only take 13 instructions,
|
||||||
# meaning that chain may have to be extended
|
# meaning that chain may have to be extended
|
||||||
def position_changing(position , to)
|
def position_changing(position , to)
|
||||||
|
Position.log.debug "Changing #{position} to #{to.to_s(16)}, bin #{Position.get(@binary)}"
|
||||||
update_index(to)
|
update_index(to)
|
||||||
instruction = position.object
|
instruction = position.object
|
||||||
return unless instruction.next
|
return unless instruction.next
|
||||||
@ -40,7 +41,7 @@ module Risc
|
|||||||
|
|
||||||
def update_index(to)
|
def update_index(to)
|
||||||
index = (to - Position.get(@binary).at) / 4
|
index = (to - Position.get(@binary).at) / 4
|
||||||
raise "Invalid negative index #{@index} , #{Position.get(@binary)}" if index < Parfait::BinaryCode.type_length
|
raise "Invalid negative index #{index} , #{Position.get(@binary)}" if index < Parfait::BinaryCode.type_length
|
||||||
while(index >= (Parfait::BinaryCode.memory_size - 1) )
|
while(index >= (Parfait::BinaryCode.memory_size - 1) )
|
||||||
@binary = @binary.ensure_next
|
@binary = @binary.ensure_next
|
||||||
index = (to - Position.get(@binary).at) / 4
|
index = (to - Position.get(@binary).at) / 4
|
||||||
|
39
lib/risc/position/label_listener.rb
Normal file
39
lib/risc/position/label_listener.rb
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
module Risc
|
||||||
|
|
||||||
|
# LabelListener is the one point that connects the BinaryCode
|
||||||
|
# and Instruction positions.
|
||||||
|
#
|
||||||
|
# LabelListener is instantiated with the first label of a Method
|
||||||
|
# and attached to the first BinaryCode.
|
||||||
|
#
|
||||||
|
# When the code moves, the label position is updated.
|
||||||
|
#
|
||||||
|
# The first code may get pushed by a previous method, and there is otherwise
|
||||||
|
# no way to react to this.
|
||||||
|
class LabelListener
|
||||||
|
|
||||||
|
attr_reader :label
|
||||||
|
|
||||||
|
# initialize with the first label of the method
|
||||||
|
def initialize(label)
|
||||||
|
@label = label
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
# The incoming position is the first BinaryCode of the method
|
||||||
|
# We simply position the Label (instance, see initialize) as the
|
||||||
|
# first entry in the BinaryCode, at BinaryCode.byte_offset
|
||||||
|
def position_changed(position)
|
||||||
|
label_pos = Position.get(@label)
|
||||||
|
label_pos.set(position + Parfait::BinaryCode.byte_offset)
|
||||||
|
end
|
||||||
|
|
||||||
|
# don't react to insertion, as the CodeListener will take care
|
||||||
|
def position_inserted(position)
|
||||||
|
end
|
||||||
|
|
||||||
|
# dont react, as we do the work in position_changed
|
||||||
|
def position_changing(position , to)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -190,3 +190,4 @@ end
|
|||||||
require_relative "position_listener"
|
require_relative "position_listener"
|
||||||
require_relative "instruction_listener"
|
require_relative "instruction_listener"
|
||||||
require_relative "code_listener"
|
require_relative "code_listener"
|
||||||
|
require_relative "label_listener"
|
||||||
|
25
test/risc/position/test_label_listener.rb
Normal file
25
test/risc/position/test_label_listener.rb
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
require_relative "helper"
|
||||||
|
|
||||||
|
module Risc
|
||||||
|
class TestLabelListener < MiniTest::Test
|
||||||
|
|
||||||
|
def setup
|
||||||
|
Risc.machine.boot
|
||||||
|
@label = Label.new("Hi","Ho" , FakeAddress.new(5))
|
||||||
|
@label_pos = Position.new(@label , -1)
|
||||||
|
@code = Parfait::BinaryCode.new(1)
|
||||||
|
@code_pos = Position.new(@code , -1)
|
||||||
|
@code_pos.position_listener( LabelListener.new(@label))
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_move
|
||||||
|
@code_pos.set(0)
|
||||||
|
assert_equal 8 , @label_pos.at
|
||||||
|
end
|
||||||
|
def test_move_twice
|
||||||
|
@code_pos.set(0)
|
||||||
|
@code_pos.set(0x40)
|
||||||
|
assert_equal 72 , @label_pos.at
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -34,6 +34,18 @@ module Risc
|
|||||||
assert_equal 5, count
|
assert_equal 5, count
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
class TestMachinePos < MiniTest::Test
|
||||||
|
def setup
|
||||||
|
@machine = Risc.machine.boot
|
||||||
|
@machine.translate(:arm)
|
||||||
|
@machine.position_all
|
||||||
|
end
|
||||||
|
def test_positions_set
|
||||||
|
@machine.objects.each do |id,obj|
|
||||||
|
assert Position.get(obj).valid? , "#{Position.get(obj)} , #{obj.object_id.to_s(16)}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
class TestMachineInit < MiniTest::Test
|
class TestMachineInit < MiniTest::Test
|
||||||
def setup
|
def setup
|
||||||
@machine = Risc.machine.boot
|
@machine = Risc.machine.boot
|
||||||
@ -45,10 +57,7 @@ module Risc
|
|||||||
assert_equal 0 , Position.get(@machine.cpu_init).at
|
assert_equal 0 , Position.get(@machine.cpu_init).at
|
||||||
end
|
end
|
||||||
def test_cpu_at
|
def test_cpu_at
|
||||||
assert_equal "0x5e08" , Position.get(@machine.cpu_init.first).to_s
|
assert_equal "0x5eb4" , Position.get(@machine.cpu_init.first).to_s
|
||||||
end
|
|
||||||
def pest_cpu_bin
|
|
||||||
assert_equal "0x5ecc" , Position.get(@machine.cpu_init).to_s
|
|
||||||
end
|
end
|
||||||
def test_cpu_label
|
def test_cpu_label
|
||||||
assert_equal Position , Position.get(@machine.cpu_init.first).class
|
assert_equal Position , Position.get(@machine.cpu_init.first).class
|
||||||
@ -61,5 +70,11 @@ module Risc
|
|||||||
bin = Parfait.object_space.get_init.binary.next
|
bin = Parfait.object_space.get_init.binary.next
|
||||||
assert 0 != bin.get_word(0) , "index 0 is 0 #{bin.inspect}"
|
assert 0 != bin.get_word(0) , "index 0 is 0 #{bin.inspect}"
|
||||||
end
|
end
|
||||||
|
def test_positions_set
|
||||||
|
@machine.objects.each do |id,obj|
|
||||||
|
assert Position.get(obj).valid? , "#{Position.get(obj)} , #{obj.object_id.to_s(16)}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user