31#ifndef AEU_MULTIPRECISION
32#define AEU_MULTIPRECISION
39#define gpu __host__ __device__
40 #include <cuda/std/utility>
41 #include <cuda/std/array>
49#if defined AESI_UNSAFE
50 #warning Enabled nightly mode for the library. Functions and methods input arguments are not checked for validity. Be really gentle
53#ifdef AESI_CRYPTOPP_INTEGRATION
54#include <cryptopp/integer.h>
57#ifdef AESI_GMP_INTEGRATION
63 using block = uint32_t;
65 constexpr inline std::size_t bitsInByte = 8;
66 constexpr inline std::size_t blockBitLength =
sizeof(block) * bitsInByte;
67 constexpr inline uint64_t blockBase = 1ULL << blockBitLength;
68 constexpr inline block blockMax = 0xff'ff'ff'ffu;
74 enum class Comparison { equal = 0, less = 1, greater = 2, equivalent = 3 };
82template <std::
size_t bitness = 512>
requires (bitness % blockBitLength == 0)
84 static_assert(bitness >
sizeof(uint64_t),
"Use built-in types for numbers 64-bit or less.");
86 static constexpr std::size_t blocksNumber = bitness / blockBitLength;
89 template <
typename T1,
typename T2>
90 using pair = cuda::std::pair<T1, T2>;
91 using blockLine = cuda::std::array<block, blocksNumber>;
93 template<
typename T1,
typename T2>
94 using pair = std::pair<T1, T2>;
95 using blockLine = std::array<block, blocksNumber>;
112 gpu
static constexpr auto addLine(blockLine& dst,
const blockLine& src)
noexcept -> uint64_t {
113 uint64_t carryOut = 0;
114 for (std::size_t i = 0; i < blocksNumber; ++i) {
115 uint64_t sum =
static_cast<uint64_t
>(dst[i]) +
static_cast<uint64_t
>(src[i]) + carryOut;
116 carryOut = sum / blockBase; dst[i] =
static_cast<block
>(sum % blockBase);
126 gpu
static constexpr auto makeComplement(blockLine& line)
noexcept -> uint64_t {
127 uint64_t carryOut = 1;
128 for(std::size_t i = 0; i < blocksNumber; ++i) {
129 const uint64_t sum = blockBase - 1ULL -
static_cast<uint64_t
>(line[i]) + carryOut;
130 carryOut = sum / blockBase; line[i] =
static_cast<block
>(sum % blockBase);
141 gpu
constexpr Aeu() noexcept = default;
146 gpu constexpr ~
Aeu() noexcept = default;
151 gpu constexpr
Aeu(const
Aeu& copy) noexcept = default;
157 gpu constexpr
Aeu& operator=(const
Aeu& other) = default;
165 template <typename Integral> requires (std::is_unsigned_v<Integral>)
166 gpu constexpr
Aeu(Integral value) noexcept {
168 uint64_t tValue = (value < 0 ? static_cast<uint64_t>(value * -1) :
static_cast<uint64_t
>(value));
169 for (std::size_t i = 0; i < blocksNumber; ++i) {
170 blocks[i] =
static_cast<block
>(tValue % blockBase);
174 makeComplement(blocks);
185 template <
typename Char>
requires (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t>)
186 gpu
constexpr Aeu(
const Char* data, std::size_t size) noexcept :
Aeu {} {
187 if(size == 0)
return;
189 constexpr const Char* characters = [] {
190 if constexpr (std::is_same_v<char, Char>) {
191 return "09aAfFoObBxX";
193 return L
"09aAfFoObBxX";
196 std::size_t position = 0;
198 if constexpr (std::is_same_v<Char, char>) {
199 for(; position < size && !std::isalnum(data[position]); ++position) ;
201 for(; position < size && !std::iswalnum(data[position]); ++position) ;
207 const auto base = [&data, &size, &position, &characters] {
208 if (data[position] == characters[0] && size > position + 1) {
209 switch (data[position + 1]) {
212 position += 2;
return 2u;
215 position += 2;
return 8u;
218 position += 2;
return 16u;
225 for(; position < size; ++position) {
226 const auto digit = [] (Char ch) {
227 if(characters[0] <= ch && ch <= characters[1])
228 return static_cast<unsigned>(ch) -
static_cast<unsigned>(characters[0]);
229 if(characters[2] <= ch && ch <= characters[4])
230 return static_cast<unsigned>(ch) -
static_cast<unsigned>(characters[2]) + 10u;
231 if(characters[3] <= ch && ch <= characters[5])
232 return static_cast<unsigned>(ch) -
static_cast<unsigned>(characters[3]) + 10u;
248 template <
typename Char, std::
size_t arrayLength>
requires (arrayLength > 1 && (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t>))
249 gpu
constexpr Aeu(
const Char (&literal)[arrayLength]) noexcept :
Aeu(literal, arrayLength) {}
256 template <
typename String,
typename Char =
typename String::value_type>
requires (std::is_same_v<std::basic_string<Char>,
257 std::decay_t<String>> || std::is_same_v<std::basic_string_view<Char>, std::decay_t<String>>)
258 gpu
constexpr Aeu(String&& stringView) noexcept :
Aeu(stringView.data(), stringView.size()) {}
260 template <
typename String,
typename Char =
typename String::value_type>
requires (std::is_same_v<std::basic_string<Char>,
261 std::decay_t<String>> || std::is_same_v<std::basic_string_view<Char>, std::decay_t<String>>)
262 gpu
constexpr Aeu(
const String& stringView) noexcept :
Aeu(stringView.data(), stringView.size()) {}
264#ifdef AESI_CRYPTOPP_INTEGRATION
269 constexpr Aeu(
const CryptoPP::Integer& value):
Aeu {} {
270 const auto byteCount = value.ByteCount();
271 if(byteCount * 8 > bitness)
272 throw std::invalid_argument(
"Accessed overflow on construction object from CryptoPP::Integer");
273 for(std::size_t i = 0; i < byteCount; ++i)
274 setByte(i, value.GetByte(i));
278#ifdef AESI_GMP_INTEGRATION
283 constexpr Aeu(
const mpz_class& value) :
Aeu {} {
284 const auto bitLength = mpz_sizeinbase(value.get_mpz_t(), 2);
285 auto tBlocksNumber = 1 + (bitLength - 1) /
sizeof(block) * 8;
287 std::basic_string<block> buffer (tBlocksNumber, 0u);
288 mpz_export(buffer.data(), &tBlocksNumber, -1,
sizeof(block), -1, 0, value.get_mpz_t());
290 for (std::size_t i = 0; i < tBlocksNumber; ++i)
291 setBlock(i, buffer[i]);
313 Aeu result = *
this; makeComplement(result.blocks);
return result;
321 for(std::size_t i = 0; i < blocksNumber; ++i) {
322 if(blocks[i] < blockMax) {
325 }
else blocks[i] = 0u;
341 for(std::size_t i = 0; i < blocksNumber; ++i) {
345 blocks[i] = blockMax;
365 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned>) [[nodiscard]]
366 gpu
constexpr friend auto operator+(
const Aeu& addition, Unsigned addendum)
noexcept ->
Aeu {
367 Aeu result = addition; result += addendum;
return result;
378 Aeu result = addition; result += addendum;
return result;
387 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned>)
388 gpu
constexpr friend auto operator+=(
Aeu& addition, Unsigned addendum)
noexcept ->
Aeu& {
389 for(std::size_t i = 0; i < blocksNumber; ++i) {
390 const auto currentSum =
static_cast<uint64_t
>(addition.blocks[i]) +
static_cast<uint64_t
>(addendum);
391 addendum = currentSum / blockBase; addition.blocks[i] = currentSum % blockBase;
403 addLine(addition.blocks, addendum.blocks);
return addition;
417 Aeu result = subtraction; result -= subtrahend;
return result;
427 return subtraction += -subtrahend;
439 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned>) [[nodiscard]]
440 gpu
constexpr friend auto operator*(
Aeu& multiplication, Unsigned factor)
noexcept ->
Aeu {
441 Aeu result = multiplication; result *= factor;
return result;
452 Aeu result = multiplication; result *= factor;
return result;
462 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned>)
463 gpu
constexpr friend auto operator*=(
Aeu& multiplication, Unsigned factor)
noexcept ->
Aeu& {
464 if constexpr (std::is_same_v<Unsigned, uint64_t>) {
466 const auto smallerLength = (factor > blockMax ? 2UL : 1UL);
469 for(std::size_t i = 0; i < longerLength; ++i) {
470 const uint64_t tBlock = multiplication.blocks[i];
471 uint64_t carryOut = 0;
473 for(std::size_t j = 0; j < smallerLength && j < buffer.size() - i; ++j) {
474 const auto product = tBlock * (factor >> blockBitLength * j & 0x00'00'00'00'ff'ff'ff'ff) + carryOut;
475 const auto block =
static_cast<uint64_t
>(buffer[i + j]) + product % blockBase;
476 carryOut = product / blockBase + block / blockBase;
477 buffer[i + j] = block % blockBase;
480 if(smallerLength + i < buffer.size())
481 buffer[smallerLength + i] += carryOut;
484 multiplication.blocks = buffer;
485 return multiplication;
487 uint64_t carryOut = 0;
488 for (std::size_t i = 0; i < blocksNumber; ++i) {
489 const auto product =
static_cast<uint64_t
>(factor) *
static_cast<uint64_t
>(multiplication.blocks[i]) + carryOut;
490 multiplication.blocks[i] = product % blockBase; carryOut = product / blockBase;
492 return multiplication;
503 constexpr auto multiplyLines = [] (
const blockLine& longerLine,
const std::size_t longerLength,
504 const blockLine& smallerLine,
const std::size_t smallerLength) {
507 for(std::size_t i = 0; i < longerLength; ++i) {
508 const uint64_t tBlock = longerLine[i];
509 uint64_t carryOut = 0;
511 for(std::size_t j = 0; j < smallerLength && j < buffer.size() - i; ++j) {
512 const auto product = tBlock *
static_cast<uint64_t
>(smallerLine[j]) + carryOut;
513 const auto block =
static_cast<uint64_t
>(buffer[i + j]) + product % blockBase;
514 carryOut = product / blockBase + block / blockBase;
515 buffer[i + j] = block % blockBase;
518 if(smallerLength < blocksNumber && smallerLength + i < buffer.size())
519 buffer[smallerLength + i] += carryOut;
526 if(
const std::size_t valueLength = factor.filledBlocksNumber(); thisLength > valueLength)
527 multiplication.blocks = multiplyLines(multiplication.blocks, thisLength, factor.blocks, valueLength);
529 multiplication.blocks = multiplyLines(factor.blocks, valueLength, multiplication.blocks, thisLength);
531 return multiplication;
546 Aeu quotient, _; divide(division, divisor, quotient, _);
return quotient;
556 gpu
constexpr friend auto operator/=(
Aeu& division,
const Aeu& divisor)
noexcept ->
Aeu& {
return division = division / divisor; }
569 Aeu _, remainder; divide(modulation, modulo, _, remainder);
return remainder;
579 return modulation = modulation % modulo;
593 for(std::size_t i = 0; i < blocksNumber; ++i)
594 result.blocks[i] = ~blocks[i];
606 Aeu result = left; result ^= right;
return result;
616 for(std::size_t i = 0; i < blocksNumber; ++i)
617 left.blocks[i] ^= right.blocks[i];
629 Aeu result = left; result &= right;
return result;
639 for(std::size_t i = 0; i < blocksNumber; ++i)
640 left.blocks[i] &= right.blocks[i];
652 Aeu result = left; result |= right;
return result;
662 for(std::size_t i = 0; i < blocksNumber; ++i)
663 left.blocks[i] |= right.blocks[i];
674 template <
typename Un
signed>
requires (std::is_integral_v<Unsigned> && std::is_unsigned_v<Unsigned>) [[nodiscard]]
675 gpu
constexpr friend auto operator<<(
const Aeu& value, Unsigned bitShift)
noexcept ->
Aeu {
676 Aeu result = value; result <<= bitShift;
return result;
686 template <
typename Un
signed>
requires (std::is_integral_v<Unsigned> && std::is_unsigned_v<Unsigned>)
687 gpu
constexpr friend auto operator<<=(
Aeu& value, Unsigned bitShift)
noexcept ->
Aeu& {
688 if(bitShift >= bitness || bitShift == 0)
return value;
690 const std::size_t quotient = bitShift / blockBitLength, remainder = bitShift % blockBitLength;
691 const block stamp = (1UL << (blockBitLength - remainder)) - 1;
693 for (
long long i = blocksNumber - 1; i >= (quotient + (remainder ? 1 : 0)); --i)
694 value.blocks[i] = (value.blocks[i - quotient] & stamp) << remainder
695 | ((value.blocks[i - quotient - (remainder ? 1 : 0)] & ~stamp) >> (blockBitLength - remainder) % blockBitLength);
697 value.blocks[quotient] = (value.blocks[0] & stamp) << remainder;
699 for (std::size_t i = 0; i < quotient; ++i)
711 template <
typename Un
signed>
requires (std::is_integral_v<Unsigned> && std::is_unsigned_v<Unsigned>) [[nodiscard]]
712 gpu
constexpr friend auto operator>>(
const Aeu& value, Unsigned bitShift)
noexcept ->
Aeu {
713 Aeu result = value; result >>= bitShift;
return result;
723 template <
typename Un
signed>
requires (std::is_integral_v<Unsigned> && std::is_unsigned_v<Unsigned>)
724 gpu
constexpr friend auto operator>>=(
Aeu& value, Unsigned bitShift)
noexcept ->
Aeu& {
725 if(bitShift >= bitness || bitShift == 0)
return value;
727 const std::size_t quotient = bitShift / blockBitLength, remainder = bitShift % blockBitLength;
728 const block stamp = (1UL << remainder) - 1;
730 for(std::size_t i = 0; i < blocksNumber - (quotient + (remainder ? 1 : 0)); ++i)
731 value.blocks[i] = ((value.blocks[i + quotient + (remainder ? 1 : 0)] & stamp) << (blockBitLength - remainder) % blockBitLength) | (value.blocks[i + quotient] & ~stamp) >> remainder;
733 value.blocks[blocksNumber - 1 - quotient] = (value.blocks[blocksNumber - 1] & ~stamp) >> remainder;
735 for(
long long i = blocksNumber - quotient; i < blocksNumber; ++i)
750 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned> &&
sizeof(Unsigned) < 8)
751 gpu
constexpr friend auto operator==(
const Aeu& our, Unsigned other)
noexcept ->
bool {
754 gpu
constexpr friend auto operator==(
const Aeu& our, uint64_t other)
noexcept ->
bool {
755 return our.
filledBlocksNumber() <= 2 && (
static_cast<uint64_t
>(our.blocks[1]) << blockBitLength |
static_cast<uint64_t
>(our.blocks[0])) == other;
764 template <std::
size_t otherBitness>
766 return our.
compareTo(other) == Comparison::equal;
778 template <
typename Un
signed>
requires (std::is_unsigned_v<Unsigned> &&
sizeof(Unsigned) < 8) [[nodiscard]]
779 gpu
constexpr auto compareTo(Unsigned other)
const noexcept -> Comparison {
780 using enum Comparison;
782 if(filledBlocksNumber() > 1)
return greater;
783 const auto cmp =
static_cast<block
>(other);
801 gpu
constexpr auto compareTo(uint64_t other)
const noexcept -> Comparison {
802 using enum Comparison;
803 if(filledBlocksNumber() > 2)
return greater;
804 const auto base = (
static_cast<uint64_t
>(blocks[1]) << blockBitLength) |
static_cast<uint64_t
>(blocks[0]);
806 return greater;
else if(base < other)
return less;
else return equal;
815 template <std::
size_t otherBitness = bitness> [[nodiscard]]
817 using enum Comparison;
819 const auto lowerBlockBorder = (blocksNumber < other.totalBlocksNumber() ? blocksNumber : other.totalBlocksNumber());
820 for(
long long i = lowerBlockBorder - 1; i >= 0; --i) {
821 const block thisBlock = blocks[i], otherBlock = other.getBlock(i);
822 if(thisBlock != otherBlock)
823 return (thisBlock > otherBlock ? greater : less);
826 if constexpr (otherBitness != blocksNumber * blockBitLength) {
827 using enum Comparison;
828 if (other.totalBlocksNumber() > blocksNumber) {
829 for (
long long i = other.totalBlocksNumber() - 1; i > lowerBlockBorder - 1; --i)
830 if (other.getBlock(i) != 0)
832 }
else if (blocksNumber > other.totalBlocksNumber()) {
833 for (
long long i = blocksNumber - 1; i > lowerBlockBorder - 1; --i)
845#if (defined(__CUDACC__) || __cplusplus < 202002L || defined (PRE_CPP_20)) && !defined DOXYGEN_SKIP
849 gpu
constexpr auto operator!=(
const Aeu& value)
const noexcept ->
bool {
return !this->operator==(value); }
850 gpu
constexpr auto operator<(
const Aeu& value)
const noexcept ->
bool {
return this->compareTo(value) == Comparison::less; }
851 gpu
constexpr auto operator<=(
const Aeu& value)
const noexcept ->
bool {
return !this->operator>(value); }
852 gpu
constexpr auto operator>(
const Aeu& value)
const noexcept ->
bool {
return this->compareTo(value) == Comparison::greater; }
853 gpu
constexpr auto operator>=(
const Aeu& value)
const noexcept ->
bool {
return !this->operator<(value); }
861 gpu
constexpr auto operator<=>(
const Aeu& other)
const noexcept -> std::strong_ordering {
862 switch(this->compareTo(other)) {
863 using enum Comparison;
865 return std::strong_ordering::less;
867 return std::strong_ordering::greater;
869 return std::strong_ordering::equal;
871 return std::strong_ordering::equivalent;
881 template <
typename Un
signed>
882 gpu
constexpr auto operator<=>(
const Unsigned& other)
const noexcept -> std::strong_ordering {
883 switch(this->compareTo(other)) {
884 using enum Comparison;
886 return std::strong_ordering::less;
888 return std::strong_ordering::greater;
890 return std::strong_ordering::equal;
892 return std::strong_ordering::equivalent;
907 gpu
constexpr auto setBit(std::size_t index,
bool bit)
noexcept ->
void {
909 if(index >= bitness)
return;
911 const std::size_t blockNumber = index / blockBitLength, bitNumber = index % blockBitLength;
912 assert(blockNumber < blocksNumber && bitNumber < blockBitLength);
914 blocks[blockNumber] |= (1U << bitNumber);
916 blocks[blockNumber] &= (~(1U << bitNumber));
926 gpu
constexpr auto getBit(std::size_t index)
const noexcept ->
bool {
928 if(index >= bitness)
return false;
930 const std::size_t blockNumber = index / blockBitLength, bitNumber = index % blockBitLength;
931 assert(blockNumber < blocksNumber && bitNumber < blockBitLength);
932 return blocks[blockNumber] & (1U << bitNumber);
941 gpu
constexpr auto setByte(std::size_t index,
byte byte)
noexcept ->
void {
943 if(index > blocksNumber *
sizeof(block))
return;
945 const std::size_t blockNumber = index /
sizeof(block), byteInBlock = index %
sizeof(block), shift = byteInBlock * bitsInByte;
946 assert(blockNumber < blocksNumber && byteInBlock <
sizeof(block));
947 blocks[blockNumber] &= ~(0xffU << shift); blocks[blockNumber] |=
static_cast<block
>(byte) << shift;
957 gpu
constexpr auto getByte(std::size_t index)
const noexcept ->
byte {
959 if(index > blocksNumber *
sizeof(block))
return 0;
961 const std::size_t blockNumber = index /
sizeof(block), byteInBlock = index %
sizeof(block), shift = byteInBlock * bitsInByte;
962 assert(blockNumber < blocksNumber && byteInBlock <
sizeof(block));
963 return (blocks[blockNumber] & (0xffU << shift)) >> shift;
972 gpu
constexpr auto setBlock(std::size_t index, block block)
noexcept ->
void {
974 if(index >= blocksNumber)
return;
976 blocks[index] = block;
986 gpu
constexpr auto getBlock(std::size_t index)
const noexcept -> block {
988 if(index >= blocksNumber)
return block();
990 return blocks[index];
998 gpu
constexpr auto byteCount() const noexcept -> std::
size_t {
999 std::size_t lastBlock = blocksNumber - 1;
1000 for(; lastBlock > 0 && blocks[lastBlock] == 0; --lastBlock)
1003 for(int8_t byteN =
sizeof(block) - 1; byteN >= 0; --byteN) {
1004 if((blocks[lastBlock] & (0xffU << (byteN * bitsInByte))) >> (byteN * bitsInByte))
1005 return lastBlock *
sizeof(block) + byteN + 1;
1007 return lastBlock *
sizeof(block);
1015 gpu
constexpr auto bitCount() const noexcept -> std::
size_t {
1016 std::size_t lastBlock = blocksNumber - 1;
1017 for(; lastBlock > 0 && blocks[lastBlock] == 0; --lastBlock)
1020 for(int8_t byteN =
sizeof(block) - 1; byteN >= 0; --byteN) {
1021 const auto byte = (blocks[lastBlock] & (0xffU << (byteN * bitsInByte))) >> (byteN * bitsInByte);
1024 for(int8_t bitN = bitsInByte - 1; bitN >= 0; --bitN) {
1025 if((
byte & (0x1u << bitN)) >> bitN)
1026 return (lastBlock *
sizeof(block) + byteN) * bitsInByte + bitN + 1;
1028 return ((lastBlock - 1) *
sizeof(block) + byteN) * bitsInByte;
1030 return lastBlock *
sizeof(block);
1038 gpu
constexpr auto isOdd() const noexcept ->
bool {
return (0x1 & blocks[0]) == 1; }
1045 gpu
constexpr auto isEven() const noexcept ->
bool {
return (0x1 & blocks[0]) == 0; }
1052 gpu
constexpr auto isZero() const noexcept ->
bool {
return filledBlocksNumber() == 0; }
1060 for(
long long i = blocksNumber - 1; i >= 0; --i)
1061 if(blocks[i])
return i + 1;
1070 gpu
static constexpr auto getBitness() noexcept -> std::
size_t {
return bitness; }
1083 gpu
constexpr auto swap(
Aeu& other)
noexcept ->
void {
1084 Aeu t = other; other.operator=(*this); this->operator=(t);
1098 gpu
static constexpr auto divide(
const Aeu& number,
const Aeu& divisor,
Aeu& quotient,
Aeu& remainder)
noexcept ->
void {
1099 const auto ratio = number.compareTo(divisor);
1101 quotient =
Aeu {}; remainder =
Aeu {};
1103 if(ratio == Comparison::greater) {
1104 const auto bitsUsed = number.filledBlocksNumber() * blockBitLength;
1105 for(
long long i = bitsUsed - 1; i >= 0; --i) {
1107 remainder.setBit(0, number.getBit(i));
1109 if(remainder >= divisor) {
1110 remainder -= divisor;
1111 quotient.setBit(i,
true);
1114 }
else if(ratio == Comparison::less)
1115 remainder = number;
else quotient = 1u;
1125 gpu
static constexpr auto divide(
const Aeu& number,
const Aeu& divisor)
noexcept -> pair<Aeu, Aeu> {
1126 pair<Aeu, Aeu> results; divide(number, divisor, results.first, results.second);
return results;
1138 gpu
static constexpr auto gcd(
const Aeu& first,
const Aeu& second,
Aeu& bezoutX,
Aeu& bezoutY)
noexcept ->
Aeu {
1139 Aeu gcd, quotient, remainder;
1141 const auto ratio = first.
compareTo(second);
1142 if(ratio == Comparison::greater) {
1144 divide(first, second, quotient, remainder);
1147 divide(second, first, quotient, remainder);
1150 bezoutX = 0u; bezoutY = 1u;
1151 for(
Aeu tX = 1u, tY = 0u; remainder != 0u; ) {
1152 Aeu tGcd = gcd; gcd = remainder;
1154 Aeu t = bezoutX; bezoutX = tX - quotient * bezoutX; tX = t;
1155 t = bezoutY; bezoutY = tY - quotient * bezoutY; tY = t;
1157 divide(tGcd, gcd, quotient, remainder);
1160 if(ratio != Comparison::greater)
1161 bezoutX.swap(bezoutY);
1173 gpu
static constexpr auto gcd(
const Aeu& first,
const Aeu& second)
noexcept ->
Aeu {
1174 Aeu gcd, quotient, remainder;
1176 const auto ratio = first.
compareTo(second);
1177 if(ratio == Comparison::greater) {
1179 divide(first, second, quotient, remainder);
1182 divide(second, first, quotient, remainder);
1185 for(
Aeu tX = 1u, tY = 0u; remainder != 0u; ) {
1186 Aeu tGcd = gcd; gcd = remainder;
1187 divide(tGcd, gcd, quotient, remainder);
1200 gpu
static constexpr auto lcm(
const Aeu& first,
const Aeu& second)
noexcept ->
Aeu {
return first / gcd(first, second) * second; }
1211 template <std::
size_t powerBitness = bitness> [[nodiscard]]
1219 auto [_, b] = divide(base, modulo);
1221 for(
unsigned iteration = 0; power.filledBlocksNumber() * blockBitLength != iteration; iteration++) {
1222 if(power.getBit(iteration)) {
1223 const auto [quotient, remainder] = divide(output * b, modulo);
1227 const auto [quotient, remainder] = divide(b * b, modulo);
1235 gpu
static constexpr auto powm(
const Aeu& base,
const Aeu& power,
const Aeu& mod)
noexcept ->
Aeu {
1237 auto [_, b] = divide(base, mod);
1239 for(std::size_t iteration = 0; power.filledBlocksNumber() * blockBitLength != iteration; iteration++) {
1240 if(power.getBit(iteration)) {
1241 const auto [quotient, remainder] = divide(output * b, mod);
1245 const auto [quotient, remainder] = divide(b * b, mod);
1259 gpu
static constexpr auto power2(std::size_t power)
noexcept ->
Aeu {
Aeu result {}; result.
setBit(power,
true);
return result; }
1268 Aeu x, y = power2((bitCount() + 1) / 2);
1272 y = (x + *
this / x) >> 1u;
1291 template <
byte base,
typename Char>
requires (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t> && (base == 2 || base == 8 || base == 10 || base == 16))
1292 gpu
constexpr auto getString(Char*
const buffer, std::size_t bufferSize,
bool showBase =
false,
bool hexUppercase =
false)
const noexcept -> std::size_t {
1293 if(bufferSize < 2)
return 0;
1295 std::size_t position = 0;
1297 if (showBase && bufferSize > 3) {
1298 if constexpr (base == 2) {
1299 if constexpr (std::is_same_v<Char, char>) {
1300 buffer[0] =
'0'; buffer[1] =
'b';
1302 buffer[0] = L
'0'; buffer[1] = L
'b';
1305 }
else if constexpr (base == 8) {
1306 if constexpr (std::is_same_v<Char, char>) {
1307 buffer[0] =
'0'; buffer[1] =
'o';
1309 buffer[0] = L
'0'; buffer[1] = L
'o';
1312 }
else if constexpr (base == 16) {
1313 if constexpr (std::is_same_v<Char, char>) {
1314 buffer[0] =
'0'; buffer[1] =
'x';
1316 buffer[0] = L
'0'; buffer[1] = L
'x';
1323 buffer[position++] = [] {
if constexpr (std::is_same_v<Char, char>) {
return '0'; }
else {
return L
'0'; }}();
1327 if constexpr (base == 16) {
1328 long long iter = blocks.size() - 1;
1329 for (; blocks[iter] == 0 && iter >= 0; --iter)
1332 if constexpr (std::is_same_v<Char, char>) {
1333 position += snprintf(buffer + position, bufferSize - position, (hexUppercase ?
"%X" :
"%x"), blocks[iter--]);
1334 for (; iter >= 0; --iter)
1335 position += snprintf(buffer + position, bufferSize - position, (hexUppercase ?
"%08X" :
"%08x"), blocks[iter]);
1337 position += swprintf(buffer + position, bufferSize - position, (hexUppercase ? L
"%X" : L
"%x"), blocks[iter--]);
1338 for (; iter >= 0; --iter)
1339 position += swprintf(buffer + position, bufferSize - position, (hexUppercase ? L
"%08X" : L
"%08x"), blocks[iter]);
1342 const auto startPosition = position;
1345 while (!copy.
isZero() && position < bufferSize) {
1346 auto [quotient, remainder] = divide(copy, base);
1347 if constexpr (std::is_same_v<Char, char>) {
1348 buffer[position++] =
'0' + remainder.template integralCast<byte>();
1350 buffer[position++] = L
'0' + remainder.template integralCast<byte>();
1354 const auto digitsTotal = position - startPosition;
1355 for (std::size_t i = 0; i * 2 < digitsTotal; ++i) {
1356 Char t = buffer[startPosition + i];
1357 buffer[startPosition + i] = buffer[startPosition + digitsTotal - 1 - i];
1358 buffer[startPosition + digitsTotal - 1 - i] = t;
1361 buffer[position++] = Char {};
1376 template <
typename Char>
requires (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t>)
1377 friend constexpr auto operator<<(std::basic_ostream<Char>& os,
const Aeu& number) -> std::basic_ostream<Char>& {
1378 auto flags = os.flags();
1380 const auto base = [] (
long baseField, std::basic_ostream<Char>& ss,
bool showbase) {
1381 auto tBase = (baseField == std::ios::hex ? 16u : (baseField == std::ios::oct ? 8u : 10u));
1382 if(showbase && tBase != 10)
1384 if constexpr (std::is_same_v<Char, char>) {
1385 return tBase == 8 ?
"0o" :
"0x";
1387 return tBase == 8 ? L
"0o" : L
"0x";
1389 } () << std::noshowbase ;
1391 } (flags & std::ios::basefield, os, flags & std::ios::showbase);
1397 long long iter = number.blocks.size() - 1;
1398 for(; number.blocks[iter] == 0 && iter >= 0; --iter)
1401 os << number.blocks[iter--];
1402 for (; iter >= 0; --iter) {
1403 os.fill([] {
if constexpr (std::is_same_v<Char, char>) {
return '0'; }
else {
return L
'0'; } } ());
1404 os.width(8); os << std::right << number.blocks[iter];
1409 constexpr auto bufferSize =
static_cast<std::size_t
>(
static_cast<double>(bitness) / 2.95);
1410 Char buffer [bufferSize] {}; std::size_t filled = 0;
1413 while(!copy.
isZero() && filled < bufferSize) {
1414 const auto [quotient, remainder] = divide(copy, base);
1415 buffer[filled++] = [] {
if constexpr (std::is_same_v<Char, char>) {
return '0'; }
else {
return L
'0'; } } () + remainder.template integralCast<byte>();
1419 for(; filled > 0; --filled)
1420 os << buffer[filled - 1];
1435 template <
typename Char>
requires (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t>)
1436 constexpr auto readBinary(std::basic_istream<Char>& is,
bool bigEndian =
true) ->
void {
1439 for(
auto it = blocks.rbegin(); it != blocks.rend(); ++it)
1440 if(!is.read(
reinterpret_cast<char*
>(&*it),
sizeof(block)))
break;
1442 for(
auto& tBlock: blocks)
1443 if(!is.read(
reinterpret_cast<char*
>(&tBlock),
sizeof(block)))
break;
1454 template <
typename Char>
requires (std::is_same_v<Char, char> || std::is_same_v<Char, wchar_t>)
1455 constexpr auto writeBinary(std::basic_ostream<Char>& os,
bool bigEndian =
true)
const noexcept ->
void {
1457 for(
auto it = blocks.rbegin(); it != blocks.rend(); ++it)
1458 if(!os.write(
reinterpret_cast<const char*
>(&*it),
sizeof(block)))
break;
1460 for(
auto& block: blocks)
1461 if(!os.write(
reinterpret_cast<const char*
>(&block),
sizeof(block)))
break;
1473 template <
typename Integral>
requires (std::is_integral_v<Integral>) [[nodiscard]]
1475 const uint64_t value = (
static_cast<uint64_t
>(blocks[1]) << blockBitLength) |
static_cast<uint64_t
>(blocks[0]);
1476 return static_cast<Integral
>(value);
1487 template <std::
size_t newBitness>
requires (newBitness != bitness) [[nodiscard]]
1492 for(std::size_t blockIdx = 0; blockIdx < blockBoarder; ++blockIdx)
1493 result.
setBlock(blockIdx, getBlock(blockIdx));
1500#if defined __CUDACC__
1508 __device__
constexpr auto tryAtomicSet(
const Aeu& value)
noexcept ->
void {
1509 for(std::size_t i = 0; i < blocksNumber; ++i)
1510 atomicExch(&blocks[i], value.blocks[i]);
1519 __device__
constexpr auto tryAtomicExchange(
const Aeu& value)
noexcept ->
void {
1520 for(std::size_t i = 0; i < blocksNumber; ++i)
1521 atomicExch(&value.blocks[i], atomicExch(&blocks[i], value.blocks[i]));
1591template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1600template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1609template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1618template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1627template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1636template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1645template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
1654template <std::
size_t bitness,
typename Integral>
requires (std::is_integral_v<Integral>)
Long precision unsigned integer.
Definition Aeu.h:83
static gpu constexpr auto gcd(const Aeu &first, const Aeu &second, Aeu &bezoutX, Aeu &bezoutY) noexcept -> Aeu
Extended Euclidean algorithm for greatest common divisor.
Definition Aeu.h:1138
gpu constexpr auto operator<=>(const Aeu &other) const noexcept -> std::strong_ordering
Three-way comparison operator.
Definition Aeu.h:861
gpu constexpr friend auto operator|=(Aeu &left, const Aeu &right) noexcept -> Aeu &
Assignment bitwise OR operator.
Definition Aeu.h:661
gpu constexpr auto bitCount() const noexcept -> std::size_t
Get amount of non-empty bits in number right to left.
Definition Aeu.h:1015
gpu constexpr friend auto operator&(const Aeu &left, const Aeu &right) noexcept -> Aeu
Bitwise AND operator.
Definition Aeu.h:628
gpu constexpr auto compareTo(Unsigned other) const noexcept -> Comparison
Internal comparison operator for built-in integral types uint8_t, uint16_t, uint32_t.
Definition Aeu.h:779
gpu constexpr auto filledBlocksNumber() const noexcept -> std::size_t
Get number of non-empty blocks inside object starting from the right.
Definition Aeu.h:1059
gpu constexpr auto operator+() const noexcept -> Aeu
Unary plus operator.
Definition Aeu.h:305
gpu constexpr auto getString(Char *const buffer, std::size_t bufferSize, bool showBase=false, bool hexUppercase=false) const noexcept -> std::size_t
Character buffer output operator.
Definition Aeu.h:1292
gpu constexpr friend auto operator/(const Aeu &division, const Aeu &divisor) noexcept -> Aeu
Division operator.
Definition Aeu.h:545
static gpu constexpr auto power2(std::size_t power) noexcept -> Aeu
Fast exponentiation for powers of 2.
Definition Aeu.h:1259
static gpu constexpr auto gcd(const Aeu &first, const Aeu &second) noexcept -> Aeu
Greatest common divisor.
Definition Aeu.h:1173
gpu constexpr auto isEven() const noexcept -> bool
Check whether number is even.
Definition Aeu.h:1045
gpu constexpr auto operator~() const noexcept -> Aeu
Bitwise complement operator.
Definition Aeu.h:591
gpu constexpr auto setByte(std::size_t index, byte byte) noexcept -> void
Set byte in number by index starting from the right.
Definition Aeu.h:941
gpu constexpr friend auto operator+(const Aeu &addition, const Aeu &addendum) noexcept -> Aeu
Addition operator.
Definition Aeu.h:377
gpu constexpr friend auto operator==(const Aeu &our, const Aeu< otherBitness > &other) noexcept -> bool
Templated Equality check operator for numbers of different precision.
Definition Aeu.h:765
gpu constexpr Aeu(String &&stringView) noexcept
String or string-view based constructor.
Definition Aeu.h:258
gpu constexpr friend auto operator%(const Aeu &modulation, const Aeu &modulo) noexcept -> Aeu
Modulo operator.
Definition Aeu.h:568
gpu constexpr auto byteCount() const noexcept -> std::size_t
Get amount of non-empty bytes in number right to left.
Definition Aeu.h:998
gpu constexpr auto setBlock(std::size_t index, block block) noexcept -> void
Set block in number by index starting from the right.
Definition Aeu.h:972
gpu constexpr friend auto operator&=(Aeu &left, const Aeu &right) noexcept -> Aeu &
Assignment bitwise AND operator.
Definition Aeu.h:638
gpu constexpr auto getByte(std::size_t index) const noexcept -> byte
Get byte in number by index starting from the right.
Definition Aeu.h:957
gpu constexpr Aeu(const Char(&literal)[arrayLength]) noexcept
C-style string literal constructor.
Definition Aeu.h:249
gpu constexpr auto operator--(int) &noexcept -> Aeu
Postfix decrement.
Definition Aeu.h:354
static gpu constexpr auto powm(const Aeu &base, const Aeu< powerBitness > &power, const Aeu &modulo) noexcept -> Aeu
Exponentiation by modulo.
Definition Aeu.h:1212
gpu constexpr auto integralCast() const noexcept -> Integral
Integral type cast operator for built-in types.
Definition Aeu.h:1474
gpu constexpr friend auto operator-=(Aeu &subtraction, const Aeu &subtrahend) noexcept -> Aeu &
Assignment subtraction operator.
Definition Aeu.h:426
gpu constexpr auto isOdd() const noexcept -> bool
Check whether number is odd.
Definition Aeu.h:1038
gpu constexpr friend auto operator^=(Aeu &left, const Aeu &right) noexcept -> Aeu &
Assignment bitwise XOR operator.
Definition Aeu.h:615
constexpr auto writeBinary(std::basic_ostream< Char > &os, bool bigEndian=true) const noexcept -> void
STD stream binary writing operator.
Definition Aeu.h:1455
static gpu constexpr auto divide(const Aeu &number, const Aeu &divisor) noexcept -> pair< Aeu, Aeu >
Integer division. Returns results by value.
Definition Aeu.h:1125
gpu constexpr auto swap(Aeu &other) noexcept -> void
Make swap between two objects.
Definition Aeu.h:1083
static gpu constexpr auto divide(const Aeu &number, const Aeu &divisor, Aeu "ient, Aeu &remainder) noexcept -> void
Integer division. Returns results by reference.
Definition Aeu.h:1098
gpu constexpr auto operator++(int) &noexcept -> Aeu
Postfix increment.
Definition Aeu.h:334
static gpu constexpr auto getBitness() noexcept -> std::size_t
Get number's precision.
Definition Aeu.h:1070
gpu constexpr friend auto operator|(const Aeu &left, const Aeu &right) noexcept -> Aeu
Bitwise OR operator.
Definition Aeu.h:651
gpu constexpr friend auto operator^(const Aeu &left, const Aeu &right) noexcept -> Aeu
Bitwise XOR operator.
Definition Aeu.h:605
gpu constexpr auto setBit(std::size_t index, bool bit) noexcept -> void
Set a bit in number by index starting from the right.
Definition Aeu.h:907
gpu constexpr auto precisionCast() const noexcept -> Aeu< newBitness >
Precision cast operator.
Definition Aeu.h:1488
static gpu constexpr auto totalBlocksNumber() noexcept -> std::size_t
Get the number of blocks (length of array of uint32_t integers) inside object.
Definition Aeu.h:1077
gpu constexpr auto isZero() const noexcept -> bool
Check whether number is zero.
Definition Aeu.h:1052
gpu constexpr auto operator<=>(const Unsigned &other) const noexcept -> std::strong_ordering
Three-way comparison operator for numbers of different precision and built-in integral types.
Definition Aeu.h:882
gpu constexpr friend auto operator%=(Aeu &modulation, const Aeu &modulo) noexcept -> Aeu &
Assignment modulo operator.
Definition Aeu.h:578
gpu constexpr friend auto operator/=(Aeu &division, const Aeu &divisor) noexcept -> Aeu &
Assignment division operator.
Definition Aeu.h:556
gpu constexpr auto operator--() noexcept -> Aeu &
Prefix decrement.
Definition Aeu.h:340
constexpr auto readBinary(std::basic_istream< Char > &is, bool bigEndian=true) -> void
STD stream binary reading operator.
Definition Aeu.h:1436
gpu constexpr Aeu(const Char *data, std::size_t size) noexcept
Pointer-based character constructor.
Definition Aeu.h:186
gpu constexpr auto compareTo(uint64_t other) const noexcept -> Comparison
Internal comparison operator for type uint64_t.
Definition Aeu.h:801
gpu constexpr friend auto operator*=(Aeu &multiplication, const Aeu &factor) noexcept -> Aeu &
Assignment multiplication operator.
Definition Aeu.h:502
gpu constexpr friend auto operator+=(Aeu &addition, const Aeu &addendum) noexcept -> Aeu &
Assignment addition operator.
Definition Aeu.h:402
static gpu constexpr auto lcm(const Aeu &first, const Aeu &second) noexcept -> Aeu
Least common multiplier.
Definition Aeu.h:1200
gpu constexpr auto getBit(std::size_t index) const noexcept -> bool
Get bit in number by index staring from the right.
Definition Aeu.h:926
gpu constexpr friend auto operator-(const Aeu &subtraction, const Aeu &subtrahend) noexcept -> Aeu
Subtraction operator.
Definition Aeu.h:416
gpu constexpr auto squareRoot() const noexcept -> Aeu
Get square root.
Definition Aeu.h:1267
gpu constexpr auto getBlock(std::size_t index) const noexcept -> block
Get block in number by index starting from the right.
Definition Aeu.h:986
gpu constexpr friend auto operator*(const Aeu &multiplication, const Aeu &factor) noexcept -> Aeu
Multiplication operator.
Definition Aeu.h:451
gpu constexpr auto operator-() const noexcept -> Aeu
Unary minus operator.
Definition Aeu.h:312
gpu constexpr auto operator++() noexcept -> Aeu &
Prefix increment.
Definition Aeu.h:320
gpu constexpr auto compareTo(const Aeu< otherBitness > &other) const noexcept -> Comparison
Internal comparison operator.
Definition Aeu.h:816
gpu constexpr Aeu() noexcept=default
Default constructor.