feat(big): add SBigSquare

This commit is contained in:
fallenoak 2023-02-03 08:15:06 -06:00
parent f01c906aa1
commit 4a1b6a0f19
No known key found for this signature in database
GPG key ID: 7628F8E61AEA070D
5 changed files with 176 additions and 0 deletions

View file

@ -49,6 +49,10 @@ void SBigMul(BigData* a, BigData* b, BigData* c) {
Mul(a->Primary(), b->Primary(), c->Primary(), a->Stack()); Mul(a->Primary(), b->Primary(), c->Primary(), a->Stack());
} }
void SBigSquare(BigData* a, BigData* b) {
Square(a->Primary(), b->Primary(), a->Stack());
}
void SBigSub(BigData* a, BigData* b, BigData* c) { void SBigSub(BigData* a, BigData* b, BigData* c) {
Sub(a->Primary(), b->Primary(), c->Primary()); Sub(a->Primary(), b->Primary(), c->Primary());
} }

View file

@ -20,6 +20,8 @@ void SBigMul(BigData* a, BigData* b, BigData* c);
void SBigNew(BigData** num); void SBigNew(BigData** num);
void SBigSquare(BigData* a, BigData* b);
void SBigSub(BigData* a, BigData* b, BigData* c); void SBigSub(BigData* a, BigData* b, BigData* c);
void SBigToBinaryBuffer(BigData* num, uint8_t* data, uint32_t maxBytes, uint32_t* bytes); void SBigToBinaryBuffer(BigData* num, uint8_t* data, uint32_t maxBytes, uint32_t* bytes);

View file

@ -170,6 +170,32 @@ void SetZero(BigBuffer& buffer) {
buffer.Clear(); buffer.Clear();
} }
void Square(BigBuffer& a, const BigBuffer& b, BigStack& stack) {
auto& aa = stack.MakeDistinct(a, &a == &b);
aa.Clear();
uint32_t i = 0;
for (i = 0; b.IsUsed(i); i++) {
uint64_t carry = 0;
uint32_t j = 0;
for (j = 0; j <= i; j++) {
auto mul = b[i] * static_cast<uint64_t>(b[j]);
auto add = mul + aa[i + j];
if (j < i) {
carry += mul;
}
aa[i + j] = ExtractLowPartLargeSum(carry, add);
}
aa[i + j] = ExtractLowPart(carry);
}
stack.UnmakeDistinct(a, aa);
}
void Sub(BigBuffer& a, const BigBuffer& b, const BigBuffer& c) { void Sub(BigBuffer& a, const BigBuffer& b, const BigBuffer& c) {
uint64_t borrow = 0; uint64_t borrow = 0;
uint32_t i = 0; uint32_t i = 0;

View file

@ -37,6 +37,8 @@ void SetOne(BigBuffer& buffer);
void SetZero(BigBuffer& buffer); void SetZero(BigBuffer& buffer);
void Square(BigBuffer& a, const BigBuffer& b, BigStack& stack);
void Sub(BigBuffer& a, const BigBuffer& b, const BigBuffer& c); void Sub(BigBuffer& a, const BigBuffer& b, const BigBuffer& c);
void ToBinary(TSGrowableArray<uint8_t>& output, const BigBuffer& buffer); void ToBinary(TSGrowableArray<uint8_t>& output, const BigBuffer& buffer);

View file

@ -855,6 +855,28 @@ TEST_CASE("SBigMul", "[big]") {
} }
} }
TEST_CASE("SBigSquare", "[square]") {
SECTION("squares 0xFFFFFFFF") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 0xFFFFFFFF);
SBigSquare(a, b);
a->Primary().Trim();
CHECK(a->Primary().Count() == 2);
CHECK(a->Primary()[0] == 0x1);
CHECK(a->Primary()[1] == 0xFFFFFFFE);
SBigDel(a);
SBigDel(b);
}
}
TEST_CASE("SBigSub", "[big]") { TEST_CASE("SBigSub", "[big]") {
SECTION("subtracts 1 from 2") { SECTION("subtracts 1 from 2") {
BigData* a; BigData* a;
@ -961,6 +983,126 @@ TEST_CASE("SetZero", "[big]") {
} }
} }
TEST_CASE("Square", "[big]") {
SECTION("squares 0") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 0);
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 0);
SBigDel(a);
SBigDel(b);
}
SECTION("squares 1") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 1);
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 1);
CHECK(a->Primary()[0] == 1);
SBigDel(a);
SBigDel(b);
}
SECTION("squares 2") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 2);
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 1);
CHECK(a->Primary()[0] == 4);
SBigDel(a);
SBigDel(b);
}
SECTION("squares 2") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 2);
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 1);
CHECK(a->Primary()[0] == 4);
SBigDel(a);
SBigDel(b);
}
SECTION("squares 0xFFFFFFFF") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
SBigFromUnsigned(b, 0xFFFFFFFF);
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 2);
CHECK(a->Primary()[0] == 0x1);
CHECK(a->Primary()[1] == 0xFFFFFFFE);
SBigDel(a);
SBigDel(b);
}
SECTION("squares 0x1111111111111111") {
BigData* a;
SBigNew(&a);
BigData* b;
SBigNew(&b);
uint64_t b_ = 0x1111111111111111;
SBigFromBinary(b, reinterpret_cast<uint8_t*>(&b_), sizeof(b_));
Square(a->Primary(), b->Primary(), a->Stack());
a->Primary().Trim();
CHECK(a->Primary().Count() == 4);
CHECK(a->Primary()[0] == 0x87654321);
CHECK(a->Primary()[1] == 0xfedcba9);
CHECK(a->Primary()[2] == 0x89abcdf0);
CHECK(a->Primary()[3] == 0x1234567);
SBigDel(a);
SBigDel(b);
}
}
TEST_CASE("Sub", "[big]") { TEST_CASE("Sub", "[big]") {
SECTION("subtracts 0 from 1") { SECTION("subtracts 0 from 1") {
BigData* a; BigData* a;