2014-06-13 23:19:12 +02:00
|
|
|
#integer related kernel functions
|
2017-01-19 08:02:29 +01:00
|
|
|
module Risc
|
2015-06-29 20:03:58 +02:00
|
|
|
module Builtin
|
|
|
|
module Integer
|
|
|
|
module ClassMethods
|
2018-03-18 17:38:35 +01:00
|
|
|
include CompileHelper
|
2015-08-07 15:46:55 +02:00
|
|
|
|
2018-04-19 09:00:55 +02:00
|
|
|
def div4(context)
|
|
|
|
compiler = compiler_for(:Integer,:div4 ,{})
|
2018-08-08 11:02:24 +02:00
|
|
|
compiler.compiler_builder(compiler.source).build do
|
|
|
|
integer << message[:receiver]
|
2018-08-08 14:49:47 +02:00
|
|
|
integer.reduce_int
|
2018-08-08 11:02:24 +02:00
|
|
|
integer_reg << 2
|
|
|
|
integer.op :>> , integer_reg
|
|
|
|
add_new_int("div4", integer , integer_reg)
|
|
|
|
message[:return_value] << integer_reg
|
|
|
|
end
|
2018-04-01 14:13:12 +02:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-06-30 22:16:17 +02:00
|
|
|
return compiler
|
2015-08-05 17:49:37 +02:00
|
|
|
end
|
2018-04-19 21:57:31 +02:00
|
|
|
def >( context )
|
2018-04-23 18:39:16 +02:00
|
|
|
comparison( :> )
|
2018-04-19 21:57:31 +02:00
|
|
|
end
|
2018-04-19 21:41:40 +02:00
|
|
|
def <( context )
|
2018-04-23 18:39:16 +02:00
|
|
|
comparison( :< )
|
2018-04-19 21:57:31 +02:00
|
|
|
end
|
2018-04-24 18:45:58 +02:00
|
|
|
def <=( context )
|
|
|
|
comparison( :<= )
|
|
|
|
end
|
|
|
|
def >=( context )
|
|
|
|
comparison( :>= )
|
|
|
|
end
|
2018-04-23 18:39:16 +02:00
|
|
|
def comparison( operator )
|
2018-04-19 21:57:31 +02:00
|
|
|
compiler = compiler_for(:Integer, operator ,{other: :Integer})
|
2018-07-09 18:32:17 +02:00
|
|
|
builder = compiler.compiler_builder(compiler.source)
|
2018-08-08 19:53:06 +02:00
|
|
|
builder.build do
|
|
|
|
integer << message[:receiver]
|
|
|
|
integer_reg << message[:arguments]
|
|
|
|
integer_reg << integer_reg[ 1]
|
|
|
|
integer.reduce_int
|
|
|
|
integer_reg.reduce_int
|
|
|
|
swap_names(:integer , :integer_reg) if(operator.to_s.start_with?('<') )
|
|
|
|
integer.op :- , integer_reg
|
|
|
|
if_minus false_label
|
|
|
|
if_zero( false_label ) if operator.to_s.length == 1
|
|
|
|
object << Parfait.object_space.true_object
|
|
|
|
branch merge_label
|
|
|
|
add_code false_label
|
|
|
|
object << Parfait.object_space.false_object
|
|
|
|
add_code merge_label
|
|
|
|
message[:return_value] << object
|
2018-04-24 18:45:58 +02:00
|
|
|
end
|
2018-04-19 21:41:40 +02:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-06-30 22:16:17 +02:00
|
|
|
return compiler
|
2018-04-19 21:41:40 +02:00
|
|
|
end
|
2018-08-08 19:53:06 +02:00
|
|
|
|
2018-04-01 14:13:12 +02:00
|
|
|
def putint(context)
|
2018-03-18 17:38:35 +01:00
|
|
|
compiler = compiler_for(:Integer,:putint ,{})
|
2018-04-01 14:17:16 +02:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-06-30 22:16:17 +02:00
|
|
|
return compiler
|
2015-08-05 17:49:37 +02:00
|
|
|
end
|
2018-04-19 21:13:52 +02:00
|
|
|
def operator_method( op_sym )
|
2018-04-01 14:26:53 +02:00
|
|
|
compiler = compiler_for(:Integer, op_sym ,{other: :Integer})
|
2018-07-09 18:32:17 +02:00
|
|
|
builder = compiler.compiler_builder(compiler.source)
|
2018-08-08 14:49:47 +02:00
|
|
|
builder.build do
|
|
|
|
integer << message[:receiver]
|
|
|
|
integer_reg << message[:arguments]
|
|
|
|
integer_reg << integer_reg[ 1]
|
|
|
|
integer.reduce_int
|
|
|
|
integer_reg.reduce_int
|
|
|
|
integer.op op_sym , integer_reg
|
|
|
|
add_new_int op_sym.to_s , integer , integer_reg
|
|
|
|
message[:return_value] << integer_reg
|
|
|
|
end
|
2018-04-01 13:56:01 +02:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-06-30 22:16:17 +02:00
|
|
|
return compiler
|
2018-03-30 16:09:02 +02:00
|
|
|
end
|
2018-08-09 20:10:05 +02:00
|
|
|
|
|
|
|
def add_receiver(builder)
|
|
|
|
message = Risc.message_reg
|
|
|
|
ret_type = builder.compiler.receiver_type
|
|
|
|
ret = builder.compiler.use_reg( ret_type )
|
|
|
|
builder.add_slot_to_reg(" load self" , message , :receiver , ret )
|
|
|
|
builder.add_slot_to_reg( "int -> fix" , ret , Parfait::Integer.integer_index , ret)
|
|
|
|
return ret
|
|
|
|
end
|
|
|
|
|
2018-03-24 15:51:26 +01:00
|
|
|
def div10( context )
|
2018-04-01 13:50:13 +02:00
|
|
|
s = "div_10 "
|
2018-03-18 17:38:35 +01:00
|
|
|
compiler = compiler_for(:Integer,:div10 ,{})
|
2018-07-09 18:32:17 +02:00
|
|
|
builder = compiler.compiler_builder(compiler.source)
|
2018-04-01 14:13:12 +02:00
|
|
|
#FIX: this could load receiver once, reduce and then transfer twice
|
2018-08-09 20:10:05 +02:00
|
|
|
me = add_receiver( builder )
|
|
|
|
tmp = add_receiver( builder )
|
|
|
|
q = add_receiver( builder )
|
2018-07-16 10:23:09 +02:00
|
|
|
const = compiler.use_reg :fixnum , value: 1
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 1 , const )
|
2015-11-21 13:20:25 +01:00
|
|
|
# int tmp = self >> 1
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# int q = self >> 2
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 2 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , q , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# q = q + tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = q >> 4
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 4 , const)
|
|
|
|
builder.add_transfer( s, q , tmp)
|
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# q = q + tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = q >> 8
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 8 , const)
|
|
|
|
builder.add_transfer( s, q , tmp)
|
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# q = q + tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = q >> 16
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 16 , const)
|
|
|
|
builder.add_transfer( s, q , tmp)
|
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# q = q + tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 13:20:25 +01:00
|
|
|
# q = q >> 3
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 3 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , q , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = q * 10
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 10 , const)
|
|
|
|
builder.add_transfer( s, q , tmp)
|
|
|
|
builder.add_code Risc.op( s , :* , tmp , const)
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = self - tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :- , me , tmp )
|
|
|
|
builder.add_transfer( s , me , tmp)
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = tmp + 6
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 6 , const)
|
|
|
|
builder.add_code Risc.op( s , :+ , tmp , const )
|
2015-11-21 13:20:25 +01:00
|
|
|
# tmp = tmp >> 4
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_load_data( s , 4 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const )
|
2015-11-21 13:20:25 +01:00
|
|
|
# return q + tmp
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2018-04-01 13:01:17 +02:00
|
|
|
|
2018-04-08 17:52:17 +02:00
|
|
|
builder.add_new_int(s,q , tmp)
|
2018-07-15 14:14:38 +02:00
|
|
|
builder.add_reg_to_slot( s , tmp , Risc.message_reg , :return_value)
|
2018-04-01 13:01:17 +02:00
|
|
|
|
2018-03-24 15:51:26 +01:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-06-30 22:16:17 +02:00
|
|
|
return compiler
|
2015-11-21 13:20:25 +01:00
|
|
|
end
|
2015-06-29 20:03:58 +02:00
|
|
|
end
|
|
|
|
extend ClassMethods
|
2014-09-11 14:08:56 +02:00
|
|
|
end
|
2014-06-13 23:19:12 +02:00
|
|
|
end
|
2015-05-24 12:31:33 +02:00
|
|
|
end
|