More on slot assignment
which turn into slot_load
This commit is contained in:
parent
2c7944af85
commit
cbbb0c2f07
@ -19,4 +19,4 @@ exit_label
|
|||||||
MethodMissing.new(compiler.source_name , word.symbol).to_risc(compiler)
|
MethodMissing.new(compiler.source_name , word.symbol).to_risc(compiler)
|
||||||
|
|
||||||
ok_label
|
ok_label
|
||||||
cache_entry.cached_method == callable_method
|
cache_entry.cached_method = callable_method
|
||||||
|
12
lib/slot_language/load_maker.rb
Normal file
12
lib/slot_language/load_maker.rb
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
module SlotLanguage
|
||||||
|
class LoadMaker
|
||||||
|
attr_reader :left , :right
|
||||||
|
|
||||||
|
def initialize(left , right)
|
||||||
|
@left = left
|
||||||
|
@right = right
|
||||||
|
raise "No Slot #{left}" unless left.is_a?(SlotMaker)
|
||||||
|
raise "No Slot #{right}" unless right.is_a?(SlotMaker)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -16,18 +16,23 @@ module SlotLanguage
|
|||||||
not_implemented(node)
|
not_implemented(node)
|
||||||
end
|
end
|
||||||
def on_send(statement)
|
def on_send(statement)
|
||||||
|
#puts "Send #{statement}"
|
||||||
kids = statement.children.dup
|
kids = statement.children.dup
|
||||||
receiver = process(kids.shift) || MessageSlot.new
|
receiver = process(kids.shift) || MessageSlot.new
|
||||||
name = kids.shift
|
name = kids.shift
|
||||||
return label(name) if(name.to_s.end_with?("_label"))
|
return label(name) if(name.to_s.end_with?("_label"))
|
||||||
return goto(name,kids) if(name == :goto)
|
return goto(name,kids) if(name == :goto)
|
||||||
return check(name,receiver, kids) if(name == :==)
|
return check(name,receiver, kids) if(name == :==)
|
||||||
|
return assign(receiver, name , kids) if(name.to_s.end_with?("="))
|
||||||
SlotMaker.new( name , receiver )
|
SlotMaker.new( name , receiver )
|
||||||
end
|
end
|
||||||
|
def on_lvar(lvar)
|
||||||
|
SlotMaker.new(lvar.children.first , nil)
|
||||||
|
end
|
||||||
def on_lvasgn expression
|
def on_lvasgn expression
|
||||||
name = expression.children[0]
|
name = expression.children[0]
|
||||||
value = process(expression.children[1])
|
value = process(expression.children[1])
|
||||||
Sol::LocalAssignment.new(name,value)
|
LoadMaker.new(SlotMaker.new(name,nil),value)
|
||||||
end
|
end
|
||||||
def on_if(expression)
|
def on_if(expression)
|
||||||
condition = process(expression.children[0])
|
condition = process(expression.children[0])
|
||||||
@ -38,7 +43,7 @@ module SlotLanguage
|
|||||||
process(exp.first)
|
process(exp.first)
|
||||||
end
|
end
|
||||||
def on_ivar expression
|
def on_ivar expression
|
||||||
Sol::InstanceVariable.new(instance_name(expression.children.first))
|
SlotMaker.new(instance_name(expression.children.first),nil)
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@ -56,13 +61,20 @@ module SlotLanguage
|
|||||||
def check(name , receiver , kids)
|
def check(name , receiver , kids)
|
||||||
raise "Only ==, not #{name}" unless name == :==
|
raise "Only ==, not #{name}" unless name == :==
|
||||||
raise "Familiy too large #{kids}" if kids.length > 1
|
raise "Familiy too large #{kids}" if kids.length > 1
|
||||||
puts "Kids " + kids.to_s
|
#puts "Kids " + kids.to_s
|
||||||
right = process(kids.first)
|
right = process(kids.first)
|
||||||
CheckMaker.new(name , receiver , right)
|
CheckMaker.new(name , receiver , right)
|
||||||
end
|
end
|
||||||
|
def assign(receiver , name , kids)
|
||||||
|
name = name.to_s[0...-1].to_sym
|
||||||
|
receiver.add_slot_name(name)
|
||||||
|
right = process kids.first
|
||||||
|
LoadMaker.new(receiver,right)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
require_relative "named_slot"
|
require_relative "named_slot"
|
||||||
require_relative "message_slot"
|
require_relative "message_slot"
|
||||||
require_relative "slot_maker"
|
require_relative "slot_maker"
|
||||||
|
require_relative "load_maker"
|
||||||
require_relative "check_maker"
|
require_relative "check_maker"
|
||||||
|
@ -1,8 +1,22 @@
|
|||||||
module SlotLanguage
|
module SlotLanguage
|
||||||
class SlotMaker
|
class SlotMaker
|
||||||
attr_reader :name
|
attr_reader :name , :leaps
|
||||||
|
|
||||||
def initialize(name , more)
|
def initialize(name , more)
|
||||||
@name = name
|
@name = name
|
||||||
|
if(more.is_a?(Array))
|
||||||
|
@leaps = more
|
||||||
|
else
|
||||||
|
@leaps = [more] if more
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def add_slot_name(name)
|
||||||
|
if(@leaps)
|
||||||
|
@leaps << name
|
||||||
|
else
|
||||||
|
@leaps = [name]
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -10,16 +10,13 @@ module SlotLanguage
|
|||||||
def test_compile
|
def test_compile
|
||||||
assert_equal SlotMaker , compile("a").class
|
assert_equal SlotMaker , compile("a").class
|
||||||
end
|
end
|
||||||
# def test_fail_args
|
|
||||||
# assert_raises{ compile("a(1)")}
|
|
||||||
# end
|
|
||||||
def test_label
|
def test_label
|
||||||
label = compile("while_label")
|
label = compile("while_label")
|
||||||
assert_equal SlotMachine::Label , label.class
|
assert_equal SlotMachine::Label , label.class
|
||||||
assert_equal :while_label , label.name
|
assert_equal :while_label , label.name
|
||||||
end
|
end
|
||||||
def test_slot_load
|
def test_slot_load
|
||||||
assert_equal Sol::LocalAssignment , compile_class("a = @b")
|
assert_equal LoadMaker , compile_class("a = @b")
|
||||||
end
|
end
|
||||||
def test_goto
|
def test_goto
|
||||||
assert_equal SlotMachine::Jump , compile_class("goto(exit_label)")
|
assert_equal SlotMachine::Jump , compile_class("goto(exit_label)")
|
||||||
@ -29,5 +26,17 @@ module SlotLanguage
|
|||||||
assert_equal CheckMaker , check.class
|
assert_equal CheckMaker , check.class
|
||||||
assert_equal SlotMachine::Jump , check.goto.class
|
assert_equal SlotMachine::Jump , check.goto.class
|
||||||
end
|
end
|
||||||
|
def test_assign
|
||||||
|
assign = compile("c = d")
|
||||||
|
assert_equal LoadMaker , assign.class
|
||||||
|
end
|
||||||
|
def test_assign1
|
||||||
|
assign = compile("c = c.next")
|
||||||
|
assert_equal LoadMaker , assign.class
|
||||||
|
end
|
||||||
|
def test_assign2
|
||||||
|
assign = compile("c.next = d")
|
||||||
|
assert_equal LoadMaker , assign.class
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user