2020-03-18 12:10:09 +01: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-18 16:50:22 +01:00
|
|
|
@release_points = Hash.new( [] )
|
2020-03-19 11:30:44 +01:00
|
|
|
@reg_names = (0 ... platform.num_registers).collect{|i| "r#{i}".to_sym }
|
2020-03-18 14:27:40 +01: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 11:30:44 +01:00
|
|
|
walk_and_mark(@compiler.risc_instructions)
|
|
|
|
pointer = @compiler.risc_instructions
|
|
|
|
while(pointer)
|
|
|
|
names = assign(pointer)
|
|
|
|
names.each {|name| release_reg(name)}
|
|
|
|
pointer = pointer.next
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def assign(instruction)
|
|
|
|
names = instruction.register_names
|
|
|
|
names.each do |for_name|
|
|
|
|
new_reg = get_reg(for_name)
|
|
|
|
# swap name out
|
|
|
|
end
|
|
|
|
names
|
2020-03-18 16:50:22 +01: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 11:30:44 +01:00
|
|
|
#
|
2020-03-18 16:50:22 +01: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
|
|
|
|
#puts instruction.class.name
|
|
|
|
instruction.register_names.each do |name|
|
|
|
|
next if released.include?(name)
|
|
|
|
@release_points[instruction] << name
|
|
|
|
released << name
|
|
|
|
end
|
|
|
|
released
|
2020-03-18 12:10:09 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def used_regs_empty?
|
|
|
|
@used_regs.empty?
|
|
|
|
end
|
|
|
|
|
2020-03-19 11:30:44 +01:00
|
|
|
def use_reg(reg , for_name)
|
|
|
|
reg = reg.symbol if reg.is_a?(RegisterValue)
|
|
|
|
raise "Stupid error #{reg}" unless reg.is_a?(Symbol)
|
|
|
|
puts "Using #{reg} for #{for_name}"
|
|
|
|
@used_regs[reg] = for_name
|
2020-03-18 12:10:09 +01:00
|
|
|
end
|
|
|
|
|
2020-03-19 11:30:44 +01:00
|
|
|
# if a register has been assigned to the given name, return that
|
2020-03-18 12:10:09 +01:00
|
|
|
#
|
2020-03-19 11:30:44 +01:00
|
|
|
# otherwise find the first free register by going through the available names
|
|
|
|
# and checking if it is used
|
|
|
|
def get_reg(for_name)
|
|
|
|
@used_regs.each {|reg,name| return reg if for_name == name }
|
|
|
|
@reg_names.each do |name|
|
|
|
|
return use_reg(name , for_name) unless @used_regs.has_key?(name)
|
|
|
|
end
|
|
|
|
raise "No more registers #{self}"
|
2020-03-18 12:10:09 +01: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)
|
|
|
|
@used_regs.delete(reg)
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
end
|