better calcite and operator to expand the interpreter test

This commit is contained in:
Torsten Ruger 2015-10-07 10:05:34 +03:00
parent af6366f2d1
commit 83ef902b55
6 changed files with 97 additions and 24 deletions

View File

@ -8,7 +8,11 @@ module Bosl
if receiver if receiver
me = process( receiver.to_a.first ) me = process( receiver.to_a.first )
else else
if @method.class.name == :Integer
me = Virtual::Self.new :int me = Virtual::Self.new :int
else
me = Virtual::Self.new :ref
end
end end
## need two step process, compile and save to frame ## need two step process, compile and save to frame
# then move from frame to new message # then move from frame to new message
@ -39,17 +43,11 @@ module Bosl
raise "Method not implemented #{me.class}.#{code.name}" unless method raise "Method not implemented #{me.class}.#{code.name}" unless method
@method.source.add_code Virtual::MethodCall.new( method ) @method.source.add_code Virtual::MethodCall.new( method )
elsif( me.is_a? Fixnum ) elsif( me.is_a? Fixnum )
name = :plus if name == :+
method = Virtual.machine.space.get_class_by_name(:Integer).get_instance_method(name) method = Virtual.machine.space.get_class_by_name(:Integer).get_instance_method(name)
puts Virtual.machine.space.get_class_by_name(:Integer).method_names.to_a #puts Virtual.machine.space.get_class_by_name(:Integer).method_names.to_a
raise "Method not implemented Integer.#{name}" unless method raise "Method not implemented Integer.#{name}" unless method
@method.source.add_code Virtual::MethodCall.new( method ) @method.source.add_code Virtual::MethodCall.new( method )
else else
# note: this is the current view: call internal send, even the method name says else
# but send is "special" and accesses the internal method name and resolves.
kernel = Virtual.machine.space.get_class_by_name(:Kernel)
method = kernel.get_instance_method(:__send)
@method.source.add_code Virtual::MethodCall.new( method )
raise "unimplemented: \n#{code} \nfor #{ref.inspect}" raise "unimplemented: \n#{code} \nfor #{ref.inspect}"
end end
else else
@ -60,7 +58,9 @@ module Bosl
raise "Method not implemented Integer.#{name}" unless method raise "Method not implemented Integer.#{name}" unless method
@method.source.add_code Virtual::MethodCall.new( method ) @method.source.add_code Virtual::MethodCall.new( method )
else else
raise "me #{me}" method = @clazz.get_instance_method(name)
raise "Method not implemented Integer.#{name}" unless method
@method.source.add_code Virtual::MethodCall.new( method )
end end
end end
raise "Method not implemented #{me.value}.#{name}" unless method raise "Method not implemented #{me.value}.#{name}" unless method

View File

@ -1,10 +1,25 @@
module Bosl module Bosl
Compiler.class_eval do Compiler.class_eval do
# operator attr_reader :operator, :left, :right
def on_operator expression def on_operator expression
operator , left , right = *expression puts "operator #{expression.inspect}"
#raise "not quite there" operator , left_e , right_e = *expression
Virtual::Return.new(:int) left_slot = process(left_e)
right_slot = process(right_e)
puts "left #{left_slot}"
puts "right #{right_slot}"
tmp1 = Register.tmp_reg
tmp2 = tmp1.next_reg_use
get = Register.get_slot_to(expression , left_slot , tmp1 )
get2 = Register.get_slot_to(expression , right_slot , tmp2 )
puts "GET #{get}"
puts "GET2 #{get2}"
@method.source.add_code get
@method.source.add_code get2
@method.source.add_code Register::OperatorInstruction.new(expression,operator, tmp1,tmp2)
Virtual::Return.new(:int )
end end
def on_assign expression def on_assign expression

View File

