|
|
|
|
|
#ifndef DLIB_BIGINT_KERNEL_2_CPp_ |
|
#define DLIB_BIGINT_KERNEL_2_CPp_ |
|
#include "bigint_kernel_2.h" |
|
|
|
#include <iostream> |
|
#include <cmath> |
|
|
|
namespace dlib |
|
{ |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bigint_kernel_2:: |
|
bigint_kernel_2 ( |
|
) : |
|
slack(25), |
|
data(new data_record(slack)) |
|
{} |
|
|
|
|
|
|
|
bigint_kernel_2:: |
|
bigint_kernel_2 ( |
|
uint32 value |
|
) : |
|
slack(25), |
|
data(new data_record(slack)) |
|
{ |
|
*(data->number) = static_cast<uint16>(value&0xFFFF); |
|
*(data->number+1) = static_cast<uint16>((value>>16)&0xFFFF); |
|
if (*(data->number+1) != 0) |
|
data->digits_used = 2; |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2:: |
|
bigint_kernel_2 ( |
|
const bigint_kernel_2& item |
|
) : |
|
slack(25), |
|
data(item.data) |
|
{ |
|
data->references += 1; |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2:: |
|
~bigint_kernel_2 ( |
|
) |
|
{ |
|
if (data->references == 1) |
|
{ |
|
delete data; |
|
} |
|
else |
|
{ |
|
data->references -= 1; |
|
} |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator+ ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
data_record* temp = new data_record ( |
|
std::max(rhs.data->digits_used,data->digits_used) + slack |
|
); |
|
long_add(data,rhs.data,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator+= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(std::max(data->digits_used,rhs.data->digits_used)+slack); |
|
data->references -= 1; |
|
long_add(data,rhs.data,temp); |
|
data = temp; |
|
} |
|
|
|
else if (data->size <= std::max(data->digits_used,rhs.data->digits_used)) |
|
{ |
|
data_record* temp = new data_record(std::max(data->digits_used,rhs.data->digits_used)+slack); |
|
long_add(data,rhs.data,temp); |
|
delete data; |
|
data = temp; |
|
} |
|
|
|
else |
|
{ |
|
long_add(data,rhs.data,data); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator- ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
data_record* temp = new data_record ( |
|
data->digits_used + slack |
|
); |
|
long_sub(data,rhs.data,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator-= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
long_sub(data,rhs.data,temp); |
|
data = temp; |
|
} |
|
else |
|
{ |
|
long_sub(data,rhs.data,data); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator* ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
data_record* temp = new data_record ( |
|
data->digits_used + rhs.data->digits_used + slack |
|
); |
|
long_mul(data,rhs.data,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator*= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
|
|
data_record* temp = new data_record(rhs.data->digits_used+data->digits_used+slack); |
|
long_mul(data,rhs.data,temp); |
|
|
|
|
|
if (data->references != 1) |
|
{ |
|
data->references -= 1; |
|
} |
|
else |
|
{ |
|
delete data; |
|
} |
|
data = temp; |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator/ ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data_record* remainder; |
|
try { |
|
remainder = new data_record(data->digits_used+slack); |
|
} catch (...) { delete temp; throw; } |
|
|
|
long_div(data,rhs.data,temp,remainder); |
|
delete remainder; |
|
|
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator/= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
|
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data_record* remainder; |
|
try { |
|
remainder = new data_record(data->digits_used+slack); |
|
} catch (...) { delete temp; throw; } |
|
|
|
long_div(data,rhs.data,temp,remainder); |
|
|
|
|
|
if (data->references != 1) |
|
{ |
|
data->references -= 1; |
|
} |
|
|
|
else |
|
{ |
|
delete data; |
|
} |
|
data = temp; |
|
delete remainder; |
|
|
|
|
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator% ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data_record* remainder; |
|
try { |
|
remainder = new data_record(data->digits_used+slack); |
|
} catch (...) { delete temp; throw; } |
|
|
|
long_div(data,rhs.data,temp,remainder); |
|
delete temp; |
|
return bigint_kernel_2(remainder,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator%= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data_record* remainder; |
|
try { |
|
remainder = new data_record(data->digits_used+slack); |
|
} catch (...) { delete temp; throw; } |
|
|
|
long_div(data,rhs.data,temp,remainder); |
|
|
|
|
|
if (data->references != 1) |
|
{ |
|
data->references -= 1; |
|
} |
|
|
|
else |
|
{ |
|
delete data; |
|
} |
|
data = remainder; |
|
delete temp; |
|
return *this; |
|
} |
|
|
|
|
|
|
|
bool bigint_kernel_2:: |
|
operator < ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
return is_less_than(data,rhs.data); |
|
} |
|
|
|
|
|
|
|
bool bigint_kernel_2:: |
|
operator == ( |
|
const bigint_kernel_2& rhs |
|
) const |
|
{ |
|
return is_equal_to(data,rhs.data); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator= ( |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
if (this == &rhs) |
|
return *this; |
|
|
|
|
|
if (data->references == 1) |
|
{ |
|
delete data; |
|
data = rhs.data; |
|
data->references += 1; |
|
} |
|
else |
|
{ |
|
data->references -= 1; |
|
data = rhs.data; |
|
data->references += 1; |
|
} |
|
|
|
return *this; |
|
} |
|
|
|
|
|
|
|
std::ostream& operator<< ( |
|
std::ostream& out_, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
std::ostream out(out_.rdbuf()); |
|
|
|
typedef bigint_kernel_2 bigint; |
|
|
|
bigint::data_record* temp = new bigint::data_record(*rhs.data,0); |
|
|
|
|
|
|
|
|
|
char* str; |
|
try { |
|
str = new char[(rhs.data->digits_used)*5+10]; |
|
} catch (...) { delete temp; throw; } |
|
|
|
char* str_start = str; |
|
str += (rhs.data->digits_used)*5+9; |
|
*str = 0; --str; |
|
|
|
|
|
uint16 remainder; |
|
rhs.short_div(temp,10000,temp,remainder); |
|
|
|
|
|
char a = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char b = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char c = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char d = remainder % 10 + '0'; |
|
remainder /= 10; |
|
|
|
|
|
*str = a; --str; |
|
*str = b; --str; |
|
*str = c; --str; |
|
*str = d; --str; |
|
|
|
|
|
|
|
while (temp->digits_used != 1 || *(temp->number) != 0) |
|
{ |
|
rhs.short_div(temp,10000,temp,remainder); |
|
|
|
|
|
char a = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char b = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char c = remainder % 10 + '0'; |
|
remainder /= 10; |
|
char d = remainder % 10 + '0'; |
|
remainder /= 10; |
|
|
|
*str = a; --str; |
|
*str = b; --str; |
|
*str = c; --str; |
|
*str = d; --str; |
|
} |
|
|
|
|
|
++str; |
|
if (*str == '0') |
|
++str; |
|
if (*str == '0') |
|
++str; |
|
if (*str == '0') |
|
++str; |
|
|
|
|
|
|
|
|
|
out << str; |
|
delete [] str_start; |
|
delete temp; |
|
return out_; |
|
|
|
} |
|
|
|
|
|
|
|
std::istream& operator>> ( |
|
std::istream& in_, |
|
bigint_kernel_2& rhs |
|
) |
|
{ |
|
std::istream in(in_.rdbuf()); |
|
|
|
|
|
while (in.peek() == ' ' || in.peek() == '\t' || in.peek() == '\n') |
|
{ |
|
in.get(); |
|
} |
|
|
|
|
|
if ( !(in.peek() >= '0' && in.peek() <= '9')) |
|
{ |
|
in_.clear(std::ios::failbit); |
|
return in_; |
|
} |
|
|
|
int num_read; |
|
bigint_kernel_2 temp; |
|
do |
|
{ |
|
|
|
|
|
num_read = 1; |
|
char a = 0; |
|
char b = 0; |
|
char c = 0; |
|
char d = 0; |
|
|
|
if (in.peek() >= '0' && in.peek() <= '9') |
|
{ |
|
num_read *= 10; |
|
a = in.get(); |
|
} |
|
if (in.peek() >= '0' && in.peek() <= '9') |
|
{ |
|
num_read *= 10; |
|
b = in.get(); |
|
} |
|
if (in.peek() >= '0' && in.peek() <= '9') |
|
{ |
|
num_read *= 10; |
|
c = in.get(); |
|
} |
|
if (in.peek() >= '0' && in.peek() <= '9') |
|
{ |
|
num_read *= 10; |
|
d = in.get(); |
|
} |
|
|
|
|
|
uint16 num = 0; |
|
if (a != 0) |
|
{ |
|
num = a - '0'; |
|
} |
|
if (b != 0) |
|
{ |
|
num *= 10; |
|
num += b - '0'; |
|
} |
|
if (c != 0) |
|
{ |
|
num *= 10; |
|
num += c - '0'; |
|
} |
|
if (d != 0) |
|
{ |
|
num *= 10; |
|
num += d - '0'; |
|
} |
|
|
|
|
|
if (num_read != 1) |
|
{ |
|
|
|
temp *= num_read; |
|
|
|
temp += num; |
|
} |
|
|
|
} while (num_read == 10000); |
|
|
|
|
|
rhs = temp; |
|
return in_; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator+ ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(rhs.data->digits_used+rhs.slack); |
|
|
|
rhs.short_add(rhs.data,lhs,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator+ ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(lhs.data->digits_used+lhs.slack); |
|
|
|
lhs.short_add(lhs.data,rhs,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator+= ( |
|
uint16 rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
short_add(data,rhs,temp); |
|
data = temp; |
|
} |
|
|
|
else if (data->digits_used == data->size) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
short_add(data,rhs,temp); |
|
delete data; |
|
data = temp; |
|
} |
|
|
|
else |
|
{ |
|
short_add(data,rhs,data); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator- ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record(rhs.slack); |
|
|
|
*(temp->number) = lhs - *(rhs.data->number); |
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator- ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(lhs.data->digits_used+lhs.slack); |
|
|
|
lhs.short_sub(lhs.data,rhs,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator-= ( |
|
uint16 rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
short_sub(data,rhs,temp); |
|
data = temp; |
|
} |
|
else |
|
{ |
|
short_sub(data,rhs,data); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator* ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(rhs.data->digits_used+rhs.slack); |
|
|
|
rhs.short_mul(rhs.data,lhs,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator* ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(lhs.data->digits_used+lhs.slack); |
|
|
|
lhs.short_mul(lhs.data,rhs,temp); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator*= ( |
|
uint16 rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
short_mul(data,rhs,temp); |
|
data = temp; |
|
} |
|
|
|
else if (data->digits_used == data->size) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
short_mul(data,rhs,temp); |
|
delete data; |
|
data = temp; |
|
} |
|
else |
|
{ |
|
short_mul(data,rhs,data); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator/ ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record(rhs.slack); |
|
|
|
|
|
if (rhs.data->digits_used == 1) |
|
{ |
|
*(temp->number) = lhs/ *(rhs.data->number); |
|
} |
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator/ ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record |
|
(lhs.data->digits_used+lhs.slack); |
|
|
|
uint16 remainder; |
|
lhs.short_div(lhs.data,rhs,temp,remainder); |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator/= ( |
|
uint16 rhs |
|
) |
|
{ |
|
uint16 remainder; |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
short_div(data,rhs,temp,remainder); |
|
data = temp; |
|
} |
|
else |
|
{ |
|
short_div(data,rhs,data,remainder); |
|
} |
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator% ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
|
|
bigint::data_record* temp = new bigint::data_record(rhs.slack); |
|
|
|
if (rhs.data->digits_used == 1) |
|
{ |
|
|
|
*(temp->number) = lhs % *(rhs.data->number); |
|
} |
|
else |
|
{ |
|
|
|
*(temp->number) = lhs; |
|
} |
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 operator% ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
typedef bigint_kernel_2 bigint; |
|
bigint::data_record* temp = new bigint::data_record(lhs.data->digits_used+lhs.slack); |
|
|
|
uint16 remainder; |
|
|
|
lhs.short_div(lhs.data,rhs,temp,remainder); |
|
temp->digits_used = 1; |
|
*(temp->number) = remainder; |
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator%= ( |
|
uint16 rhs |
|
) |
|
{ |
|
uint16 remainder; |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
short_div(data,rhs,temp,remainder); |
|
data = temp; |
|
} |
|
else |
|
{ |
|
short_div(data,rhs,data,remainder); |
|
} |
|
|
|
data->digits_used = 1; |
|
*(data->number) = remainder; |
|
return *this; |
|
} |
|
|
|
|
|
|
|
bool operator < ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
return (rhs.data->digits_used > 1 || lhs < *(rhs.data->number) ); |
|
} |
|
|
|
|
|
|
|
bool operator < ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
return (lhs.data->digits_used == 1 && *(lhs.data->number) < rhs); |
|
} |
|
|
|
|
|
|
|
bool operator == ( |
|
const bigint_kernel_2& lhs, |
|
uint16 rhs |
|
) |
|
{ |
|
return (lhs.data->digits_used == 1 && *(lhs.data->number) == rhs); |
|
} |
|
|
|
|
|
|
|
bool operator == ( |
|
uint16 lhs, |
|
const bigint_kernel_2& rhs |
|
) |
|
{ |
|
return (rhs.data->digits_used == 1 && *(rhs.data->number) == lhs); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator= ( |
|
uint16 rhs |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data->references -= 1; |
|
try { |
|
data = new data_record(slack); |
|
} catch (...) { data->references += 1; throw; } |
|
} |
|
else |
|
{ |
|
data->digits_used = 1; |
|
} |
|
|
|
*(data->number) = rhs; |
|
|
|
return *this; |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator++ ( |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
increment(data,temp); |
|
data = temp; |
|
} |
|
|
|
else if (data->digits_used == data->size) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
increment(data,temp); |
|
delete data; |
|
data = temp; |
|
} |
|
else |
|
{ |
|
increment(data,data); |
|
} |
|
|
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator++ ( |
|
int |
|
) |
|
{ |
|
data_record* temp; |
|
|
|
data_record* temp2 = new data_record(data->digits_used+slack); |
|
increment(data,temp2); |
|
|
|
temp = data; |
|
data = temp2; |
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
bigint_kernel_2& bigint_kernel_2:: |
|
operator-- ( |
|
) |
|
{ |
|
|
|
if (data->references != 1) |
|
{ |
|
data_record* temp = new data_record(data->digits_used+slack); |
|
data->references -= 1; |
|
decrement(data,temp); |
|
data = temp; |
|
} |
|
else |
|
{ |
|
decrement(data,data); |
|
} |
|
|
|
return *this; |
|
} |
|
|
|
|
|
|
|
const bigint_kernel_2 bigint_kernel_2:: |
|
operator-- ( |
|
int |
|
) |
|
{ |
|
data_record* temp; |
|
|
|
data_record* temp2 = new data_record(data->digits_used+slack); |
|
decrement(data,temp2); |
|
|
|
temp = data; |
|
data = temp2; |
|
|
|
return bigint_kernel_2(temp,0); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
short_add ( |
|
const data_record* data, |
|
uint16 value, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
uint32 temp = value; |
|
temp <<= 16; |
|
|
|
|
|
const uint16* number = data->number; |
|
const uint16* end = number + data->digits_used; |
|
uint16* r = result->number; |
|
|
|
while (number != end) |
|
{ |
|
|
|
temp = *number + (temp>>16); |
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++number; |
|
++r; |
|
} |
|
|
|
|
|
if ((temp>>16) != 0) |
|
{ |
|
result->digits_used = data->digits_used + 1; |
|
|
|
*r = static_cast<uint16>(temp>>16); |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
short_sub ( |
|
const data_record* data, |
|
uint16 value, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
|
|
const uint16* number = data->number; |
|
const uint16* end = number + data->digits_used - 1; |
|
uint16* r = result->number; |
|
|
|
uint32 temp = *number - value; |
|
|
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
|
|
while (number != end) |
|
{ |
|
++number; |
|
++r; |
|
|
|
|
|
temp = *number - (temp>>31); |
|
|
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
} |
|
|
|
|
|
if (*r == 0) |
|
{ |
|
if (data->digits_used == 1) |
|
result->digits_used = 1; |
|
else |
|
result->digits_used = data->digits_used - 1; |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used; |
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
short_mul ( |
|
const data_record* data, |
|
uint16 value, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
uint32 temp = 0; |
|
|
|
|
|
const uint16* number = data->number; |
|
uint16* r = result->number; |
|
const uint16* end = r + data->digits_used; |
|
|
|
|
|
|
|
while ( r != end) |
|
{ |
|
|
|
|
|
temp = *number*(uint32)value + (temp>>16); |
|
|
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++number; |
|
++r; |
|
} |
|
|
|
|
|
if ((temp>>16) != 0) |
|
{ |
|
result->digits_used = data->digits_used + 1; |
|
|
|
*r = static_cast<uint16>(temp>>16); |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used; |
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
short_div ( |
|
const data_record* data, |
|
uint16 value, |
|
data_record* result, |
|
uint16& rem |
|
) const |
|
{ |
|
|
|
uint16 remainder = 0; |
|
uint32 temp; |
|
|
|
|
|
|
|
const uint16* number = data->number + data->digits_used - 1; |
|
const uint16* end = number - data->digits_used; |
|
uint16* r = result->number + data->digits_used - 1; |
|
|
|
|
|
|
|
if (*number < value) |
|
{ |
|
if (data->digits_used == 1) |
|
result->digits_used = 1; |
|
else |
|
result->digits_used = data->digits_used - 1; |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used; |
|
} |
|
|
|
|
|
|
|
while (number != end) |
|
{ |
|
|
|
temp = *number + (((uint32)remainder)<<16); |
|
|
|
*r = static_cast<uint16>(temp/value); |
|
remainder = static_cast<uint16>(temp%value); |
|
|
|
--number; |
|
--r; |
|
} |
|
|
|
rem = remainder; |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
long_add ( |
|
const data_record* lhs, |
|
const data_record* rhs, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
uint32 temp=0; |
|
|
|
uint16* min_num; |
|
uint16* max_num; |
|
uint16* min_end; |
|
uint16* max_end; |
|
uint16* r = result->number; |
|
|
|
uint32 max_digits_used; |
|
if (lhs->digits_used < rhs->digits_used) |
|
{ |
|
max_digits_used = rhs->digits_used; |
|
min_num = lhs->number; |
|
max_num = rhs->number; |
|
min_end = min_num + lhs->digits_used; |
|
max_end = max_num + rhs->digits_used; |
|
} |
|
else |
|
{ |
|
max_digits_used = lhs->digits_used; |
|
min_num = rhs->number; |
|
max_num = lhs->number; |
|
min_end = min_num + rhs->digits_used; |
|
max_end = max_num + lhs->digits_used; |
|
} |
|
|
|
|
|
|
|
|
|
while (min_num != min_end) |
|
{ |
|
|
|
temp = *min_num + *max_num + (temp>>16); |
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++min_num; |
|
++max_num; |
|
++r; |
|
} |
|
|
|
|
|
while (max_num != max_end) |
|
{ |
|
|
|
temp = *max_num + (temp>>16); |
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++max_num; |
|
++r; |
|
} |
|
|
|
|
|
if ((temp>>16) != 0) |
|
{ |
|
result->digits_used = max_digits_used + 1; |
|
|
|
*r = static_cast<uint16>(temp>>16); |
|
} |
|
else |
|
{ |
|
result->digits_used = max_digits_used; |
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
long_sub ( |
|
const data_record* lhs, |
|
const data_record* rhs, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
|
|
const uint16* number1 = lhs->number; |
|
const uint16* number2 = rhs->number; |
|
const uint16* end = number2 + rhs->digits_used; |
|
uint16* r = result->number; |
|
|
|
|
|
|
|
uint32 temp =0; |
|
|
|
|
|
while (number2 != end) |
|
{ |
|
|
|
|
|
temp = *number1 - *number2 - (temp>>31); |
|
|
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++number1; |
|
++number2; |
|
++r; |
|
} |
|
|
|
end = lhs->number + lhs->digits_used; |
|
while (number1 != end) |
|
{ |
|
|
|
|
|
temp = *number1 - (temp>>31); |
|
|
|
|
|
*r = static_cast<uint16>(temp & 0xFFFF); |
|
|
|
++number1; |
|
++r; |
|
} |
|
|
|
result->digits_used = lhs->digits_used; |
|
|
|
--r; |
|
while (*r == 0 && result->digits_used > 1) |
|
{ |
|
--r; |
|
--result->digits_used; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
long_div ( |
|
const data_record* lhs, |
|
const data_record* rhs, |
|
data_record* result, |
|
data_record* remainder |
|
) const |
|
{ |
|
|
|
result->digits_used = 1; |
|
*(result->number) = 0; |
|
|
|
uint16* a; |
|
uint16* b; |
|
uint16* end; |
|
|
|
|
|
remainder->digits_used = lhs->digits_used; |
|
a = remainder->number; |
|
end = a + remainder->digits_used; |
|
b = lhs->number; |
|
while (a != end) |
|
{ |
|
*a = *b; |
|
++a; |
|
++b; |
|
} |
|
|
|
|
|
|
|
|
|
if (is_less_than(lhs,rhs)) |
|
{ |
|
return; |
|
} |
|
|
|
|
|
|
|
data_record temp(lhs->digits_used + slack); |
|
|
|
|
|
|
|
|
|
uint32 shifts; |
|
shifts = (lhs->digits_used - rhs->digits_used) * 16; |
|
|
|
shift_left(rhs,&temp,shifts); |
|
|
|
|
|
|
|
while (is_less_than(&temp,lhs)) |
|
{ |
|
shift_left(&temp,&temp,1); |
|
++shifts; |
|
} |
|
|
|
while (is_less_than(lhs,&temp)) |
|
{ |
|
shift_right(&temp,&temp); |
|
--shifts; |
|
} |
|
|
|
|
|
|
|
|
|
++shifts; |
|
while (shifts != 0) |
|
{ |
|
shift_left(result,result,1); |
|
|
|
if (!is_less_than(remainder,&temp)) |
|
{ |
|
long_sub(remainder,&temp,remainder); |
|
|
|
|
|
uint16* r = result->number; |
|
uint16* end = r + result->digits_used; |
|
while (true) |
|
{ |
|
++(*r); |
|
|
|
if (*r != 0) |
|
break; |
|
|
|
++r; |
|
|
|
|
|
|
|
if (r == end) |
|
{ |
|
*r = 1; |
|
++(result->digits_used); |
|
break; |
|
} |
|
} |
|
} |
|
shift_right(&temp,&temp); |
|
--shifts; |
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
long_mul ( |
|
const data_record* lhs, |
|
const data_record* rhs, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
if (std::min(lhs->digits_used, rhs->digits_used) < 10) |
|
{ |
|
|
|
result->digits_used = 1; |
|
*(result->number) = 0; |
|
|
|
|
|
const data_record* aa; |
|
const data_record* bb; |
|
|
|
if (lhs->digits_used < rhs->digits_used) |
|
{ |
|
|
|
|
|
aa = lhs; |
|
bb = rhs; |
|
} |
|
else |
|
{ |
|
|
|
|
|
aa = rhs; |
|
bb = lhs; |
|
} |
|
|
|
|
|
data_record b(*bb,aa->digits_used+slack); |
|
|
|
|
|
uint32 shift_value = 0; |
|
uint16* anum = aa->number; |
|
uint16* end = anum + aa->digits_used; |
|
while (anum != end ) |
|
{ |
|
uint16 bit = 0x0001; |
|
|
|
for (int i = 0; i < 16; ++i) |
|
{ |
|
|
|
if ((*anum & bit) != 0) |
|
{ |
|
shift_left(&b,&b,shift_value); |
|
shift_value = 0; |
|
long_add(&b,result,result); |
|
} |
|
++shift_value; |
|
bit <<= 1; |
|
} |
|
|
|
++anum; |
|
} |
|
} |
|
else |
|
|
|
{ |
|
uint32 size = 1; |
|
|
|
while (size < (lhs->digits_used + rhs->digits_used)*2) |
|
{ |
|
size *= 2; |
|
} |
|
|
|
|
|
ct* a = new ct[size]; |
|
ct* b; try {b = new ct[size]; } catch (...) { delete [] a; throw; } |
|
|
|
|
|
|
|
|
|
|
|
for (uint32 i = 0; i < lhs->digits_used; ++i) |
|
{ |
|
a[i*2] = ct((t)(lhs->number[i]&0xFF),0); |
|
a[i*2+1] = ct((t)(lhs->number[i]>>8),0); |
|
} |
|
for (uint32 i = lhs->digits_used*2; i < size; ++i) |
|
{ |
|
a[i] = 0; |
|
} |
|
|
|
|
|
for (uint32 i = 0; i < rhs->digits_used; ++i) |
|
{ |
|
b[i*2] = ct((t)(rhs->number[i]&0xFF),0); |
|
b[i*2+1] = ct((t)(rhs->number[i]>>8),0); |
|
} |
|
for (uint32 i = rhs->digits_used*2; i < size; ++i) |
|
{ |
|
b[i] = 0; |
|
} |
|
|
|
|
|
fft(a,size); |
|
fft(b,size); |
|
|
|
const double l = 1.0/size; |
|
|
|
|
|
|
|
for (unsigned long i = 0; i < size; ++i) |
|
{ |
|
a[i] = l*a[i]*b[i]; |
|
} |
|
|
|
|
|
|
|
|
|
ifft(a,size); |
|
|
|
|
|
|
|
|
|
uint64 carry = 0; |
|
result->digits_used = 0; |
|
int zeros = 0; |
|
const uint32 len = lhs->digits_used + rhs->digits_used; |
|
for (unsigned long i = 0; i < len; ++i) |
|
{ |
|
uint64 num1 = static_cast<uint64>(std::floor(a[i*2].real()+0.5)); |
|
num1 += carry; |
|
carry = 0; |
|
if (num1 > 255) |
|
{ |
|
carry = num1 >> 8; |
|
num1 = (num1&0xFF); |
|
} |
|
|
|
uint64 num2 = static_cast<uint64>(std::floor(a[i*2+1].real()+0.5)); |
|
num2 += carry; |
|
carry = 0; |
|
if (num2 > 255) |
|
{ |
|
carry = num2 >> 8; |
|
num2 = (num2&0xFF); |
|
} |
|
|
|
|
|
num1 = (num2<<8) | num1; |
|
result->number[i] = static_cast<uint16>(num1); |
|
|
|
|
|
if (num1 == 0) |
|
++zeros; |
|
else |
|
zeros = 0; |
|
++(result->digits_used); |
|
} |
|
|
|
|
|
|
|
result->digits_used -= zeros; |
|
|
|
|
|
if (result->digits_used == 0) |
|
{ |
|
|
|
result->digits_used = 1; |
|
*(result->number) = 0; |
|
} |
|
|
|
|
|
delete [] a; |
|
delete [] b; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
shift_left ( |
|
const data_record* data, |
|
data_record* result, |
|
uint32 shift_amount |
|
) const |
|
{ |
|
uint32 offset = shift_amount/16; |
|
shift_amount &= 0xf; |
|
|
|
uint16* r = result->number + data->digits_used + offset; |
|
uint16* end = data->number; |
|
uint16* s = end + data->digits_used; |
|
const uint32 temp = 16 - shift_amount; |
|
|
|
*r = (*(--s) >> temp); |
|
|
|
|
|
if (*r == 0) |
|
{ |
|
result->digits_used = data->digits_used + offset; |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used + offset + 1; |
|
} |
|
--r; |
|
|
|
while (s != end) |
|
{ |
|
*r = ((*s << shift_amount) | ( *(s-1) >> temp)); |
|
--r; |
|
--s; |
|
} |
|
*r = *s << shift_amount; |
|
|
|
|
|
end = result->number; |
|
while (r != end) |
|
*(--r) = 0; |
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
shift_right ( |
|
const data_record* data, |
|
data_record* result |
|
) const |
|
{ |
|
|
|
uint16* r = result->number; |
|
uint16* s = data->number; |
|
uint16* end = s + data->digits_used - 1; |
|
|
|
while (s != end) |
|
{ |
|
*r = (*s >> 1) | (*(s+1) << 15); |
|
++r; |
|
++s; |
|
} |
|
*r = *s >> 1; |
|
|
|
|
|
|
|
if (*r == 0) |
|
{ |
|
if (data->digits_used != 1) |
|
result->digits_used = data->digits_used - 1; |
|
else |
|
result->digits_used = 1; |
|
} |
|
else |
|
{ |
|
result->digits_used = data->digits_used; |
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
bool bigint_kernel_2:: |
|
is_less_than ( |
|
const data_record* lhs, |
|
const data_record* rhs |
|
) const |
|
{ |
|
uint32 lhs_digits_used = lhs->digits_used; |
|
uint32 rhs_digits_used = rhs->digits_used; |
|
|
|
|
|
if (lhs_digits_used < rhs_digits_used ) |
|
return true; |
|
|
|
else if (lhs_digits_used > rhs_digits_used) |
|
return false; |
|
else |
|
{ |
|
uint16* end = lhs->number; |
|
uint16* l = end + lhs_digits_used; |
|
uint16* r = rhs->number + rhs_digits_used; |
|
|
|
while (l != end) |
|
{ |
|
--l; |
|
--r; |
|
if (*l < *r) |
|
return true; |
|
else if (*l > *r) |
|
return false; |
|
} |
|
|
|
|
|
return false; |
|
} |
|
|
|
} |
|
|
|
|
|
|
|
bool bigint_kernel_2:: |
|
is_equal_to ( |
|
const data_record* lhs, |
|
const data_record* rhs |
|
) const |
|
{ |
|
|
|
if (lhs->digits_used != rhs->digits_used ) |
|
{ |
|
return false; |
|
} |
|
else |
|
{ |
|
uint16* l = lhs->number; |
|
uint16* r = rhs->number; |
|
uint16* end = l + lhs->digits_used; |
|
|
|
while (l != end) |
|
{ |
|
if (*l != *r) |
|
return false; |
|
++l; |
|
++r; |
|
} |
|
|
|
|
|
return true; |
|
} |
|
|
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
increment ( |
|
const data_record* source, |
|
data_record* dest |
|
) const |
|
{ |
|
uint16* s = source->number; |
|
uint16* d = dest->number; |
|
uint16* end = s + source->digits_used; |
|
while (true) |
|
{ |
|
*d = *s + 1; |
|
|
|
|
|
if (*d != 0) |
|
{ |
|
dest->digits_used = source->digits_used; |
|
|
|
|
|
++d; ++s; |
|
while (s != end) |
|
{ |
|
*d = *s; |
|
++d; |
|
++s; |
|
} |
|
|
|
break; |
|
} |
|
|
|
|
|
++s; |
|
|
|
|
|
|
|
if (s == end) |
|
{ |
|
++d; |
|
dest->digits_used = source->digits_used + 1; |
|
*d = 1; |
|
break; |
|
} |
|
|
|
++d; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
decrement ( |
|
const data_record* source, |
|
data_record* dest |
|
) const |
|
{ |
|
uint16* s = source->number; |
|
uint16* d = dest->number; |
|
uint16* end = s + source->digits_used; |
|
while (true) |
|
{ |
|
*d = *s - 1; |
|
|
|
|
|
if (*d != 0xFFFF) |
|
{ |
|
|
|
if (*d == 0 && s+1 == end) |
|
{ |
|
if (source->digits_used == 1) |
|
dest->digits_used = 1; |
|
else |
|
dest->digits_used = source->digits_used - 1; |
|
} |
|
else |
|
{ |
|
dest->digits_used = source->digits_used; |
|
} |
|
break; |
|
} |
|
else |
|
{ |
|
++d; |
|
++s; |
|
} |
|
|
|
} |
|
|
|
|
|
++d; |
|
++s; |
|
while (s != end) |
|
{ |
|
*d = *s; |
|
++d; |
|
++s; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
fft ( |
|
ct* data, |
|
unsigned long len |
|
) const |
|
{ |
|
const t pi2 = -2.0*3.1415926535897932384626433832795028841971693993751; |
|
|
|
const unsigned long half = len/2; |
|
|
|
std::vector<ct> twiddle_factors; |
|
twiddle_factors.resize(half); |
|
|
|
|
|
const t temp = pi2/len; |
|
ct w = ct(std::cos(temp),std::sin(temp)); |
|
|
|
ct w_pow = 1; |
|
|
|
|
|
for (std::vector<ct>::size_type j = 0; j < twiddle_factors.size(); ++j) |
|
{ |
|
twiddle_factors[j] = w_pow; |
|
w_pow *= w; |
|
} |
|
|
|
ct a, b; |
|
|
|
|
|
|
|
unsigned long skip = 1; |
|
for (unsigned long step = half; step != 0; step >>= 1) |
|
{ |
|
|
|
for (unsigned long j = 0; j < len; j += step*2) |
|
{ |
|
|
|
for (unsigned long k = 0; k < step; ++k) |
|
{ |
|
const unsigned long a_idx = j+k; |
|
const unsigned long b_idx = j+k+step; |
|
a = data[a_idx] + data[b_idx]; |
|
b = (data[a_idx] - data[b_idx])*twiddle_factors[k*skip]; |
|
data[a_idx] = a; |
|
data[b_idx] = b; |
|
} |
|
} |
|
skip *= 2; |
|
} |
|
} |
|
|
|
|
|
|
|
void bigint_kernel_2:: |
|
ifft( |
|
ct* data, |
|
unsigned long len |
|
) const |
|
{ |
|
const t pi2 = 2.0*3.1415926535897932384626433832795028841971693993751; |
|
|
|
const unsigned long half = len/2; |
|
|
|
std::vector<ct> twiddle_factors; |
|
twiddle_factors.resize(half); |
|
|
|
|
|
const t temp = pi2/len; |
|
ct w = ct(std::cos(temp),std::sin(temp)); |
|
|
|
ct w_pow = 1; |
|
|
|
|
|
for (std::vector<ct>::size_type j = 0; j < twiddle_factors.size(); ++j) |
|
{ |
|
twiddle_factors[j] = w_pow; |
|
w_pow *= w; |
|
} |
|
|
|
ct a, b; |
|
|
|
|
|
|
|
unsigned long skip = half; |
|
for (unsigned long step = 1; step <= half; step <<= 1) |
|
{ |
|
|
|
for (unsigned long j = 0; j < len; j += step*2) |
|
{ |
|
|
|
for (unsigned long k = 0; k < step; ++k) |
|
{ |
|
const unsigned long a_idx = j+k; |
|
const unsigned long b_idx = j+k+step; |
|
data[b_idx] *= twiddle_factors[k*skip]; |
|
a = data[a_idx] + data[b_idx]; |
|
b = data[a_idx] - data[b_idx]; |
|
data[a_idx] = a; |
|
data[b_idx] = b; |
|
} |
|
} |
|
skip /= 2; |
|
} |
|
} |
|
|
|
|
|
|
|
} |
|
#endif |
|
|
|
|