2018-03-26 19:46:38 +03:00
|
|
|
require "util/list"
|
2014-05-03 15:13:44 +03:00
|
|
|
|
2018-03-18 10:36:01 +05:30
|
|
|
module Risc
|
2015-10-23 21:27:36 +03:00
|
|
|
|
2018-03-18 10:36:01 +05:30
|
|
|
# the register machine has at least 8 registers, named r0-r5 , :lr and :pc
|
|
|
|
# (for historical reasons , r for register, pc for ProgramCounter ie next instruction address
|
|
|
|
# and lr means LinkRegister, ie the location where to return to when in a function)
|
|
|
|
#
|
|
|
|
# We can load and store their contents, move data between them and
|
|
|
|
# access (get/set) memory at a constant offset from a register
|
2019-10-03 20:55:41 +03:00
|
|
|
# While SlotMachine works with objects, the register machine has registers,
|
2018-03-18 10:36:01 +05:30
|
|
|
# but we keep the names for better understanding, r2-5 are temporary/scratch
|
|
|
|
# There is no direct memory access, only through registers
|
|
|
|
# Constants can/must be loaded into registers before use
|
|
|
|
|
|
|
|
# At compile time, Instructions form a linked list (:next attribute is the link)
|
|
|
|
# At run time Instructions are traversesed as a graph
|
|
|
|
#
|
2015-10-23 21:27:36 +03:00
|
|
|
# Branches fan out, Labels collect
|
|
|
|
# Labels are the only valid branch targets
|
2018-03-18 10:36:01 +05:30
|
|
|
#
|
2014-10-03 10:25:10 +03:00
|
|
|
class Instruction
|
2018-03-26 20:05:30 +03:00
|
|
|
include Util::List
|
2014-10-03 10:25:10 +03:00
|
|
|
|
2018-03-26 20:05:30 +03:00
|
|
|
def initialize( source , nekst = nil )
|
2015-10-28 21:38:23 +02:00
|
|
|
@source = source
|
|
|
|
@next = nekst
|
|
|
|
return unless source
|
2018-04-18 19:27:46 +03:00
|
|
|
raise "Source must be string or Instruction, not #{source.class}" unless source.is_a?(String) or
|
2019-10-03 20:55:41 +03:00
|
|
|
source.is_a?(SlotMachine::Instruction) or source.is_a?(Parfait::Callable)
|
2015-07-18 11:21:49 +03:00
|
|
|
end
|
2015-07-27 12:13:39 +03:00
|
|
|
attr_reader :source
|
2015-07-18 11:21:49 +03:00
|
|
|
|
2018-03-26 14:54:41 +03:00
|
|
|
def to_arr
|
|
|
|
ret = []
|
|
|
|
self.each {|ins| ret << ins}
|
2015-10-23 21:27:36 +03:00
|
|
|
ret
|
2014-06-08 01:41:56 +03:00
|
|
|
end
|
2015-10-23 21:27:36 +03:00
|
|
|
|
2018-06-17 13:53:17 +03:00
|
|
|
# just part of the protocol, noop in this case
|
|
|
|
def precheck
|
|
|
|
end
|
|
|
|
|
2020-03-18 17:49:23 +02:00
|
|
|
# return an array of names of registers that is used by the instruction
|
2020-03-20 13:58:40 +02:00
|
|
|
def register_attributes
|
2020-03-18 17:49:23 +02:00
|
|
|
raise "Not implemented in #{self.class}"
|
|
|
|
end
|
|
|
|
|
2020-03-20 13:58:40 +02:00
|
|
|
# return an array of names of registers that is used by the instruction
|
|
|
|
def register_names
|
|
|
|
register_attributes.collect {|attr| get_register(attr)}
|
|
|
|
end
|
|
|
|
|
|
|
|
# get the register value that has the name (ie attribute by that name)
|
|
|
|
# return the registers name ie RegisterValue's symbol
|
|
|
|
def get_register(name)
|
|
|
|
instance_variable_get("@#{name}".to_sym).symbol
|
|
|
|
end
|
|
|
|
|
|
|
|
# set all registers that has the name "name"
|
|
|
|
# going through the register_names and setting all where the
|
|
|
|
# get_register would return name
|
2020-03-20 18:33:37 +02:00
|
|
|
# Create new RegisterValue with new name and swap the variable out
|
2020-03-20 13:58:40 +02:00
|
|
|
def set_registers(name , value)
|
|
|
|
register_attributes.each do |attr|
|
|
|
|
reg = instance_variable_get("@#{attr}".to_sym)
|
|
|
|
next unless reg.symbol == name
|
2020-03-20 18:33:37 +02:00
|
|
|
new_reg = reg.dup(value)
|
|
|
|
instance_variable_set("@#{attr}".to_sym , new_reg)
|
2020-03-20 13:58:40 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-25 19:38:59 +03:00
|
|
|
def to_cpu( translator )
|
|
|
|
translator.translate( self )
|
|
|
|
end
|
|
|
|
|
2018-03-22 18:38:19 +02:00
|
|
|
def class_source( derived)
|
|
|
|
"#{self.class.name.split("::").last}: #{derived} #{source_mini}"
|
|
|
|
end
|
|
|
|
def source_mini
|
|
|
|
return "(no source)" unless source
|
2018-05-23 18:05:22 +03:00
|
|
|
return "(from: #{source[0..50]})" if source.is_a?(String)
|
2018-03-22 18:38:19 +02:00
|
|
|
"(from: #{source.class.name.split("::").last})"
|
|
|
|
end
|
2014-05-02 08:02:25 +03:00
|
|
|
end
|
2015-05-24 20:00:11 +03:00
|
|
|
|
2014-05-02 08:02:25 +03:00
|
|
|
end
|
2014-10-03 11:05:17 +03:00
|
|
|
|
2020-03-02 17:58:13 +02:00
|
|
|
require_relative "setter"
|
|
|
|
require_relative "getter"
|
|
|
|
require_relative "reg_to_slot"
|
|
|
|
require_relative "slot_to_reg"
|
|
|
|
require_relative "reg_to_byte"
|
|
|
|
require_relative "byte_to_reg"
|
|
|
|
require_relative "load_constant"
|
|
|
|
require_relative "load_data"
|
|
|
|
require_relative "syscall"
|
|
|
|
require_relative "function_call"
|
|
|
|
require_relative "function_return"
|
|
|
|
require_relative "transfer"
|
|
|
|
require_relative "label"
|
|
|
|
require_relative "branch"
|
|
|
|
require_relative "dynamic_jump"
|
|
|
|
require_relative "operator_instruction"
|