@ -101,6 +101,13 @@ module Interpreter
false false
end end
def execute_IsZeroBranch
puts @instruction.inspect
target = @instruction.block
set_block target
false
end
def execute_LoadConstant def execute_LoadConstant
to = @instruction.register to = @instruction.register
value = @instruction.constant value = @instruction.constant
@ -173,18 +180,25 @@ module Interpreter
end end
def execute_OperatorInstruction def execute_OperatorInstruction
case @instruction.operator
when :add
left = get_register(@instruction.left) left = get_register(@instruction.left)
rr = @instruction.right rr = @instruction.right
right = get_register(rr) right = get_register(rr)
case @instruction.operator
when :add
result = left + right result = left + right
when "/"
result = left / right
when "-"
result = left - right
when "<"
result = left < right
when "=="
result = left == right
else
raise "unimplemented #{@instruction.operator} #{@instruction}"
end
puts "#{@instruction} == #{result}" puts "#{@instruction} == #{result}"
right = set_register(rr , result) right = set_register(rr , result)
else
raise "unimplemented operator #{@instruction}"
end
true true
end end
end end

View File

@ -45,4 +45,25 @@ module Register
GetSlot.new( source , array , index , to) GetSlot.new( source , array , index , to)
end end
def self.get_slot_to source , slot , to
array = nil
index = nil
case slot
when Virtual::Self
array = :message
index = :receiver
when Virtual::Return
array = :message
index = :return_value
when Virtual::FrameSlot
array = :frame
index = slot.index
when Virtual::ArgSlot
array = :message
index = slot.index
else
raise "not done #{slot}"
end
get_slot( source , array , index , to)
end
end end

View File

@ -1,2 +1,3 @@
require_relative "test_puts" require_relative "test_puts"
require_relative "test_puti"
require_relative "test_add" require_relative "test_add"

View File

@ -7,6 +7,23 @@ class AddTest < MiniTest::Test
def test_puti def test_puti
@string_input = <<HERE @string_input = <<HERE
class Integer < Object class Integer < Object
ref digit( int rest )
if( rest == 5 )
return "5"
end
if( rest == 1 )
return "1"
end
if( rest == 2 )
return "2"
end
if( rest == 3 )
return "3"
end
if( rest == 4 )
return "4"
end
end
ref add_string(ref str) ref add_string(ref str)
int div int div
div = self / 10 div = self / 10
@ -48,10 +65,15 @@ HERE
"RegisterTransfer" ,"GetSlot" , "FunctionCall" ,"SaveReturn" , "GetSlot" , "RegisterTransfer" ,"GetSlot" , "FunctionCall" ,"SaveReturn" , "GetSlot" ,
"LoadConstant", "SetSlot", "GetSlot" , "GetSlot" , "SetSlot" , "LoadConstant", "SetSlot", "GetSlot" , "GetSlot" , "SetSlot" ,
"LoadConstant", "SetSlot" , "GetSlot" , "SetSlot" , "RegisterTransfer", "LoadConstant", "SetSlot" , "GetSlot" , "SetSlot" , "RegisterTransfer",
"GetSlot", "FunctionCall", "SaveReturn", "GetSlot", "GetSlot", "GetSlot", "FunctionCall", "SaveReturn", "GetSlot", "LoadConstant",
"SetSlot", "GetSlot", "SetSlot", "FunctionCall", "FunctionCall", "SetSlot", "GetSlot", "GetSlot", "OperatorInstruction", "GetSlot",
"FunctionCall", "FunctionCall", "FunctionCall", "FunctionCall", "FunctionCall", "SetSlot", "GetSlot", "GetSlot", "OperatorInstruction", "GetSlot",
"NilClass"].each_with_index do |name , index| "SetSlot", "LoadConstant", "SetSlot", "GetSlot", "GetSlot",
"OperatorInstruction", "Branch", "GetSlot", "GetSlot", "SetSlot",
"LoadConstant", "SetSlot", "GetSlot", "SetSlot", "RegisterTransfer",
"GetSlot", "FunctionCall", "SaveReturn", "LoadConstant", "SetSlot",
"GetSlot", "GetSlot", "OperatorInstruction", "Branch", "LoadConstant",
"SetSlot"].each_with_index do |name , index|
got = ticks(1) got = ticks(1)
puts got puts got
assert got.class.name.index(name) , "Wrong class for #{index+1}, expect #{name} , got #{got}" assert got.class.name.index(name) , "Wrong class for #{index+1}, expect #{name} , got #{got}"