mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2024-12-15 09:11:24 +01:00
148 lines
4.8 KiB
C++
148 lines
4.8 KiB
C++
|
/*
|
||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
* version 2, as published by the Free Software Foundation.
|
||
|
*
|
||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||
|
* more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
#include <vapours.hpp>
|
||
|
|
||
|
namespace ams::crypto::impl {
|
||
|
|
||
|
void BigNum::ImportImpl(Word *out, size_t out_size, const u8 *src, size_t src_size) {
|
||
|
size_t octet_ofs = src_size;
|
||
|
size_t word_ofs = 0;
|
||
|
|
||
|
/* Parse octets into words. */
|
||
|
while (word_ofs < out_size && octet_ofs > 0) {
|
||
|
Word w = 0;
|
||
|
for (size_t shift = 0; octet_ofs > 0 && shift < BITSIZEOF(Word); shift += BITSIZEOF(u8)) {
|
||
|
w |= static_cast<Word>(src[--octet_ofs]) << shift;
|
||
|
}
|
||
|
out[word_ofs++] = w;
|
||
|
}
|
||
|
|
||
|
/* Zero-fill upper words. */
|
||
|
while (word_ofs < out_size) {
|
||
|
out[word_ofs++] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void BigNum::ExportImpl(u8 *out, size_t out_size, const Word *src, size_t src_size) {
|
||
|
size_t octet_ofs = out_size;
|
||
|
|
||
|
/* Parse words into octets. */
|
||
|
for (size_t word_ofs = 0; word_ofs < src_size && octet_ofs > 0; word_ofs++) {
|
||
|
const Word w = src[word_ofs];
|
||
|
for (size_t shift = 0; octet_ofs > 0 && shift < BITSIZEOF(Word); shift += BITSIZEOF(u8)) {
|
||
|
out[--octet_ofs] = static_cast<u8>(w >> shift);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Zero-clear remaining octets. */
|
||
|
while (octet_ofs > 0) {
|
||
|
out[--octet_ofs] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
size_t BigNum::GetSize() const {
|
||
|
if (this->num_words == 0) {
|
||
|
return 0;
|
||
|
}
|
||
|
static_assert(sizeof(Word) == 4);
|
||
|
|
||
|
size_t size = this->num_words * sizeof(Word);
|
||
|
const Word last = this->words[this->num_words - 1];
|
||
|
AMS_ASSERT(last != 0);
|
||
|
if (last >= 0x01000000u) {
|
||
|
return size - 0;
|
||
|
} else if (last >= 0x00010000u) {
|
||
|
return size - 1;
|
||
|
} else if (last >= 0x00000100u) {
|
||
|
return size - 2;
|
||
|
} else {
|
||
|
return size - 3;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool BigNum::Import(const void *src, size_t src_size) {
|
||
|
AMS_ASSERT((src != nullptr) || (src_size != 0));
|
||
|
|
||
|
/* Ignore leading zeroes. */
|
||
|
const u8 *data = static_cast<const u8 *>(src);
|
||
|
while (src_size > 0 && *data == 0) {
|
||
|
++data;
|
||
|
--src_size;
|
||
|
}
|
||
|
|
||
|
/* Ensure we have space for the number. */
|
||
|
AMS_ASSERT(src_size <= this->max_words * sizeof(Word));
|
||
|
if (AMS_UNLIKELY(!(src_size <= this->max_words * sizeof(Word)))) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* Import. */
|
||
|
this->num_words = util::AlignUp(src_size, sizeof(Word)) / sizeof(Word);
|
||
|
|
||
|
ImportImpl(this->words, this->max_words, data, src_size);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void BigNum::Export(void *dst, size_t dst_size) {
|
||
|
AMS_ASSERT(dst_size >= this->GetSize());
|
||
|
ExportImpl(static_cast<u8 *>(dst), dst_size, this->words, this->num_words);
|
||
|
}
|
||
|
|
||
|
bool BigNum::ExpMod(void *dst, const void *src, size_t size, const BigNum &exp, u32 *work_buf, size_t work_buf_size) const {
|
||
|
/* Can't exponentiate with or about zero. */
|
||
|
if (this->IsZero() || exp.IsZero()) {
|
||
|
return false;
|
||
|
}
|
||
|
AMS_ASSERT(size == this->GetSize());
|
||
|
|
||
|
/* Create an allocator. */
|
||
|
WordAllocator allocator(work_buf, work_buf_size / sizeof(Word));
|
||
|
ON_SCOPE_EXIT { ClearMemory(work_buf, allocator.GetMaxUsedSize()); };
|
||
|
|
||
|
/* Create a BigNum for the signature. */
|
||
|
BigNum signature;
|
||
|
auto signature_words = allocator.Allocate(size / sizeof(Word));
|
||
|
if (!signature_words.IsValid()) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* Import data for the signature. */
|
||
|
signature.ReserveStatic(signature_words.GetBuffer(), signature_words.GetCount());
|
||
|
if (!signature.Import(src, size)) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* Perform the exponentiation. */
|
||
|
if (!ExpMod(signature.words, signature.words, exp.words, exp.num_words, this->words, this->num_words, std::addressof(allocator))) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/* We succeeded, so export. */
|
||
|
signature.UpdateCount();
|
||
|
signature.Export(dst, size);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void BigNum::ClearToZero() {
|
||
|
std::memset(this->words, 0, this->num_words * sizeof(Word));
|
||
|
}
|
||
|
|
||
|
void BigNum::UpdateCount() {
|
||
|
this->num_words = CountWords(this->words, this->max_words);
|
||
|
}
|
||
|
|
||
|
}
|