rpl.rb/lib/core/operations.rb

159 lines
4.6 KiB
Ruby
Raw Normal View History

module Rpl
2021-12-07 15:50:58 +01:00
module Lang
module Core
module_function
# addition
def add( stack )
addable = %i[numeric string name]
stack, args = Rpl::Lang::Core.stack_extract( stack, [addable, addable] )
result = { type: case args[1][:type]
when :name
:name
when :string
:string
2021-12-07 15:50:58 +01:00
when :numeric
if args[0][:type] == :numeric
:numeric
else
:string
end
end }
args.each do |elt|
elt[:value] = elt[:value][1..-2] unless elt[:type] == :numeric
end
result[:value] = case result[:type]
when :name
"'#{args[1][:value]}#{args[0][:value]}'"
when :string
"\"#{args[1][:value]}#{args[0][:value]}\""
when :numeric
args[1][:value] + args[0][:value]
end
2021-12-07 15:50:58 +01:00
result[:base] = 10 if result[:type] == :numeric # TODO: what if operands have other bases ?
2021-12-07 15:50:58 +01:00
stack << result
end
2021-12-07 15:50:58 +01:00
# substraction
def subtract( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric], %i[numeric]] )
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[1][:value] - args[0][:value] }
end
2021-12-07 15:50:58 +01:00
# negation
def negate( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[0][:value] * -1 }
end
2021-12-07 15:50:58 +01:00
# multiplication
def multiply( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric], %i[numeric]] )
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[1][:value] * args[0][:value] }
end
2021-12-07 15:50:58 +01:00
# division
def divide( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric], %i[numeric]] )
2021-12-07 15:50:58 +01:00
raise 'Division by 0' if args[0][:value].zero?
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[1][:value] / args[0][:value] }
end
2021-12-07 15:50:58 +01:00
# inverse
def inverse( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
2021-12-07 15:50:58 +01:00
raise 'Division by 0' if args[0][:value].zero?
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: 1.0 / args[0][:value] }
end
2021-12-07 15:50:58 +01:00
# power
def power( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric], %i[numeric]] )
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[1][:value]**args[0][:value] }
end
2021-11-18 15:58:59 +01:00
2021-12-07 15:50:58 +01:00
# rpn_square root
def sqrt( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
2021-11-18 15:58:59 +01:00
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: BigMath.sqrt( BigDecimal( args[0][:value] ), Rpl::Lang::Core.precision ) }
end
2021-11-18 15:58:59 +01:00
2021-12-07 15:50:58 +01:00
# rpn_square
def sq( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
2021-11-18 15:58:59 +01:00
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[0][:value] * args[0][:value] }
end
2021-11-18 15:58:59 +01:00
2021-12-07 15:50:58 +01:00
# absolute value
def abs( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
2021-12-07 15:50:58 +01:00
stack << { type: :numeric, base: 10,
value: args[0][:value].abs }
end
2021-12-07 15:50:58 +01:00
# decimal representation
def dec( stack )
base( stack << { type: :numeric, base: 10, value: 10 } )
end
2021-12-07 15:50:58 +01:00
# hexadecimal representation
def hex( stack )
base( stack << { type: :numeric, base: 10, value: 16 } )
end
2021-12-07 15:50:58 +01:00
# binary representation
def bin( stack )
base( stack << { type: :numeric, base: 10, value: 2 } )
end
2021-12-07 15:50:58 +01:00
# arbitrary base representation
def base( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric], %i[numeric]] )
2021-12-07 15:50:58 +01:00
args[1][:base] = args[0][:value]
2021-12-07 15:50:58 +01:00
stack << args[1]
end
2021-12-07 15:50:58 +01:00
# 1 if number at stack level 1 is > 0, 0 if == 0, -1 if <= 0
def sign( stack )
stack, args = Rpl::Lang::Core.stack_extract( stack, [%i[numeric]] )
value = if args[0][:value].positive?
1
elsif args[0][:value].negative?
-1
else
0
end
stack << { type: :numeric, base: 10,
value: value }
end
end
end
end