-
-
Notifications
You must be signed in to change notification settings - Fork 5.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Exploit mpq functions for Rational{BigInt}? #9826
Comments
Alternatively, we could try to just make |
There are other issues here. The mpq_t interface is quite standard and supported by many libraries (including numerous libraries written by people interested in using Julia). If you use a different rational format to GMP, you no longer get the benefit of any new rational function implementations GMP provides, or easy/efficient interface to those libraries that represent rationals as mpq_t's or at least interface to them. GMP rationals are extremely efficient for numerous reasons:
No matter how sophisticated Julia becomes, it isn't going to approach that kind of optimised performance if it uses a generic implementation to handle rational arithmetic. GMP is twenty years old at least and meticulously and insanely optimised. |
I know this is quite an old issue but just for reference, there is now a wrapper around GMP rational functions in https://github.com/Liozou/BigRationals.jl, in case people need to use it. |
Indeed—GMP objects need to be finalized and immutable objects cannot be finalized. |
Can this be closed? |
It might be possible to support this, but you would have to convert from a |
It looks like conversion would be extremely cheap, since it just involves copying 6 scalar fields back and forth (compare BigInt to BigRational). |
I wasn't sure if we needed to init the mpq object separately from the mpz ones, but it appears that is not the case. Here is a simple proof-of-concept: import Base.GMP: Limb
mutable struct MPQ
num_alloc::Cint
num_size::Cint
num_d::Ptr{Limb}
den_alloc::Cint
den_size::Cint
den_d::Ptr{Limb}
end
MPQ(x::Rational{BigInt}) = MPQ(x.num.alloc, x.num.size, x.num.d,
x.den.alloc, x.den.size, x.den.d)
function update!(x::Rational{BigInt}, xq::MPQ)
x.num.alloc = xq.num_alloc
x.num.size = xq.num_size
x.num.d = xq.num_d
x.den.alloc = xq.den_alloc
x.den.size = xq.den_size
x.den.d = xq.den_d
end
const mpq_t = Ref{MPQ}
function add(x::Rational{BigInt}, y::Rational{BigInt})
z = Rational{BigInt}(0)
zq = MPQ(z)
ccall((:__gmpq_add, :libgmp), Cvoid,
(mpq_t,mpq_t,mpq_t), zq, MPQ(x), MPQ(y))
update!(z,zq)
return z
end julia> add(big(2//3), big(2//3))
4//3 |
One difference is that |
This is quite a fragile "interface". As soon you call |
Good point. The easiest solution would be to tack on |
As discussed on julia-dev, there is some performance advantage to using the GMP
mpq
functions forRational{BigInt}
operations.The easiest way to do this would be:
BigInt
animmutable
type. (It currently has immutable semantics anyway.) This way,Rational{BigInt}
would be binary compatible with GMP'smpq_t
(a pointer to an__mpq_struct
consisting of the numerator__mpz_struct
followed by the denominator), since ourBigInt
type is already a mirror of__mpz_struct
.+
etc. methods forRational{BigInt}
that call GMPmpq
functions.I get the impression that the main reason that
BigInt
is atype
and notimmutable
is that this makes it easier to pass by reference to GMP functions. So changing this toimmutable
would benefit greatly from a better way to passccall
"out" arguments by reference, as discussed in #2322.Alternatively, if you want to leave
BigInt
as-is, then one needs an easy way to convertBigInt
to/from animmutable
version thereof, and this requires us to add an additional "internal" constructorfunction BigInt(alloc::Cint, size::Cint, d::Ptr{Limb})
.The text was updated successfully, but these errors were encountered: