rubyx/lib/virtual/method_definition.rb

83 lines
2.3 KiB
Ruby
Raw Normal View History

2014-07-14 11:29:38 +03:00
require_relative "object"
2014-07-14 14:06:09 +03:00
2014-07-10 17:14:38 +03:00
module Virtual
# static description of a method
# name
# args (with defaults)
# code
# return arg (usually mystery, but for coded ones can be more specific)
# known local variable names
# temp variables (numbered)
#
2014-07-16 19:24:41 +03:00
class MethodDefinition < Virtual::Object
2014-07-10 17:14:38 +03:00
#return the main function (the top level) into which code is compiled
2014-07-16 19:24:41 +03:00
def MethodDefinition.main
MethodDefinition.new(:main , [] , Virtual::SelfReference )
2014-07-10 17:14:38 +03:00
end
2014-07-14 11:29:38 +03:00
def attributes
2014-07-14 16:19:47 +03:00
[:name , :args , :receiver , :return_type , :start]
2014-07-14 11:29:38 +03:00
end
def initialize name , args , receiver = Virtual::SelfReference.new , return_type = Virtual::Mystery , start = MethodEnter.new(MethodReturn.new)
2014-07-14 11:29:38 +03:00
@name = name.to_sym
2014-07-10 17:14:38 +03:00
@args = args
@locals = []
2014-07-15 00:00:00 +03:00
@tmps = []
@receiver = receiver
@return_type = return_type
2014-07-14 16:19:47 +03:00
@start = start
2014-07-10 17:14:38 +03:00
@current = @start
end
attr_reader :name , :args , :receiver , :start
2014-07-16 20:16:10 +03:00
attr_accessor :return_type , :current
2014-07-10 17:14:38 +03:00
# add an instruction after the current (insertion point)
# the added instruction will become the new insertion point
2014-07-10 17:14:38 +03:00
def add instruction
2014-07-15 00:00:00 +03:00
raise instruction.inspect unless instruction.is_a? Instruction
@current.insert(instruction) #insert after current
2014-07-10 17:14:38 +03:00
@current = instruction
end
# determine whether this method has a variable by the given name
# variables are locals and and arguments
# used to determine if a send must be issued
def has_var name
2014-07-16 13:20:47 +03:00
name = name.to_sym
var = @args.find {|a| a.name == name }
var = @locals.find {|a| a.name == name } unless var
var = @tmps.find {|a| a.name == name } unless var
var
end
2014-07-25 20:28:38 +03:00
# determine whether this method has an argument by the name
def has_arg name
name = name.to_sym
var = @args.find {|a| a.name == name }
var
end
2014-07-16 13:20:47 +03:00
def set_var name , var
v = has_var name
if( v )
2014-07-27 10:09:31 +03:00
v.type = var
2014-07-16 13:20:47 +03:00
else
v = Local.new(name , var)
@locals << v
end
v
end
def get_var name
var = has_var name
raise "no var #{name} in method #{self.name} , #{@locals} #{@args}" unless var
2014-07-10 17:14:38 +03:00
var
end
2014-07-15 10:35:29 +03:00
2014-07-15 00:00:00 +03:00
def get_tmp
name = "__tmp__#{@tmps.length}"
@tmps << name
Ast::NameExpression.new(name)
end
2014-07-10 17:14:38 +03:00
end
end