require File.expand_path('../spec_helper', __FILE__)
load_extension("bignum")
def ensure_bignum(n)
raise "Bignum#coerce returned Fixnum" if fixnum_min <= n && n <= fixnum_max
n
end
describe "CApiBignumSpecs" do
before :each do
@s = CApiBignumSpecs.new
@max_long = ensure_bignum(2**(0.size * 8 - 1) - 1)
@min_long = ensure_bignum(-@max_long - 1)
@max_ulong = ensure_bignum(2**(0.size * 8) - 1)
end
describe "rb_big2long" do
it "converts a Bignum" do
@s.rb_big2long(@max_long).should == @max_long
@s.rb_big2long(@min_long).should == @min_long
end
it "raises RangeError if passed Bignum overflow long" do
lambda { @s.rb_big2long(ensure_bignum(@max_long + 1)) }.should raise_error(RangeError)
lambda { @s.rb_big2long(ensure_bignum(@min_long - 1)) }.should raise_error(RangeError)
end
end
describe "rb_big2ll" do
it "converts a Bignum" do
@s.rb_big2ll(@max_long).should == @max_long
@s.rb_big2ll(@min_long).should == @min_long
end
it "raises RangeError if passed Bignum overflow long" do
lambda { @s.rb_big2ll(ensure_bignum(@max_long << 40)) }.should raise_error(RangeError)
lambda { @s.rb_big2ll(ensure_bignum(@min_long << 40)) }.should raise_error(RangeError)
end
end
describe "rb_big2ulong" do
it "converts a Bignum" do
@s.rb_big2ulong(@max_ulong).should == @max_ulong
end
it "wraps around if passed a negative bignum" do
@s.rb_big2ulong(ensure_bignum(@min_long + 1)).should == -(@min_long - 1)
end
it "raises RangeError if passed Bignum overflow long" do
lambda { @s.rb_big2ulong(ensure_bignum(@max_ulong + 1)) }.should raise_error(RangeError)
lambda { @s.rb_big2ulong(ensure_bignum(@min_long - 1)) }.should raise_error(RangeError)
end
it "wraps around if passed a negative bignum" do
@s.rb_big2ulong(ensure_bignum(@min_long)).should == -(@min_long)
end
end
describe "rb_big2dbl" do
it "converts a Bignum to a double value" do
@s.rb_big2dbl(ensure_bignum(Float::MAX.to_i)).eql?(Float::MAX).should == true
end
it "returns Infinity if the number is too big for a double" do
huge_bignum = ensure_bignum(Float::MAX.to_i * 2)
@s.rb_big2dbl(huge_bignum).should == infinity_value
end
it "returns -Infinity if the number is negative and too big for a double" do
huge_bignum = -ensure_bignum(Float::MAX.to_i * 2)
@s.rb_big2dbl(huge_bignum).should == -infinity_value
end
end
describe "rb_big2str" do
it "converts a Bignum to a string with base 10" do
@s.rb_big2str(ensure_bignum(4611686018427387904), 10).eql?("4611686018427387904").should == true
end
it "converts a Bignum to a string with a different base" do
@s.rb_big2str(ensure_bignum(4611686018427387904), 16).eql?("4000000000000000").should == true
end
end
describe "rb_big_cmp" do
it "compares a Bignum with a Bignum" do
@s.rb_big_cmp(bignum_value, bignum_value(1)).should == -1
end
it "compares a Bignum with a Fixnum" do
@s.rb_big_cmp(bignum_value, 5).should == 1
end
end
describe "rb_big_pack" do
it "packs a Bignum into an unsigned long" do
val = @s.rb_big_pack(@max_ulong)
val.should == @max_ulong
end
end
describe "rb_dbl2big" do
it "returns a Fixnum for a Fixnum input value" do
val = @s.rb_dbl2big(2)
val.kind_of?(Fixnum).should == true
val.should == 2
end
it "returns a Fixnum for a Float input value" do
val = @s.rb_dbl2big(2.5)
val.kind_of?(Fixnum).should == true
val.should == 2
end
it "returns a Bignum for a large enough Float input value" do
input = 219238102380912830988.5 val = @s.rb_dbl2big(input)
val.kind_of?(Bignum).should == true
val.should == 219238102380912836608
end
it "raises FloatDomainError for Infinity values" do
inf = 1.0 / 0
lambda { @s.rb_dbl2big(inf) }.should raise_error(FloatDomainError)
end
it "raises FloatDomainError for NaN values" do
nan = 0.0 / 0
lambda { @s.rb_dbl2big(nan) }.should raise_error(FloatDomainError)
end
end
ruby_version_is ""..."2.2" do
describe "RBIGNUM_SIGN" do
it "returns C true if the Bignum has a positive sign" do
@s.RBIGNUM_SIGN(bignum_value).should be_true
end
it "retuns C false if the Bignum has a negative sign" do
@s.RBIGNUM_SIGN(-bignum_value).should be_false
end
end
describe "RBIGNUM_POSITIVE_P" do
it "returns C true if the Bignum has a positive sign" do
@s.RBIGNUM_POSITIVE_P(bignum_value).should be_true
end
it "retuns C false if the Bignum has a negative sign" do
@s.RBIGNUM_POSITIVE_P(-bignum_value).should be_false
end
end
describe "RBIGNUM_NEGATIVE_P" do
it "returns C false if the Bignum has a positive sign" do
@s.RBIGNUM_NEGATIVE_P(bignum_value).should be_false
end
it "retuns C true if the Bignum has a negative sign" do
@s.RBIGNUM_NEGATIVE_P(-bignum_value).should be_true
end
end
describe "RBIGNUM_LEN" do
it "returns the number of BDIGITS needed for the bignum" do
@s.RBIGNUM_LEN(bignum_value * 2).should == 3
end
end
end
end