2020-03-18 13:10:09 +02:00
|
|
|
module Risc
|
|
|
|
#
|
|
|
|
# StandardAllocator keeps a list of registers whole names it accuires from the
|
|
|
|
# Plaform.
|
|
|
|
#
|
|
|
|
# It allocates registers by going throught the instructions and dishing
|
|
|
|
# out sequential registers.
|
|
|
|
# Before doing that it determines live ranges, so registers can be returned
|
|
|
|
# when not used anymore, and thus reused.
|
|
|
|
class StandardAllocator
|
|
|
|
|
|
|
|
def initialize( compiler , platform)
|
|
|
|
@compiler = compiler
|
|
|
|
@platform = platform
|
|
|
|
@used_regs = {}
|
2020-03-19 14:10:39 +02:00
|
|
|
@release_points = Hash.new {|hash , key | hash[key] = [] }
|
2020-03-19 18:18:22 +02:00
|
|
|
@reg_names = platform.register_names
|
2020-03-18 15:27:40 +02:00
|
|
|
end
|
|
|
|
attr_reader :used_regs , :compiler , :platform , :reg_names
|
|
|
|
|
|
|
|
# main entry point and the whole reason for the class.
|
|
|
|
# Allocate regs by changing the names of compiler instructions to
|
|
|
|
# register names according to platform.
|
|
|
|
# Ie on arm register names are r0 .. . r15 , so it keeps a list of unused
|
|
|
|
# regs and frees regs according to live ranges
|
|
|
|
def allocate_regs
|
2020-03-19 12:30:44 +02:00
|
|
|
walk_and_mark(@compiler.risc_instructions)
|
|
|
|
pointer = @compiler.risc_instructions
|
|
|
|
while(pointer)
|
|
|
|
names = assign(pointer)
|
2020-03-19 14:10:39 +02:00
|
|
|
names.each {|name| release_after(pointer, name)}
|
2020-03-19 12:30:44 +02:00
|
|
|
pointer = pointer.next
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-03-19 14:10:39 +02:00
|
|
|
# if the instruction has a release point with the given name, release it
|
|
|
|
# Release points store the last use of a register (in ssa)
|
|
|
|
# This method is called after machine registers have been assigned,
|
|
|
|
# and give us the chance to reclaim any unsued machine regs
|
|
|
|
# (via the precalculated release_points)
|
|
|
|
def release_after(instruction , ssa_name)
|
2020-03-20 16:15:11 +02:00
|
|
|
#puts "release request for #{ssa_name}"
|
2020-03-19 14:10:39 +02:00
|
|
|
release = @release_points[instruction]
|
|
|
|
return unless release
|
|
|
|
return unless release.include?(ssa_name)
|
|
|
|
#puts "ReleasePoint #{ssa_name} for #{instruction} #{release}"
|
|
|
|
pois = reverse_used(ssa_name)
|
|
|
|
release_reg( pois ) if pois
|
|
|
|
end
|
|
|
|
|
2020-03-19 12:30:44 +02:00
|
|
|
def assign(instruction)
|
|
|
|
names = instruction.register_names
|
2020-03-19 18:18:22 +02:00
|
|
|
#puts "AT #{instruction}"
|
|
|
|
names.each do |ssa_name|
|
2020-03-20 16:15:11 +02:00
|
|
|
# BUG: clearly we do NOT do ssa, because some of the instances of
|
|
|
|
# RegisterValue are the same. This causes the symbols/names to
|
|
|
|
# have changed through the previous assign. Hence without next line check
|
|
|
|
# we assign risc regs to risc reg names. Easily avoided, but not clean
|
|
|
|
next if @reg_names.include?(ssa_name)
|
2020-03-19 18:18:22 +02:00
|
|
|
new_reg = get_reg(ssa_name)
|
2020-03-20 16:15:11 +02:00
|
|
|
instruction.set_registers( ssa_name , new_reg)
|
2020-03-19 18:18:22 +02:00
|
|
|
#puts "Assign #{new_reg} for #{ssa_name}"
|
2020-03-19 12:30:44 +02:00
|
|
|
end
|
|
|
|
names
|
2020-03-18 17:50:22 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
# determines when registers can be freed
|
|
|
|
#
|
|
|
|
# this is done by walking the instructions backwards and saving the first
|
|
|
|
# occurence of a register name. (The last, as we walk backwards)
|
2020-03-19 12:30:44 +02:00
|
|
|
#
|
2020-03-18 17:50:22 +02:00
|
|
|
# First walk down, and on the way up mark register occurences, unless they
|
|
|
|
# have been marked already
|
|
|
|
def walk_and_mark(instruction)
|
|
|
|
released = []
|
|
|
|
released = walk_and_mark(instruction.next) if instruction.next
|
2020-03-19 14:10:39 +02:00
|
|
|
#puts "Walking #{instruction}"
|
2020-03-20 16:15:11 +02:00
|
|
|
instruction.register_names.each do |ssa_name|
|
|
|
|
next if released.include?(ssa_name)
|
|
|
|
@release_points[instruction] << ssa_name
|
|
|
|
#puts "ADDING #{ssa_name}"
|
|
|
|
released << ssa_name
|
2020-03-18 17:50:22 +02:00
|
|
|
end
|
|
|
|
released
|
2020-03-18 13:10:09 +02:00
|
|
|
end
|
|
|
|
|
2020-03-19 14:10:39 +02:00
|
|
|
# use the given reg (first) parameter and mark it as assigned to
|
|
|
|
# it's ssa form, the second parameter.
|
|
|
|
# forward check is trivial, and reverse_used provides reverse check
|
|
|
|
def use_reg(reg , ssa_name)
|
2020-03-19 12:30:44 +02:00
|
|
|
reg = reg.symbol if reg.is_a?(RegisterValue)
|
|
|
|
raise "Stupid error #{reg}" unless reg.is_a?(Symbol)
|
2020-03-19 14:10:39 +02:00
|
|
|
#puts "Using #{reg} for #{ssa_name}"
|
|
|
|
@used_regs[reg] = ssa_name
|
2020-03-19 18:18:22 +02:00
|
|
|
reg
|
2020-03-19 14:10:39 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
# Check whether a register has been assigned to the given ssa form given.
|
|
|
|
# Ie a reverse check on the used_regs hash
|
|
|
|
def reverse_used( ssa_name )
|
|
|
|
@used_regs.each {|reg,name| return reg if ssa_name == name }
|
|
|
|
return nil
|
2020-03-18 13:10:09 +02:00
|
|
|
end
|
|
|
|
|
2020-03-19 14:10:39 +02:00
|
|
|
# if a register has been assigned to the given ssa name, return that
|
2020-03-18 13:10:09 +02:00
|
|
|
#
|
2020-03-19 12:30:44 +02:00
|
|
|
# otherwise find the first free register by going through the available names
|
|
|
|
# and checking if it is used
|
2020-03-19 14:10:39 +02:00
|
|
|
def get_reg(ssa_name)
|
|
|
|
name = reverse_used( ssa_name )
|
|
|
|
return name if name
|
2020-03-19 18:18:22 +02:00
|
|
|
get_next_free(ssa_name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_next_free(ssa_name)
|
|
|
|
reg = platform.assign_reg?( ssa_name )
|
|
|
|
return use_reg(reg , ssa_name) if reg
|
2020-03-19 12:30:44 +02:00
|
|
|
@reg_names.each do |name|
|
2020-03-19 14:10:39 +02:00
|
|
|
return use_reg(name , ssa_name) unless @used_regs.has_key?(name)
|
2020-03-19 12:30:44 +02:00
|
|
|
end
|
|
|
|
raise "No more registers #{self}"
|
2020-03-18 13:10:09 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
# releasing a register (accuired by use_reg) makes it available for use again
|
|
|
|
# thus avoiding possibly using too many registers
|
|
|
|
def release_reg( reg )
|
|
|
|
reg = reg.symbol if reg.is_a?(RegisterValue)
|
|
|
|
raise "not symbol #{reg}:#{reg.class}" unless reg.is_a?(Symbol)
|
2020-03-19 14:10:39 +02:00
|
|
|
#puts "Releasing #{reg} "
|
2020-03-18 13:10:09 +02:00
|
|
|
@used_regs.delete(reg)
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|