2014-05-03 14:13:44 +02:00
|
|
|
require_relative "block"
|
|
|
|
|
|
|
|
module Vm
|
|
|
|
|
|
|
|
# Functions are similar to Blocks. Where Blocks can be jumped to, Functions can be called.
|
|
|
|
|
2014-05-19 10:28:13 +02:00
|
|
|
# Functions also have arguments and a return. These are Value subclass instances, ie specify
|
|
|
|
# type (by class type) and register by instance
|
|
|
|
|
2014-05-06 20:36:28 +02:00
|
|
|
# Functions have a exactly three blocks, entry, exit and body, which are created for you
|
|
|
|
# with straight branches between them.
|
|
|
|
|
2014-05-19 10:28:13 +02:00
|
|
|
# Also remember that if your body exists of several blocks, they must be wrapped in a
|
2014-05-06 20:36:28 +02:00
|
|
|
# block as the function really only has the one, and blocks only assemble their codes,
|
|
|
|
# not their next links
|
|
|
|
# This comes at zero runtime cost though, as the wrapper is just the sum of it's codes
|
2014-05-03 14:13:44 +02:00
|
|
|
|
2014-05-06 20:36:28 +02:00
|
|
|
# If you change the body block to point elsewhere, remember to end up at exit
|
2014-05-03 14:13:44 +02:00
|
|
|
|
2014-05-06 20:36:28 +02:00
|
|
|
class Function < Code
|
2014-05-03 14:13:44 +02:00
|
|
|
|
2014-05-19 10:28:13 +02:00
|
|
|
def initialize(name , args = [] , return_type = nil)
|
2014-05-06 20:36:28 +02:00
|
|
|
super()
|
|
|
|
@name = name
|
2014-05-19 11:18:01 +02:00
|
|
|
@args = Array.new(args.length)
|
|
|
|
args.each_with_index do |arg , i|
|
|
|
|
if arg.is_a?(Value)
|
|
|
|
@args[i] = arg
|
|
|
|
raise "arg in non std register #{arg.inspect}" unless i == arg.register
|
|
|
|
else
|
|
|
|
@args[i] = arg.new(i)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
@return_type = return_type || Vm::Integer
|
|
|
|
if @return_type.is_a?(Value)
|
|
|
|
raise "return in non std register #{@return_type.inspect}" unless 0 == @return_type.register
|
|
|
|
else
|
|
|
|
@return_type = @return_type.new(0)
|
|
|
|
end
|
|
|
|
@entry = Core::Kernel::function_entry( Vm::Block.new("#{name}_entry") ,name )
|
|
|
|
@exit = Core::Kernel::function_exit( Vm::Block.new("#{name}_exit") , name )
|
|
|
|
@body = Block.new("#{name}_body")
|
|
|
|
@reg_count = 0
|
|
|
|
branch_body
|
|
|
|
@entry = Core::Kernel::function_entry( Vm::Block.new("#{name}_entry") ,name )
|
|
|
|
@exit = Core::Kernel::function_exit( Vm::Block.new("#{name}_exit") , name )
|
|
|
|
@body = Block.new("#{name}_body")
|
|
|
|
@reg_count = 0
|
|
|
|
branch_body
|
2014-05-03 14:13:44 +02:00
|
|
|
end
|
2014-05-06 20:36:28 +02:00
|
|
|
attr_reader :args , :entry , :exit , :body , :name
|
2014-05-13 20:06:12 +02:00
|
|
|
attr_accessor :return_type
|
2014-05-06 20:36:28 +02:00
|
|
|
|
2014-05-03 14:13:44 +02:00
|
|
|
def arity
|
|
|
|
@args.length
|
|
|
|
end
|
|
|
|
|
2014-05-13 17:21:24 +02:00
|
|
|
def next_register
|
|
|
|
next_free = @reg_count
|
|
|
|
@reg_count += 1
|
|
|
|
next_free + args.length
|
|
|
|
end
|
|
|
|
|
2014-05-05 23:12:04 +02:00
|
|
|
def link_at address , context
|
2014-05-06 20:36:28 +02:00
|
|
|
raise "undefined code #{inspect}" if @body.nil?
|
|
|
|
super #just sets the position
|
2014-05-05 23:12:04 +02:00
|
|
|
@entry.link_at address , context
|
|
|
|
address += @entry.length
|
2014-05-06 20:36:28 +02:00
|
|
|
@body.link_at(address , context)
|
2014-05-14 21:04:03 +02:00
|
|
|
address += @body.length
|
2014-05-05 23:12:04 +02:00
|
|
|
@exit.link_at(address,context)
|
|
|
|
end
|
2014-05-16 18:56:13 +02:00
|
|
|
def position
|
|
|
|
@entry.position
|
|
|
|
end
|
2014-05-05 08:35:40 +02:00
|
|
|
def length
|
2014-05-06 20:36:28 +02:00
|
|
|
@entry.length + @exit.length + @body.length
|
2014-05-05 08:35:40 +02:00
|
|
|
end
|
|
|
|
|
2014-05-05 23:12:04 +02:00
|
|
|
def assemble io
|
2014-05-06 20:36:28 +02:00
|
|
|
@entry.assemble(io)
|
|
|
|
@body.assemble(io)
|
2014-05-05 23:12:04 +02:00
|
|
|
@exit.assemble(io)
|
2014-05-03 14:13:44 +02:00
|
|
|
end
|
|
|
|
|
2014-05-06 20:36:28 +02:00
|
|
|
private
|
|
|
|
# set up the braches from entry to body and body to exit (unless that exists, see set_body)
|
|
|
|
def branch_body
|
|
|
|
@entry.set_next(@body)
|
|
|
|
@body.set_next(@exit) if @body and !@body.next
|
|
|
|
end
|
2014-05-03 14:13:44 +02:00
|
|
|
end
|
|
|
|
end
|