fix and test assign
also fix field def and externalized assign
This commit is contained in:
parent
0179638f0e
commit
6f856315d7
@ -56,6 +56,7 @@ module Phisol
|
|||||||
end
|
end
|
||||||
|
|
||||||
require_relative "ast_helper"
|
require_relative "ast_helper"
|
||||||
|
require_relative "compiler/assignment"
|
||||||
require_relative "compiler/basic_values"
|
require_relative "compiler/basic_values"
|
||||||
require_relative "compiler/call_site"
|
require_relative "compiler/call_site"
|
||||||
require_relative "compiler/class_field"
|
require_relative "compiler/class_field"
|
||||||
|
31
lib/phisol/compiler/assignment.rb
Normal file
31
lib/phisol/compiler/assignment.rb
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
module Phisol
|
||||||
|
Compiler.class_eval do
|
||||||
|
|
||||||
|
def on_assignment statement
|
||||||
|
reset_regs # statements reset registers, ie have all at their disposal
|
||||||
|
puts statement.inspect
|
||||||
|
name , value = *statement
|
||||||
|
name = name.to_a.first
|
||||||
|
v = process(value)
|
||||||
|
raise "Not register #{v}" unless v.is_a?(Register::RegisterValue)
|
||||||
|
code = nil
|
||||||
|
if( index = @method.has_arg(name))
|
||||||
|
# TODO, check type @method.arguments[index].type
|
||||||
|
code = Register.set_slot(statement , v , :message , index )
|
||||||
|
else # or a local so it is in the frame
|
||||||
|
index = @method.has_local( name )
|
||||||
|
if(index)
|
||||||
|
# TODO, check type @method.locals[index].type
|
||||||
|
code = Register.set_slot(statement , v , :frame , index )
|
||||||
|
end
|
||||||
|
end
|
||||||
|
if( code )
|
||||||
|
puts "addin code #{code}"
|
||||||
|
@method.source.add_code code
|
||||||
|
else
|
||||||
|
raise "must define variable #{name} before using it in #{@method.inspect}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
@ -1,16 +1,14 @@
|
|||||||
module Phisol
|
module Phisol
|
||||||
Compiler.class_eval do
|
Compiler.class_eval do
|
||||||
|
include AST::Sexp
|
||||||
|
|
||||||
def on_field_def statement
|
def on_field_def statement
|
||||||
#puts statement.inspect
|
reset_regs # field_def is a statement, no return and all regs
|
||||||
|
puts statement.inspect
|
||||||
type , name , value = *statement
|
type , name , value = *statement
|
||||||
|
@method.ensure_local( name, type ) unless( @method.has_arg(name))
|
||||||
index = @method.ensure_local( name , type )
|
# if there is a value assigned, process it as am assignemnt statement (kind of call on_assign)
|
||||||
|
process( s(:assignment , s(:name , name) , value ) ) if value
|
||||||
if value
|
|
||||||
value = process( value )
|
|
||||||
end
|
|
||||||
# field_def is a statement, no return
|
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -14,24 +14,5 @@ module Phisol
|
|||||||
@method.source.add_code Register::OperatorInstruction.new(statement,operator,left_reg,right_reg)
|
@method.source.add_code Register::OperatorInstruction.new(statement,operator,left_reg,right_reg)
|
||||||
return left_reg # though this has wrong value attached
|
return left_reg # though this has wrong value attached
|
||||||
end
|
end
|
||||||
|
|
||||||
def on_assignment statement
|
|
||||||
puts statement.inspect
|
|
||||||
name , value = *statement
|
|
||||||
name = name.to_a.first
|
|
||||||
v = process(value)
|
|
||||||
index = @method.has_local( name )
|
|
||||||
if(index)
|
|
||||||
@method.source.add_code Virtual::Set.new(Virtual::FrameSlot.new(index, :int ) , v )
|
|
||||||
else
|
|
||||||
index = @method.has_arg( name )
|
|
||||||
if(index)
|
|
||||||
@method.source.add_code Virtual::Set.new(Virtual::ArgSlot.new(index , :int ) , v )
|
|
||||||
else
|
|
||||||
raise "must define variable #{name} before using it in #{@method.inspect}"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -4,7 +4,8 @@ require_relative '../../helper'
|
|||||||
module Statements
|
module Statements
|
||||||
|
|
||||||
def check
|
def check
|
||||||
machine = Virtual.machine.boot
|
machine = Virtual.machine
|
||||||
|
machine.boot unless machine.booted
|
||||||
machine.parse_and_compile @string_input
|
machine.parse_and_compile @string_input
|
||||||
produced = Virtual.machine.space.get_main.source
|
produced = Virtual.machine.space.get_main.source
|
||||||
assert @expect , "No output given"
|
assert @expect , "No output given"
|
||||||
|
87
test/compiler/statements/test_assign.rb
Normal file
87
test/compiler/statements/test_assign.rb
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
require_relative 'helper'
|
||||||
|
|
||||||
|
module Register
|
||||||
|
class TestAssignStatement < MiniTest::Test
|
||||||
|
include Statements
|
||||||
|
|
||||||
|
def setup
|
||||||
|
Virtual.machine.boot
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_arg
|
||||||
|
Virtual.machine.space.get_main.arguments.push Parfait::Variable.new(:Integer , :bar)
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main(int bar)
|
||||||
|
bar = 5
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,LoadConstant,SetSlot] , [Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_int
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main()
|
||||||
|
int r = 5
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,LoadConstant,SetSlot] , [Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_op
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main()
|
||||||
|
int n = 10 + 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,LoadConstant,LoadConstant,
|
||||||
|
OperatorInstruction,SetSlot],[Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_local
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main()
|
||||||
|
int runner
|
||||||
|
runner = 5
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,LoadConstant,SetSlot] , [Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_local_assign
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main()
|
||||||
|
int runner = 5
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,LoadConstant, SetSlot] , [Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_assign_call
|
||||||
|
@string_input = <<HERE
|
||||||
|
class Object
|
||||||
|
int main()
|
||||||
|
int r = main()
|
||||||
|
end
|
||||||
|
end
|
||||||
|
HERE
|
||||||
|
@expect = [[Virtual::MethodEnter,GetSlot,SetSlot, LoadConstant,
|
||||||
|
SetSlot,Virtual::MethodCall,GetSlot,SetSlot] , [Virtual::MethodReturn]]
|
||||||
|
check
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue
Block a user