fix positioning and the assembly works
This commit is contained in:
parent
34b16a2332
commit
e012f16d7f
@ -25,7 +25,7 @@ module Parfait
|
|||||||
end
|
end
|
||||||
def each_word
|
def each_word
|
||||||
index = 1
|
index = 1
|
||||||
while( index <= 14)
|
while( index < 14)
|
||||||
yield get_internal_word(index)
|
yield get_internal_word(index)
|
||||||
index += 1
|
index += 1
|
||||||
end
|
end
|
||||||
|
@ -49,7 +49,7 @@ module Parfait
|
|||||||
1
|
1
|
||||||
end
|
end
|
||||||
def padded_length
|
def padded_length
|
||||||
2
|
2 * 4
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -58,7 +58,7 @@ module Parfait
|
|||||||
7
|
7
|
||||||
end
|
end
|
||||||
def padded_length
|
def padded_length
|
||||||
8
|
8 * 4
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
class Data16 < DataObject
|
class Data16 < DataObject
|
||||||
@ -66,7 +66,7 @@ module Parfait
|
|||||||
15
|
15
|
||||||
end
|
end
|
||||||
def padded_length
|
def padded_length
|
||||||
16
|
16 * 4
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -8,7 +8,7 @@ module Risc
|
|||||||
|
|
||||||
class Assembler
|
class Assembler
|
||||||
include Logging
|
include Logging
|
||||||
log_level :debug
|
log_level :info
|
||||||
|
|
||||||
MARKER = 0xA51AF00D
|
MARKER = 0xA51AF00D
|
||||||
|
|
||||||
@ -35,7 +35,6 @@ module Risc
|
|||||||
all.each do |objekt|
|
all.each do |objekt|
|
||||||
next if objekt.is_a?(Risc::Label)
|
next if objekt.is_a?(Risc::Label)
|
||||||
log.debug "Linked #{objekt.class}:0x#{objekt.object_id.to_s(16)} at 0x#{Positioned.position(objekt).to_s(16)} / 0x#{objekt.padded_length.to_s(16)}"
|
log.debug "Linked #{objekt.class}:0x#{objekt.object_id.to_s(16)} at 0x#{Positioned.position(objekt).to_s(16)} / 0x#{objekt.padded_length.to_s(16)}"
|
||||||
Positioned.position(objekt)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -32,7 +32,9 @@ module Risc
|
|||||||
translate(Arm::Translator.new)
|
translate(Arm::Translator.new)
|
||||||
end
|
end
|
||||||
|
|
||||||
# translate the methods to whatever cpu the translator translates to
|
# translate code to whatever cpu the translator translates to
|
||||||
|
# this means translating the initial jump (cpu_init into binary_init)
|
||||||
|
# and then translating all methods
|
||||||
def translate( translator )
|
def translate( translator )
|
||||||
methods = Parfait.object_space.collect_methods
|
methods = Parfait.object_space.collect_methods
|
||||||
translate_methods( methods , translator )
|
translate_methods( methods , translator )
|
||||||
@ -40,6 +42,7 @@ module Risc
|
|||||||
@binary_init = Parfait::BinaryCode.new(1)
|
@binary_init = Parfait::BinaryCode.new(1)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# go through all methods and translate them to cpu, given the translator
|
||||||
def translate_methods(methods , translator)
|
def translate_methods(methods , translator)
|
||||||
methods.each do |method|
|
methods.each do |method|
|
||||||
log.debug "Translate method #{method.name}"
|
log.debug "Translate method #{method.name}"
|
||||||
@ -52,6 +55,15 @@ module Risc
|
|||||||
@objects ||= Collector.collect_space
|
@objects ||= Collector.collect_space
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# To create binaries, objects (and labels) need to have a position
|
||||||
|
# (so objects can be loaded and branches know where to jump)
|
||||||
|
#
|
||||||
|
# Position in the order
|
||||||
|
# - initial jump
|
||||||
|
# - all object
|
||||||
|
# - all code
|
||||||
|
# As code length amy change during assembly, this way at least the objects stay
|
||||||
|
# in place and we don't have to deal with chaning loading code
|
||||||
def position_all
|
def position_all
|
||||||
translate_arm unless @translated
|
translate_arm unless @translated
|
||||||
#need the initial jump at 0 and then functions
|
#need the initial jump at 0 and then functions
|
||||||
@ -63,43 +75,44 @@ module Risc
|
|||||||
position_code_from( at )
|
position_code_from( at )
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# go through everything that is not code (BinaryCode) and set position
|
||||||
|
# padded_length is what determines an objects (byte) length
|
||||||
def position_objects( at )
|
def position_objects( at )
|
||||||
at += 8 # thats the padding
|
|
||||||
# want to have the objects first in the executable
|
# want to have the objects first in the executable
|
||||||
objects.each do | id , objekt|
|
objects.each do | id , objekt|
|
||||||
case objekt
|
next if objekt.is_a?( Parfait::BinaryCode) or objekt.is_a?( Risc::Label )
|
||||||
when Parfait::BinaryCode
|
|
||||||
when Risc::Label
|
|
||||||
else
|
|
||||||
Positioned.set_position(objekt,at)
|
Positioned.set_position(objekt,at)
|
||||||
at += objekt.padded_length
|
at += objekt.padded_length
|
||||||
end
|
end
|
||||||
end
|
|
||||||
at
|
at
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Position all BinaryCode.
|
||||||
|
#
|
||||||
|
# So that all code from one method is layed out linearly (for debuggin)
|
||||||
|
# we go through methods, and then through all codes from the method
|
||||||
def position_code_from( at )
|
def position_code_from( at )
|
||||||
objects.each do |id , method|
|
objects.each do |id , method|
|
||||||
next unless method.is_a? Parfait::TypedMethod
|
next unless method.is_a? Parfait::TypedMethod
|
||||||
log.debug "CODE1 #{method.name}:#{at}"
|
log.debug "CODE1 #{method.name}:#{at}"
|
||||||
method.cpu_instructions.set_position( at + 12) # BinaryCode header
|
method.cpu_instructions.set_position( at )
|
||||||
before = at
|
before = at
|
||||||
nekst = method.binary
|
nekst = method.binary
|
||||||
while(nekst)
|
while(nekst)
|
||||||
Positioned.set_position(nekst , at)
|
Positioned.set_position(nekst , at)
|
||||||
at += nekst.padded_length
|
at += nekst.padded_length
|
||||||
nekst = nekst.next
|
nekst = nekst.next
|
||||||
#puts "LENGTH #{len}"
|
|
||||||
end
|
end
|
||||||
log.debug "CODE2 #{method.name}:#{at} len: #{at - before}"
|
log.debug "CODE2 #{method.name}:#{at} len: #{at - before}"
|
||||||
end
|
end
|
||||||
at
|
at
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Create Binary code for all methods and the initial jump
|
||||||
|
# BinaryWriter handles the writing from instructions into BinaryCode objects
|
||||||
def create_binary
|
def create_binary
|
||||||
objects.each do |id , method|
|
objects.each do |id , method|
|
||||||
next unless method.is_a? Parfait::TypedMethod
|
next unless method.is_a? Parfait::TypedMethod
|
||||||
#puts "Binary for #{method.name}:#{}"
|
|
||||||
writer = BinaryWriter.new(method.binary)
|
writer = BinaryWriter.new(method.binary)
|
||||||
writer.assemble(method.cpu_instructions)
|
writer.assemble(method.cpu_instructions)
|
||||||
end
|
end
|
||||||
|
@ -72,7 +72,7 @@ module Parfait
|
|||||||
def test_each
|
def test_each
|
||||||
len = 0
|
len = 0
|
||||||
@code.each_word{ len += 1}
|
@code.each_word{ len += 1}
|
||||||
assert_equal 14 , len
|
assert_equal 13 , len
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -6,10 +6,10 @@ module Risc
|
|||||||
def setup
|
def setup
|
||||||
@machine = Risc.machine.boot
|
@machine = Risc.machine.boot
|
||||||
end
|
end
|
||||||
def est_init
|
def test_init
|
||||||
@assembler = Assembler.new(@machine)
|
@assembler = Assembler.new(@machine)
|
||||||
end
|
end
|
||||||
def est_write_fails
|
def test_write_fails
|
||||||
@assembler = Assembler.new(@machine)
|
@assembler = Assembler.new(@machine)
|
||||||
assert_raises{ @assembler.write_as_string} #must translate first
|
assert_raises{ @assembler.write_as_string} #must translate first
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user