The goal of this project is to develop a fast and handy multi-precision library that can be used with GPU parallelization frameworks such as CUDA, OpenCL, and Metal. The library should correspond to modern C++ standards, support constexpr expressions, and move semantics.
Important
Project is currently in the testing and development stage to support the Cuda framework. Please be aware that errors and problems may occur. OpenCL support is next in line for development. Metal support is scheduled after some time, due to the presence of significant differences in the framework from Cuda and OpenCL.
Library supports each arithmetic (binary and unary), bitwise, and boolean operations. Various functions from number theory are being added to the library, among which the greatest common divisor, the least common multiplier, and exponentiation by modulo have already been implemented.
Package could be downloaded to project's directory, or accessed directly through CMake:
include(FetchContent)
FetchContent_Declare(AesiMultiprecision
GIT_REPOSITORY https://github.com/Alvov1/Aesi-Multiprecision.git
GIT_TAG main)
FetchContent_MakeAvailable(AesiMultiprecision)
...
target_include_directories(Target PRIVATE ${AesiMultiprecision_SOURCE_DIR})
Further library could be included in project with standard preprocessor command:
#include <Aeu.h>
The library is a header only to avoid difficulties while building. In this case, it can be used simultaneously in C++ and CUDA projects without changing the file extension from .cu to .cpp and backwards. Library supports an object-oriented style of data management. Class operators are overloaded for use in expressions. The number's bitness is passed to the class object as a template parameter and has a default value of 512 bits. It should be a multiple of 32.
Number's initialization could be done with numbers, strings, string-views, string literals, or library objects with different precision. User-defined string literals are planned to be released in the future.
Library supports display operations with STD streams (char and wchar_t based only), along with stream modifications (std::showbase, std::uppercase, std::hex, std::dec, std::oct). std::format support is planned to be released in the future.
#include <iostream>
#include "Aeu.h"
int main() {
Aeu<512> f = 1u;
for(unsigned i = 2; i <= 50; ++i)
f *= i;
std::cout << std::showbase << std::hex << f << std::endl;
}
0x49eebc961ed279b02b1ef4f28d19a84f5973a1d2c7800000000000
__global__ void test() {
const auto tid = blockIdx.x * blockDim.x + threadIdx.x;
if(tid != 0) return;
Aesi<128> amp = 1562144106091796071UL;
printf("Were in kernel thread and number is %lu\n", amp.integralCast<unsigned long>());
}
int main() {
test<<<32, 32>>>();
return cudaSuccess != cudaDeviceSynchronize();
}
Were in kernel thread and number is 1562144106091796071
This project is licensed under the BSD 2-Clause License. See the LICENSE file for details.
It is admissible to use numbers of different precision inside the majority of operations, but it is not recommended cause it leads to redundant copying inside type conversions. Operation-assignment expressions (+=, -=, &=, etc...) require the bitness of the assignment to be greater or equal to the bitness of the assignable. The precision cast operator could be called by a user directly.
Aeu<128> base = "10888869450418352160768000001";
Aeu<96> power = "99990001";
Aeu<256> mod = "8683317618811886495518194401279999999";
cout << Aeu<256>::powm(base.precisionCast<256>(), power.precisionCast<256>(), mod) << endl;
Aeu<128> m128 = "127958277599458332250117";
Aeu<192> m192 = "279256103987149586783914830";
cout << m128.precisionCast<192>() * m192 << endl;
// Cast number of 128 bits to 256 bits, than multiply by number of 160 bits
6680141832773294447513292887050873529 35733130075330889632933652650476631619495985535110
An exception to the rule above is using longer precision boundaries inside functions, susceptible to overflow. As far as the number's precision is fixed on the stage of compilation, functions that require number multiplication or exponentiation may easily lead to overflow:
Aeu<128> base = "340199290171201906239764863559915798527",
power = "340282366920937859000464800151540596704",
modulo = "338953138925230918806032648491249958912";
cout << Aeu<128>::powm(base, power, modulo) << endl; // Overflow !!!
cout << Aeu<256>::powm(base.precisionCast<256>(), // Fine
power,
modulo.precisionCast<256>()) << endl;
*Overflowed* 201007033690655614485250957754150944769
Library is relatively slow in comparison to popular CPU-directional multiple precision libraries: