improve argument handling for puntint
This commit is contained in:
parent
8ba79d8de8
commit
76ce9aa654
@ -33,47 +33,45 @@ module Core
|
||||
end
|
||||
|
||||
def putint context , arg = Vm::Integer
|
||||
arg = Vm::Integer.new( 0 )
|
||||
function = Vm::Function.new(:putint , [arg ] , arg )
|
||||
buffer = Vm::StringConstant.new(" ")
|
||||
context.program.add_object buffer
|
||||
str_addr = Vm::Integer.new(1)
|
||||
context.str_addr = str_addr
|
||||
reg1 = Vm::Integer.new(2)
|
||||
itos_fun = context.program.get_or_create_function(:utoa)
|
||||
function.body.instance_eval do
|
||||
mov( reg1 , str_addr ) #move arg up
|
||||
add( str_addr , buffer ,nil ) # string to write to
|
||||
add( str_addr , str_addr , (buffer.length-3))
|
||||
call( itos_fun )
|
||||
putint_function = Vm::Function.new(:putint , [arg] , arg )
|
||||
buffer = Vm::StringConstant.new(" ") # create a buffer
|
||||
context.program.add_object buffer # and save it (function local variable: a no no)
|
||||
int = putint_function.args.first
|
||||
moved_int = Vm::Integer.new(1)
|
||||
utoa = context.program.get_or_create_function(:utoa)
|
||||
putint_function.body.instance_eval do
|
||||
mov( moved_int , int ) #move arg up
|
||||
add( int , buffer ,nil ) # string to write to
|
||||
add( int , int , (buffer.length-3))
|
||||
call( utoa )
|
||||
# And now we "just" have to print it, using the write_stdout
|
||||
add( str_addr , buffer , nil ) # string to write to
|
||||
mov( reg1 , buffer.length )
|
||||
add( int , buffer , nil ) # string to write to
|
||||
mov( moved_int , buffer.length )
|
||||
end
|
||||
ret = Vm::CMachine.instance.write_stdout(function.body)
|
||||
function
|
||||
Vm::CMachine.instance.write_stdout(putint_function.body)
|
||||
putint_function
|
||||
end
|
||||
|
||||
# The conversion to base10 is quite a bit more complicated than i thought. The bulk of it is in div10
|
||||
# We set up variables, do the devision and write the result to the string
|
||||
# then check if were done and recurse if neccessary
|
||||
# As we write before we recurse (save a push) we write the number backwards
|
||||
# arguments: string address , integer
|
||||
def utoa context
|
||||
function = Vm::Function.new(:utoa , [Vm::Integer , Vm::Integer ] )
|
||||
block = function.body
|
||||
str_addr = context.str_addr
|
||||
number = Vm::Integer.new(str_addr.register + 1)
|
||||
utoa_function = Vm::Function.new(:utoa , [Vm::Integer , Vm::Integer ] , Vm::Integer )
|
||||
str_addr = utoa_function.args[0]
|
||||
number = utoa_function.args[1]
|
||||
remainder = Vm::Integer.new( number.register + 1)
|
||||
Vm::CMachine.instance.div10( block , number , remainder )
|
||||
Vm::CMachine.instance.div10( utoa_function.body , number , remainder )
|
||||
# make char out of digit (by using ascii encoding) 48 == "0"
|
||||
block.instance_eval do
|
||||
utoa_function.body.instance_eval do
|
||||
add( remainder , remainder , 48 )
|
||||
strb( remainder, right: str_addr )
|
||||
sub( str_addr, str_addr , 1 )
|
||||
cmp( number , 0 )
|
||||
callne( function )
|
||||
callne( utoa_function )
|
||||
end
|
||||
return function
|
||||
return utoa_function
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -22,13 +22,31 @@ module Vm
|
||||
def initialize(name , args = [] , return_type = nil)
|
||||
super()
|
||||
@name = name
|
||||
@args = args
|
||||
@return_type = return_type
|
||||
@entry = Core::Kernel::function_entry( Vm::Block.new("#{name}_entry") ,name )
|
||||
@exit = Core::Kernel::function_exit( Vm::Block.new("#{name}_exit") , name )
|
||||
@body = Block.new("#{name}_body")
|
||||
@reg_count = 0
|
||||
branch_body
|
||||
@args = Array.new(args.length)
|
||||
args.each_with_index do |arg , i|
|
||||
if arg.is_a?(Value)
|
||||
@args[i] = arg
|
||||
raise "arg in non std register #{arg.inspect}" unless i == arg.register
|
||||
else
|
||||
@args[i] = arg.new(i)
|
||||
end
|
||||
end
|
||||
@return_type = return_type || Vm::Integer
|
||||
if @return_type.is_a?(Value)
|
||||
raise "return in non std register #{@return_type.inspect}" unless 0 == @return_type.register
|
||||
else
|
||||
@return_type = @return_type.new(0)
|
||||
end
|
||||
@entry = Core::Kernel::function_entry( Vm::Block.new("#{name}_entry") ,name )
|
||||
@exit = Core::Kernel::function_exit( Vm::Block.new("#{name}_exit") , name )
|
||||
@body = Block.new("#{name}_body")
|
||||
@reg_count = 0
|
||||
branch_body
|
||||
@entry = Core::Kernel::function_entry( Vm::Block.new("#{name}_entry") ,name )
|
||||
@exit = Core::Kernel::function_exit( Vm::Block.new("#{name}_exit") , name )
|
||||
@body = Block.new("#{name}_body")
|
||||
@reg_count = 0
|
||||
branch_body
|
||||
end
|
||||
attr_reader :args , :entry , :exit , :body , :name
|
||||
attr_accessor :return_type
|
||||
|
Loading…
Reference in New Issue
Block a user