2018-06-02 20:59:41 +02:00
|
|
|
require "util/eventable"
|
|
|
|
|
2018-05-05 18:47:18 +02:00
|
|
|
module Risc
|
|
|
|
# Positions are very different during compilation and run-time.
|
|
|
|
# At run-time they are inherrent to the object, and fixed.
|
|
|
|
# While during compilation we can move things about, and do not use the
|
|
|
|
# objects memory position at all.
|
|
|
|
#
|
|
|
|
# Furthermore, there are differnet kind of positions during compilation.
|
|
|
|
# Off course the object position as hinted above, but also instruction
|
|
|
|
# positions, that do not reflect the position of the object, but of the
|
|
|
|
# assembled instruction in the binary.
|
|
|
|
#
|
2018-05-10 19:56:12 +02:00
|
|
|
# The Position module keeps a hash of all compile time positions.
|
2018-05-05 18:47:18 +02:00
|
|
|
#
|
2018-05-10 19:56:12 +02:00
|
|
|
# While the (different)Position objects transmit the change that (re) positioning
|
2018-05-05 18:47:18 +02:00
|
|
|
# entails to affected objects.
|
|
|
|
|
2018-06-02 20:59:41 +02:00
|
|
|
class Position
|
2018-05-20 14:52:13 +02:00
|
|
|
include Util::Logging
|
2018-05-24 13:27:53 +02:00
|
|
|
log_level :info
|
2018-05-20 14:52:13 +02:00
|
|
|
|
2018-06-02 20:59:41 +02:00
|
|
|
include Util::Eventable
|
|
|
|
|
|
|
|
attr_reader :at , :object
|
|
|
|
|
|
|
|
# initialize with a given object, first parameter
|
|
|
|
# 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 = pos
|
|
|
|
@object = object
|
|
|
|
Position.set_to(self , pos)
|
|
|
|
end
|
|
|
|
|
2018-06-02 22:02:59 +02:00
|
|
|
# utility to register events of type :position_changed
|
|
|
|
# can give an object and a PositionListener will be created for it
|
|
|
|
def position_listener(listener)
|
|
|
|
unless listener.class.name.include?("Listener")
|
|
|
|
listener = PositionListener.new(listener)
|
|
|
|
end
|
|
|
|
register_event(:position_changed , listener)
|
|
|
|
end
|
|
|
|
|
2018-06-06 09:00:07 +02:00
|
|
|
# When instruction get inserted, we have to move listeners around, remove given
|
|
|
|
def remove_position_listener(list)
|
|
|
|
unregister_event(:position_changed, list)
|
|
|
|
end
|
|
|
|
|
2018-06-02 22:02:59 +02:00
|
|
|
# utility to get all registered listeners to the :position_changed event
|
|
|
|
# returns an array
|
|
|
|
def position_listeners
|
|
|
|
event_table[:position_changed]
|
|
|
|
end
|
2018-06-02 22:48:12 +02:00
|
|
|
|
2018-06-02 20:59:41 +02:00
|
|
|
#look for InstructionListener and return its code if found
|
|
|
|
def get_code
|
|
|
|
listener = event_table.find{|one| one.class == InstructionListener}
|
|
|
|
return nil unless listener
|
|
|
|
listener.code
|
|
|
|
end
|
|
|
|
|
2018-06-06 00:16:00 +02:00
|
|
|
def valid?
|
|
|
|
@at != -1
|
|
|
|
end
|
|
|
|
|
2018-06-02 22:48:12 +02:00
|
|
|
def set(int)
|
2018-06-05 17:11:25 +02:00
|
|
|
return int if int == self.at
|
2018-06-02 22:48:12 +02:00
|
|
|
Position.set_to(self , int)
|
|
|
|
@at = int
|
2018-06-06 09:19:18 +02:00
|
|
|
trigger_changed
|
2018-06-02 22:48:12 +02:00
|
|
|
int
|
|
|
|
end
|
|
|
|
|
2018-06-06 09:19:18 +02:00
|
|
|
# helper to fire the event that the position changed
|
|
|
|
# Note: set checks if the position actually has changed, before fireing
|
|
|
|
# but during insert it is helpful to trigger just to set the next
|
|
|
|
def trigger_changed
|
|
|
|
trigger(:position_changed , self )
|
|
|
|
end
|
|
|
|
|
2018-06-06 09:00:07 +02:00
|
|
|
def trigger_inserted
|
|
|
|
event_table[:position_changed].each { |handler| handler.position_inserted( self) }
|
|
|
|
end
|
|
|
|
|
2018-06-02 20:59:41 +02:00
|
|
|
def +(offset)
|
|
|
|
offset = offset.at if offset.is_a?(Position)
|
|
|
|
@at + offset
|
|
|
|
end
|
|
|
|
|
|
|
|
def -(offset)
|
|
|
|
offset = offset.at if offset.is_a?(Position)
|
|
|
|
@at - offset
|
|
|
|
end
|
2018-06-05 23:53:41 +02:00
|
|
|
|
|
|
|
def <(right)
|
|
|
|
right = right.at if right.is_a?(Position)
|
|
|
|
@at < right
|
|
|
|
end
|
|
|
|
|
|
|
|
def >(right)
|
|
|
|
right = right.at if right.is_a?(Position)
|
|
|
|
@at > right
|
|
|
|
end
|
|
|
|
|
2018-06-02 20:59:41 +02:00
|
|
|
def to_s
|
|
|
|
"0x#{@at.to_s(16)}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def next_slot
|
|
|
|
return -1 if at < 0
|
2018-06-05 17:11:25 +02:00
|
|
|
self.log.debug "Next Slot @#{at.to_s(16)} for #{object.class} == #{(at + object.byte_length).to_s(16)}"
|
2018-06-02 20:59:41 +02:00
|
|
|
at + object.byte_length
|
|
|
|
end
|
|
|
|
|
|
|
|
## class level forward and reverse cache
|
2018-05-05 18:47:18 +02:00
|
|
|
@positions = {}
|
2018-05-24 13:27:53 +02:00
|
|
|
@reverse_cache = {}
|
2018-05-05 18:47:18 +02:00
|
|
|
|
|
|
|
def self.positions
|
|
|
|
@positions
|
|
|
|
end
|
|
|
|
|
2018-05-24 13:27:53 +02:00
|
|
|
def self.clear_positions
|
|
|
|
@positions = {}
|
|
|
|
@reverse_cache = {}
|
|
|
|
end
|
2018-05-24 15:26:56 +02:00
|
|
|
|
2018-05-17 19:13:33 +02:00
|
|
|
def self.at( int )
|
2018-05-24 13:27:53 +02:00
|
|
|
@reverse_cache[int]
|
2018-05-17 19:13:33 +02:00
|
|
|
end
|
|
|
|
|
2018-05-06 19:04:02 +02:00
|
|
|
def self.set?(object)
|
|
|
|
self.positions.has_key?(object)
|
|
|
|
end
|
|
|
|
|
2018-06-05 17:11:25 +02:00
|
|
|
# get a position from the cache (object -> position)
|
|
|
|
# unless it's a label, then get the position of it's next
|
2018-05-06 19:04:02 +02:00
|
|
|
def self.get(object)
|
2018-05-05 18:47:18 +02:00
|
|
|
pos = self.positions[object]
|
|
|
|
if pos == nil
|
2018-06-02 20:59:41 +02:00
|
|
|
str = "position accessed but not initialized, "
|
2018-05-05 18:47:18 +02:00
|
|
|
str += "0x#{object.object_id.to_s(16)}\n"
|
2018-05-30 09:29:38 +02:00
|
|
|
str += "for #{object.class} "
|
|
|
|
str += "byte_length #{object.byte_length}" if object.respond_to?(:byte_length)
|
|
|
|
str += " for #{object.to_s[0...130]}"
|
2018-05-05 18:47:18 +02:00
|
|
|
raise str
|
|
|
|
end
|
|
|
|
pos
|
|
|
|
end
|
|
|
|
|
2018-06-05 17:11:25 +02:00
|
|
|
# populate the position caches (forward and revese) with the given position
|
|
|
|
# forward caches object -> position
|
|
|
|
# reverse caches position.at > position
|
|
|
|
# Labels do not participatein reverse cache
|
2018-06-02 15:12:01 +02:00
|
|
|
def self.set_to( position , to)
|
2018-06-02 20:59:41 +02:00
|
|
|
postest = Position.positions[position.object] unless to < 0
|
2018-06-02 15:12:01 +02:00
|
|
|
raise "Mismatch #{position}" if postest and postest != position
|
2018-06-05 17:11:25 +02:00
|
|
|
@reverse_cache.delete(position.at) unless position.object.is_a?(Label)
|
2018-06-02 20:59:41 +02:00
|
|
|
testing = self.at( position.at ) unless position.at < 0
|
2018-06-05 17:11:25 +02:00
|
|
|
if testing and testing.object.class != position.object.class
|
2018-06-05 23:53:41 +02:00
|
|
|
raise "Mismatch (at #{to.to_s(16)}) was:#{position} #{position.class} #{position.object} , should #{testing}#{testing.class}"
|
2018-05-31 18:01:10 +02:00
|
|
|
end
|
2018-06-02 15:12:01 +02:00
|
|
|
self.positions[position.object] = position
|
2018-06-05 17:11:25 +02:00
|
|
|
@reverse_cache[to] = position unless position.object.is_a?(Label)
|
|
|
|
log.debug "Set #{position} (#{to.to_s(16)}) for #{position.object.class} #{position.object.object_id.to_s(16)}"
|
2018-05-07 21:30:43 +02:00
|
|
|
position
|
2018-05-05 23:34:59 +02:00
|
|
|
end
|
|
|
|
end
|
2018-05-05 18:47:18 +02:00
|
|
|
end
|
2018-06-02 22:48:12 +02:00
|
|
|
require_relative "position_listener"
|
2018-06-02 20:20:15 +02:00
|
|
|
require_relative "instruction_listener"
|
2018-06-02 15:34:44 +02:00
|
|
|
require_relative "code_listener"
|