2014-06-14 00:19:12 +03:00
|
|
|
#integer related kernel functions
|
2017-01-19 09:02:29 +02:00
|
|
|
module Risc
|
2015-06-29 21:03:58 +03:00
|
|
|
module Builtin
|
|
|
|
module Integer
|
|
|
|
module ClassMethods
|
2018-03-18 22:08:35 +05:30
|
|
|
include CompileHelper
|
2015-08-07 16:46:55 +03:00
|
|
|
|
2018-04-19 10:00:55 +03:00
|
|
|
def div4(context)
|
|
|
|
source = "div4"
|
|
|
|
compiler = compiler_for(:Integer,:div4 ,{})
|
2018-04-18 19:12:30 +03:00
|
|
|
builder = compiler.builder(true, compiler.method)
|
2018-04-08 18:52:17 +03:00
|
|
|
me = builder.add_known( :receiver )
|
|
|
|
builder.reduce_int( source , me )
|
2018-04-01 15:13:12 +03:00
|
|
|
two = compiler.use_reg :fixnum , 2
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( source , 2 , two )
|
|
|
|
builder.add_code Risc.op( source , :>> , me , two)
|
|
|
|
builder.add_new_int(source,me , two)
|
|
|
|
builder.add_reg_to_slot( source , two , :message , :return_value)
|
2018-04-01 15:13:12 +03:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2015-10-28 21:37:42 +02:00
|
|
|
return compiler.method
|
2015-08-05 18:49:37 +03:00
|
|
|
end
|
2018-04-19 22:57:31 +03:00
|
|
|
def >( context )
|
2018-04-23 19:39:16 +03:00
|
|
|
comparison( :> )
|
2018-04-19 22:57:31 +03:00
|
|
|
end
|
2018-04-19 22:41:40 +03:00
|
|
|
def <( context )
|
2018-04-23 19:39:16 +03:00
|
|
|
comparison( :< )
|
2018-04-19 22:57:31 +03:00
|
|
|
end
|
2018-04-24 19:45:58 +03:00
|
|
|
def <=( context )
|
|
|
|
comparison( :<= )
|
|
|
|
end
|
|
|
|
def >=( context )
|
|
|
|
comparison( :>= )
|
|
|
|
end
|
2018-04-23 19:39:16 +03:00
|
|
|
def comparison( operator )
|
2018-04-19 22:57:31 +03:00
|
|
|
compiler = compiler_for(:Integer, operator ,{other: :Integer})
|
2018-04-19 22:41:40 +03:00
|
|
|
builder = compiler.builder(true, compiler.method)
|
2018-04-19 22:57:31 +03:00
|
|
|
me , other = builder.self_and_int_arg("#{operator} load receiver and arg")
|
2018-05-01 19:20:16 +03:00
|
|
|
false_label = Risc.label(compiler.method , "false_label_#{builder.object_id.to_s(16)}")
|
|
|
|
merge_label = Risc.label(compiler.method , "merge_label_#{builder.object_id.to_s(16)}")
|
2018-04-19 22:57:31 +03:00
|
|
|
builder.reduce_int( "#{operator} fix me", me )
|
|
|
|
builder.reduce_int( "#{operator} fix arg", other )
|
2018-04-24 19:45:58 +03:00
|
|
|
if(operator.to_s.start_with?('<') )
|
2018-04-23 19:39:16 +03:00
|
|
|
me , other = other , me
|
|
|
|
end
|
2018-04-19 22:57:31 +03:00
|
|
|
builder.add_code Risc.op( "#{operator} operator", :- , me , other)
|
2018-04-23 19:39:16 +03:00
|
|
|
builder.add_code IsMinus.new( "#{operator} if", false_label)
|
2018-04-24 19:45:58 +03:00
|
|
|
if(operator.to_s.length == 1)
|
|
|
|
builder.add_code IsZero.new( "#{operator} if", false_label)
|
|
|
|
end
|
2018-04-19 22:57:31 +03:00
|
|
|
builder.add_load_constant("#{operator} new int", Parfait.object_space.true_object , other)
|
2018-04-19 22:41:40 +03:00
|
|
|
builder.add_code Risc::Branch.new("jump over false", merge_label)
|
|
|
|
builder.add_code false_label
|
2018-04-19 22:57:31 +03:00
|
|
|
builder.add_load_constant("#{operator} new int", Parfait.object_space.false_object , other)
|
2018-04-19 22:41:40 +03:00
|
|
|
builder.add_code merge_label
|
2018-04-19 22:57:31 +03:00
|
|
|
builder.add_reg_to_slot( "#{operator} save ret" , other , :message , :return_value)
|
2018-04-19 22:41:40 +03:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
|
|
|
return compiler.method
|
|
|
|
end
|
2018-04-01 15:13:12 +03:00
|
|
|
def putint(context)
|
2018-03-18 22:08:35 +05:30
|
|
|
compiler = compiler_for(:Integer,:putint ,{})
|
2018-04-01 15:17:16 +03:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2015-10-28 21:37:42 +02:00
|
|
|
return compiler.method
|
2015-08-05 18:49:37 +03:00
|
|
|
end
|
2018-04-19 22:13:52 +03:00
|
|
|
def operator_method( op_sym )
|
2018-04-01 15:26:53 +03:00
|
|
|
compiler = compiler_for(:Integer, op_sym ,{other: :Integer})
|
2018-04-18 19:12:30 +03:00
|
|
|
builder = compiler.builder(true, compiler.method)
|
2018-04-19 22:13:52 +03:00
|
|
|
me , other = builder.self_and_int_arg(op_sym.to_s + "load receiver and arg")
|
|
|
|
builder.reduce_int( op_sym.to_s + " fix me", me )
|
|
|
|
builder.reduce_int( op_sym.to_s + " fix arg", other )
|
|
|
|
builder.add_code Risc.op( op_sym.to_s + " operator", op_sym , me , other)
|
|
|
|
builder.add_new_int(op_sym.to_s + " new int", me , other)
|
|
|
|
builder.add_reg_to_slot( op_sym.to_s + "save ret" , other , :message , :return_value)
|
2018-04-01 14:56:01 +03:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2018-03-30 17:09:02 +03:00
|
|
|
return compiler.method
|
|
|
|
end
|
2018-03-24 16:51:26 +02:00
|
|
|
def div10( context )
|
2018-04-01 14:50:13 +03:00
|
|
|
s = "div_10 "
|
2018-03-18 22:08:35 +05:30
|
|
|
compiler = compiler_for(:Integer,:div10 ,{})
|
2018-04-18 19:12:30 +03:00
|
|
|
builder = compiler.builder(true, compiler.method)
|
2018-04-01 15:13:12 +03:00
|
|
|
#FIX: this could load receiver once, reduce and then transfer twice
|
2018-04-08 18:52:17 +03:00
|
|
|
me = builder.add_known( :receiver )
|
|
|
|
tmp = builder.add_known( :receiver )
|
|
|
|
q = builder.add_known( :receiver )
|
|
|
|
builder.reduce_int( s , me )
|
|
|
|
builder.reduce_int( s , tmp )
|
|
|
|
builder.reduce_int( s , q )
|
2018-04-01 14:50:13 +03:00
|
|
|
const = compiler.use_reg :fixnum , 1
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( s , 1 , const )
|
2015-11-21 14:20:25 +02:00
|
|
|
# int tmp = self >> 1
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const)
|
2015-11-21 14:20:25 +02:00
|
|
|
# int q = self >> 2
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( s , 2 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , q , const)
|
2015-11-21 14:20:25 +02:00
|
|
|
# q = q + tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = q >> 4
|
2018-04-08 18:52:17 +03: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 14:20:25 +02:00
|
|
|
# q = q + tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = q >> 8
|
2018-04-08 18:52:17 +03: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 14:20:25 +02:00
|
|
|
# q = q + tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = q >> 16
|
2018-04-08 18:52:17 +03: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 14:20:25 +02:00
|
|
|
# q = q + tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2015-11-21 14:20:25 +02:00
|
|
|
# q = q >> 3
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( s , 3 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , q , const)
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = q * 10
|
2018-04-08 18:52:17 +03: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 14:20:25 +02:00
|
|
|
# tmp = self - tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :- , me , tmp )
|
|
|
|
builder.add_transfer( s , me , tmp)
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = tmp + 6
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( s , 6 , const)
|
|
|
|
builder.add_code Risc.op( s , :+ , tmp , const )
|
2015-11-21 14:20:25 +02:00
|
|
|
# tmp = tmp >> 4
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_load_data( s , 4 , const)
|
|
|
|
builder.add_code Risc.op( s , :>> , tmp , const )
|
2015-11-21 14:20:25 +02:00
|
|
|
# return q + tmp
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_code Risc.op( s , :+ , q , tmp )
|
2018-04-01 14:01:17 +03:00
|
|
|
|
2018-04-08 18:52:17 +03:00
|
|
|
builder.add_new_int(s,q , tmp)
|
|
|
|
builder.add_reg_to_slot( s , tmp , :message , :return_value)
|
2018-04-01 14:01:17 +03:00
|
|
|
|
2018-03-24 16:51:26 +02:00
|
|
|
compiler.add_mom( Mom::ReturnSequence.new)
|
2015-11-21 14:20:25 +02:00
|
|
|
return compiler.method
|
|
|
|
end
|
2015-06-29 21:03:58 +03:00
|
|
|
end
|
|
|
|
extend ClassMethods
|
2014-09-11 15:08:56 +03:00
|
|
|
end
|
2014-06-14 00:19:12 +03:00
|
|
|
end
|
2015-05-24 13:31:33 +03:00
|
|
|
end
|