create return address as own class to hold return addresses
to distinguish from integer, which does not need adjusting
This commit is contained in:
parent
e86ca5ae9d
commit
e39e96f646
@ -49,6 +49,13 @@ module Parfait
|
|||||||
# :>, :>=, :<, :<=, :between?
|
# :>, :>=, :<, :<=, :between?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Marker class. Same functionality as Integer
|
||||||
|
# (ie carrying an int, the return address)
|
||||||
|
#
|
||||||
|
# But the integer (address) needs to be adjusted by load address.
|
||||||
|
class ReturnAddress < Integer
|
||||||
|
end
|
||||||
|
|
||||||
# adding other base classes in here for now:
|
# adding other base classes in here for now:
|
||||||
class FalseClass < Data4
|
class FalseClass < Data4
|
||||||
#FIXME: this is "just" for compilation
|
#FIXME: this is "just" for compilation
|
||||||
|
@ -119,7 +119,7 @@ module Risc
|
|||||||
def super_class_names
|
def super_class_names
|
||||||
{ Data4: :DataObject , Data8: :DataObject ,Data16: :DataObject ,
|
{ Data4: :DataObject , Data8: :DataObject ,Data16: :DataObject ,
|
||||||
BinaryCode: :Data16 , Integer: :Data4 , Word: :Data8 ,
|
BinaryCode: :Data16 , Integer: :Data4 , Word: :Data8 ,
|
||||||
Object: :BasicObject , List: :Data16}
|
Object: :BasicObject , List: :Data16 , ReturnAddress: :Integer}
|
||||||
end
|
end
|
||||||
|
|
||||||
# the function really just returns a constant (just avoiding the constant)
|
# the function really just returns a constant (just avoiding the constant)
|
||||||
@ -132,6 +132,7 @@ module Risc
|
|||||||
return_address: :Integer, return_value: :Object,
|
return_address: :Integer, return_value: :Object,
|
||||||
caller: :Message , name: :Word , arguments: :NamedList },
|
caller: :Message , name: :Word , arguments: :NamedList },
|
||||||
Integer: {next_integer: :Integer},
|
Integer: {next_integer: :Integer},
|
||||||
|
ReturnAddress: {next_integer: :Integer},
|
||||||
DataObject: {},
|
DataObject: {},
|
||||||
Data4: {},
|
Data4: {},
|
||||||
Data8: {},
|
Data8: {},
|
||||||
|
@ -66,8 +66,7 @@ module Risc
|
|||||||
# if none was given
|
# if none was given
|
||||||
def self.label( source , name , position = nil , nekst = nil)
|
def self.label( source , name , position = nil , nekst = nil)
|
||||||
unless position
|
unless position
|
||||||
position = Parfait.object_space.get_integer
|
position = Risc.machine.get_address
|
||||||
Risc.machine.add_constant(position)
|
|
||||||
end
|
end
|
||||||
Label.new( source , name , position, nekst )
|
Label.new( source , name , position, nekst )
|
||||||
end
|
end
|
||||||
|
@ -20,6 +20,7 @@ module Risc
|
|||||||
@booted = false
|
@booted = false
|
||||||
@risc_init = nil
|
@risc_init = nil
|
||||||
@constants = []
|
@constants = []
|
||||||
|
@next_address = nil
|
||||||
end
|
end
|
||||||
attr_reader :constants , :cpu_init
|
attr_reader :constants , :cpu_init
|
||||||
attr_reader :booted , :translated
|
attr_reader :booted , :translated
|
||||||
@ -61,6 +62,17 @@ module Risc
|
|||||||
@constants << const
|
@constants << const
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# hand out a return address for use as constant the address is added
|
||||||
|
def get_address
|
||||||
|
10.times do # 10 for whole pages
|
||||||
|
@next_address = Parfait::ReturnAddress.new(0,@next_address)
|
||||||
|
add_constant( @next_address )
|
||||||
|
end unless @next_address
|
||||||
|
addr = @next_address
|
||||||
|
@next_address = @next_address.next_integer
|
||||||
|
addr
|
||||||
|
end
|
||||||
|
|
||||||
# To create binaries, objects (and labels) need to have a position
|
# To create binaries, objects (and labels) need to have a position
|
||||||
# (so objects can be loaded and branches know where to jump)
|
# (so objects can be loaded and branches know where to jump)
|
||||||
#
|
#
|
||||||
|
@ -148,8 +148,8 @@ module Risc
|
|||||||
def write_integer( int )
|
def write_integer( int )
|
||||||
write_ref_for( int.get_type )
|
write_ref_for( int.get_type )
|
||||||
write_ref_for( int.next_integer )
|
write_ref_for( int.next_integer )
|
||||||
@stream.write_signed_int_32( int.value )
|
write_ref_for( int.value )
|
||||||
@stream.write_signed_int_32( 0 )
|
write_ref_for( 0 )
|
||||||
log.debug "Integer witten stream 0x#{@stream.length.to_s(16)}"
|
log.debug "Integer witten stream 0x#{@stream.length.to_s(16)}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -48,4 +48,18 @@ module Parfait
|
|||||||
assert_equal 1 , @int.value
|
assert_equal 1 , @int.value
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
class AddressTest < ParfaitTest
|
||||||
|
def test_address
|
||||||
|
assert ReturnAddress.new(55)
|
||||||
|
end
|
||||||
|
def test_value
|
||||||
|
assert_equal 55 , ReturnAddress.new(55).value
|
||||||
|
end
|
||||||
|
def test_value_set
|
||||||
|
addr = ReturnAddress.new(55)
|
||||||
|
addr.set_value(33)
|
||||||
|
assert_equal 33 , addr.value
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
@ -46,7 +46,7 @@ module Risc
|
|||||||
ret = main_ticks(63)
|
ret = main_ticks(63)
|
||||||
assert_equal FunctionReturn , ret.class
|
assert_equal FunctionReturn , ret.class
|
||||||
assert_equal :r1 , ret.register.symbol
|
assert_equal :r1 , ret.register.symbol
|
||||||
assert_equal 19460 , @interpreter.get_register(ret.register)
|
assert_equal 20612 , @interpreter.get_register(ret.register)
|
||||||
end
|
end
|
||||||
def test_sys
|
def test_sys
|
||||||
sys = main_ticks(65)
|
sys = main_ticks(65)
|
||||||
|
@ -52,7 +52,7 @@ module Risc
|
|||||||
end
|
end
|
||||||
def test_pc1
|
def test_pc1
|
||||||
@interpreter.tick
|
@interpreter.tick
|
||||||
assert_equal 19224 , @interpreter.pc
|
assert_equal 20376 , @interpreter.pc
|
||||||
end
|
end
|
||||||
def test_tick2
|
def test_tick2
|
||||||
@interpreter.tick
|
@interpreter.tick
|
||||||
@ -66,7 +66,7 @@ module Risc
|
|||||||
def test_pc2
|
def test_pc2
|
||||||
@interpreter.tick
|
@interpreter.tick
|
||||||
@interpreter.tick
|
@interpreter.tick
|
||||||
assert_equal 19228 , @interpreter.pc
|
assert_equal 20380 , @interpreter.pc
|
||||||
end
|
end
|
||||||
def test_tick_14_jump
|
def test_tick_14_jump
|
||||||
14.times {@interpreter.tick}
|
14.times {@interpreter.tick}
|
||||||
|
@ -17,6 +17,22 @@ module Risc
|
|||||||
def test_constant
|
def test_constant
|
||||||
assert @machine.add_constant( Parfait::Integer.new(5) )
|
assert @machine.add_constant( Parfait::Integer.new(5) )
|
||||||
end
|
end
|
||||||
|
def test_address_get
|
||||||
|
assert_equal Parfait::ReturnAddress , @machine.get_address.class
|
||||||
|
end
|
||||||
|
def test_address_is_constant
|
||||||
|
addr = @machine.get_address
|
||||||
|
assert @machine.constants.include?(addr)
|
||||||
|
end
|
||||||
|
def test_address_count
|
||||||
|
addr = @machine.get_address
|
||||||
|
count = 0
|
||||||
|
while(addr)
|
||||||
|
count += 1
|
||||||
|
addr = addr.next_integer
|
||||||
|
end
|
||||||
|
assert_equal 5, count
|
||||||
|
end
|
||||||
end
|
end
|
||||||
class TestMachineInit < MiniTest::Test
|
class TestMachineInit < MiniTest::Test
|
||||||
def setup
|
def setup
|
||||||
@ -29,10 +45,10 @@ module Risc
|
|||||||
assert_equal 0 , Position.get(@machine.cpu_init).at
|
assert_equal 0 , Position.get(@machine.cpu_init).at
|
||||||
end
|
end
|
||||||
def test_cpu_at
|
def test_cpu_at
|
||||||
assert_equal "0x5a54" , Position.get(@machine.cpu_init.first).to_s
|
assert_equal "0x5ed4" , Position.get(@machine.cpu_init.first).to_s
|
||||||
end
|
end
|
||||||
def test_cpu_bin
|
def test_cpu_bin
|
||||||
assert_equal "0x5a4c" , Position.get(Position.get(@machine.cpu_init.first).binary).to_s
|
assert_equal "0x5ecc" , Position.get(Position.get(@machine.cpu_init.first).binary).to_s
|
||||||
end
|
end
|
||||||
def test_cpu_label
|
def test_cpu_label
|
||||||
assert_equal Position::InstructionPosition , Position.get(@machine.cpu_init.first).class
|
assert_equal Position::InstructionPosition , Position.get(@machine.cpu_init.first).class
|
||||||
|
Loading…
Reference in New Issue
Block a user