2014-05-06 11:47:07 +02:00
|
|
|
module Core
|
2014-05-06 11:42:43 +02:00
|
|
|
class Kernel
|
|
|
|
|
|
|
|
#there are no Kernel instances, only class methods.
|
2014-05-06 20:36:28 +02:00
|
|
|
# We use this module syntax to avoid the (ugly) self (also eases searching).
|
2014-05-06 11:42:43 +02:00
|
|
|
module ClassMethods
|
2014-05-13 17:21:24 +02:00
|
|
|
def main_start block
|
2014-05-06 11:42:43 +02:00
|
|
|
#TODO extract args into array of strings
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.main_start block
|
2014-05-14 09:47:30 +02:00
|
|
|
block
|
2014-05-06 11:42:43 +02:00
|
|
|
end
|
2014-05-13 17:21:24 +02:00
|
|
|
def main_exit block
|
2014-05-06 11:42:43 +02:00
|
|
|
# Machine.exit mov r7 , 0 + swi 0
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.main_exit block
|
2014-05-14 09:47:30 +02:00
|
|
|
block
|
2014-05-06 11:42:43 +02:00
|
|
|
end
|
2014-05-13 17:21:24 +02:00
|
|
|
def function_entry block , f_name
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.function_entry block , f_name
|
2014-05-06 11:42:43 +02:00
|
|
|
end
|
2014-05-13 17:21:24 +02:00
|
|
|
def function_exit block , f_name
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.function_exit block , f_name
|
2014-05-06 11:42:43 +02:00
|
|
|
end
|
2014-06-01 13:24:54 +02:00
|
|
|
|
|
|
|
# in ruby, how this goes is
|
|
|
|
# def _get_instance_variable var
|
|
|
|
# i = self.index_of(var)
|
|
|
|
# return at_index(i)
|
|
|
|
# end
|
|
|
|
# The at_index is just "below" the api, somehting we need but don't want to expose, so we can't code the above in ruby
|
|
|
|
def _get_instance_variable context , name = Vm::Integer
|
|
|
|
get_function = Vm::Function.new(:_get_instance_variable , [Vm::Integer , Vm::Integer ] , Vm::Integer )
|
|
|
|
me = get_function.args[0]
|
|
|
|
var_name = get_function.args[1]
|
2014-06-01 20:03:08 +02:00
|
|
|
return_to = get_function.return_type
|
2014-06-01 13:24:54 +02:00
|
|
|
index_function = context.object_space.get_or_create_class(:Object).get_or_create_function(:index_of)
|
|
|
|
b = get_function.body
|
|
|
|
b.push( me )
|
|
|
|
index = b.call( index_function )
|
|
|
|
b.pop(me)
|
2014-06-01 20:03:08 +02:00
|
|
|
return_to.at_index( get_function.body , me , index )
|
|
|
|
get_function.set_return return_to
|
2014-06-01 13:24:54 +02:00
|
|
|
return get_function
|
|
|
|
end
|
|
|
|
|
2014-05-13 20:06:12 +02:00
|
|
|
#TODO this is in the wrong place. It is a function that returns a function object
|
|
|
|
# while all other methods add their code into some block. --> kernel
|
2014-05-19 10:29:18 +02:00
|
|
|
def putstring context , string = Vm::Integer , length = Vm::Integer
|
|
|
|
function = Vm::Function.new(:putstring , [string , length ] , string)
|
2014-05-13 20:06:12 +02:00
|
|
|
block = function.body
|
|
|
|
# should be another level of indirection, ie write(io,str)
|
2014-05-21 18:43:46 +02:00
|
|
|
ret = Vm::RegisterMachine.instance.write_stdout(block)
|
2014-05-25 07:43:07 +02:00
|
|
|
function.set_return ret
|
2014-05-13 20:06:12 +02:00
|
|
|
function
|
2014-05-06 11:42:43 +02:00
|
|
|
end
|
2014-05-15 15:54:23 +02:00
|
|
|
|
2014-05-19 10:29:18 +02:00
|
|
|
def putint context , arg = Vm::Integer
|
2014-05-19 11:18:01 +02:00
|
|
|
putint_function = Vm::Function.new(:putint , [arg] , arg )
|
|
|
|
buffer = Vm::StringConstant.new(" ") # create a buffer
|
2014-05-31 11:52:29 +02:00
|
|
|
context.object_space.add_object buffer # and save it (function local variable: a no no)
|
2014-05-19 11:18:01 +02:00
|
|
|
int = putint_function.args.first
|
2014-05-21 18:06:21 +02:00
|
|
|
moved_int = putint_function.new_local
|
2014-05-31 15:19:44 +02:00
|
|
|
utoa = context.object_space.get_or_create_class(:Object).get_or_create_function(:utoa)
|
2014-05-19 14:44:12 +02:00
|
|
|
b = putint_function.body
|
|
|
|
b.mov( moved_int , int ) #move arg up
|
|
|
|
#b.a buffer => int # string to write to
|
|
|
|
|
|
|
|
b.add( int , buffer ,nil ) # string to write to
|
|
|
|
b.a int + (buffer.length-3) => int
|
|
|
|
b.call( utoa )
|
2014-05-16 22:08:45 +02:00
|
|
|
# And now we "just" have to print it, using the write_stdout
|
2014-05-19 14:44:12 +02:00
|
|
|
b.add( int , buffer , nil ) # string to write to
|
|
|
|
b.mov( moved_int , buffer.length )
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.write_stdout(putint_function.body)
|
2014-05-19 11:18:01 +02:00
|
|
|
putint_function
|
2014-05-15 15:54:23 +02:00
|
|
|
end
|
2014-05-16 18:55:46 +02:00
|
|
|
|
2014-05-16 22:24:19 +02:00
|
|
|
# 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
|
2014-05-19 11:18:01 +02:00
|
|
|
# arguments: string address , integer
|
2014-05-16 18:55:46 +02:00
|
|
|
def utoa context
|
2014-05-19 11:18:01 +02:00
|
|
|
utoa_function = Vm::Function.new(:utoa , [Vm::Integer , Vm::Integer ] , Vm::Integer )
|
|
|
|
str_addr = utoa_function.args[0]
|
|
|
|
number = utoa_function.args[1]
|
2014-05-21 18:06:21 +02:00
|
|
|
remainder = utoa_function.new_local
|
2014-05-21 18:43:46 +02:00
|
|
|
Vm::RegisterMachine.instance.div10( utoa_function.body , number , remainder )
|
2014-05-16 22:24:19 +02:00
|
|
|
# make char out of digit (by using ascii encoding) 48 == "0"
|
2014-05-21 18:06:21 +02:00
|
|
|
b = utoa_function.body.scope binding
|
|
|
|
b.remainder = remainder + 48
|
2014-05-19 14:44:12 +02:00
|
|
|
b.strb( remainder, right: str_addr )
|
|
|
|
b.sub( str_addr, str_addr , 1 )
|
|
|
|
b.cmp( number , 0 )
|
|
|
|
b.callne( utoa_function )
|
2014-05-19 11:18:01 +02:00
|
|
|
return utoa_function
|
2014-05-16 18:55:46 +02:00
|
|
|
end
|
2014-05-19 16:32:41 +02:00
|
|
|
|
2014-05-25 12:34:06 +02:00
|
|
|
# testing method, hand coded fibo, expects arg in 1
|
|
|
|
# result comes in 7
|
2014-05-19 16:32:41 +02:00
|
|
|
# a hand coded version of the fibonachi numbers
|
|
|
|
# not my hand off course, found in the net from a basic introduction
|
|
|
|
def fibo context
|
2014-05-25 12:34:06 +02:00
|
|
|
fibo_function = Vm::Function.new(:fibo , [Vm::Integer] , Vm::Integer )
|
|
|
|
result = Vm::Integer.new(7)
|
|
|
|
int = fibo_function.args[0]
|
2014-05-21 18:27:27 +02:00
|
|
|
count = fibo_function.new_local
|
|
|
|
f1 = fibo_function.new_local
|
|
|
|
f2 = fibo_function.new_local
|
2014-05-21 11:47:40 +02:00
|
|
|
b = fibo_function.body.scope binding
|
|
|
|
|
|
|
|
b.a int == 1
|
|
|
|
b.mov( result, int , condition_code: :le)
|
|
|
|
b.mov( :pc , :lr , condition_code: :le)
|
|
|
|
b.push [ count , f1 , f2 , :lr]
|
|
|
|
b.f1 = 1
|
|
|
|
b.f2 = 0
|
|
|
|
b.count = int - 2
|
|
|
|
|
2014-05-25 12:34:06 +02:00
|
|
|
l = fibo_function.body.new_block("loop").scope binding
|
2014-05-20 09:28:34 +02:00
|
|
|
|
2014-05-21 11:47:40 +02:00
|
|
|
l.f1 = f1 + f2
|
|
|
|
l.f2 = f1 - f2
|
|
|
|
l.count = (count - 1).set_update_status
|
2014-05-25 12:34:06 +02:00
|
|
|
l.bpl( l )
|
|
|
|
l.mov( result , f1 )
|
|
|
|
fibo_function.set_return result
|
2014-05-21 11:47:40 +02:00
|
|
|
l.pop [ count , f1 , f2 , :pc]
|
2014-05-19 16:32:41 +02:00
|
|
|
fibo_function
|
|
|
|
end
|
|
|
|
|
2014-05-02 07:02:25 +02:00
|
|
|
end
|
2014-05-06 11:42:43 +02:00
|
|
|
|
|
|
|
extend ClassMethods
|
2014-05-02 07:02:25 +02:00
|
|
|
end
|
|
|
|
end
|