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
|
2014-07-18 11:56:46 +03:00
|
|
|
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 = []
|
2014-07-12 21:59:17 +03:00
|
|
|
@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
|
2014-07-14 14:29:33 +03:00
|
|
|
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
|
|
|
|
2014-07-18 11:56:46 +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
|
2014-07-18 11:56:46 +03:00
|
|
|
@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
|