rubyx/test/support/risc_interpreter.rb

120 lines
3.1 KiB
Ruby
Raw Normal View History

require "util/eventable"
require "risc/interpreter"
require_relative "compiling"
2017-04-10 16:12:15 +03:00
module Risc
module Ticker
include CompilerHelper
def setup
Risc.machine.boot
do_clean_compile
Vool::VoolCompiler.ruby_to_vool( @string_input )
Collector.collect_space
@interpreter = Interpreter.new
@interpreter.start Risc.machine.risc_init
end
2018-04-23 14:05:37 +03:00
alias :do_setup :setup
# must be after boot, but before main compile, to define method
def do_clean_compile
end
2017-04-10 16:12:15 +03:00
# check the given array of instructions is what the interpreter actually does
# possible second argument ignores the given amount, usually up until main
def check_chain( should , start_at = 0 )
ticks( start_at ) if start_at > 0
2017-04-10 16:12:15 +03:00
should.each_with_index do |name , index|
got = ticks(1)
assert_equal got.class ,name , "Wrong class for #{index + 1 - start_at}, expect #{name} , got #{got}"
2017-04-10 16:12:15 +03:00
end
end
# check the main only, ignoring the __init instructions
def check_main_chain( should )
check_chain( should , main_at)
end
# how many instruction up until the main starts, ie
# ticks(main_at) will be the label for main
def main_at
2018-04-07 18:58:44 +03:00
26
end
2017-04-10 16:12:15 +03:00
def get_return
2017-04-10 16:12:15 +03:00
assert_equal Parfait::Message , @interpreter.get_register(:r0).class
@interpreter.get_register(:r0).return_value
2017-04-10 16:12:15 +03:00
end
# do as many as given ticks in the main, ie main_at more
def main_ticks(num)
ticks( main_at + num)
end
# do a certain number of steps in the interpreter and return the last executed instruction
2018-03-23 18:55:23 +02:00
def ticks( num )
2017-04-10 16:12:15 +03:00
last = nil
num.times do
last = @interpreter.instruction
@interpreter.tick
end
return last
end
# collect the classes of all executed istructions
2018-04-23 14:05:37 +03:00
def all_classes(max = 300)
2017-04-10 16:12:15 +03:00
classes = []
tick = 1
begin
2018-04-23 14:05:37 +03:00
while true and (classes.length < max)
2017-04-10 16:12:15 +03:00
cl = ticks(1).class
tick += 1
classes << cl
break if cl == NilClass
end
rescue => e
puts "Error at tick #{tick}"
puts e
puts e.backtrace
end
classes
end
2018-04-26 12:33:33 +03:00
# do the setup, compile and run the input (a main) to the end
2018-04-24 20:12:49 +03:00
def run_main(input)
@string_input = as_main(input)
do_setup
2018-04-23 14:05:37 +03:00
run_all
end
2018-04-26 12:33:33 +03:00
# wrap the input in Space (main is assumed to be part of it)
def run_space(input)
@string_input = in_Space(input)
do_setup
run_all
end
2018-04-23 14:05:37 +03:00
def run_all
@interpreter.tick while(@interpreter.instruction)
end
# for chaning the tests quickly output all instructions that are executed
def show_ticks
classes = all_classes
output_classes(classes)
end
# show all instructions of the main only
def show_main_ticks
classes = all_classes
classes = classes.slice(main_at , 1000)
output_classes(classes)
end
def output_classes(classes)
2017-04-10 16:12:15 +03:00
str = classes.to_s.gsub("Risc::","")
all = str.split(",").each_slice(5).collect {|line| " " + line.join(",")}
puts all.join(",\n")
2017-04-10 16:12:15 +03:00
puts "length = #{classes.length}"
exit(1)
end
end
end