Source code

Revision control

Copy as Markdown

Other Tools

/*
* (C) 2024 Jack Lloyd
*
* Botan is released under the Simplified BSD License (see license.txt)
*/
#include <botan/internal/ec_inner_pc.h>
#include <botan/mem_ops.h>
namespace Botan {
const EC_Scalar_Data_PC& EC_Scalar_Data_PC::checked_ref(const EC_Scalar_Data& data) {
const auto* p = dynamic_cast<const EC_Scalar_Data_PC*>(&data);
if(!p) {
throw Invalid_State("Failed conversion to EC_Scalar_Data_PC");
}
return *p;
}
const std::shared_ptr<const EC_Group_Data>& EC_Scalar_Data_PC::group() const {
return m_group;
}
size_t EC_Scalar_Data_PC::bytes() const {
return this->group()->order_bytes();
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::clone() const {
return std::make_unique<EC_Scalar_Data_PC>(this->group(), this->value());
}
bool EC_Scalar_Data_PC::is_zero() const {
auto& pcurve = this->group()->pcurve();
return pcurve.scalar_is_zero(m_v);
}
bool EC_Scalar_Data_PC::is_eq(const EC_Scalar_Data& other) const {
auto& pcurve = group()->pcurve();
return pcurve.scalar_equal(m_v, checked_ref(other).m_v);
}
void EC_Scalar_Data_PC::assign(const EC_Scalar_Data& other) {
m_v = checked_ref(other).value();
}
void EC_Scalar_Data_PC::square_self() {
// TODO square in place
m_v = m_group->pcurve().scalar_square(m_v);
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::negate() const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, m_group->pcurve().scalar_negate(m_v));
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::invert() const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, m_group->pcurve().scalar_invert(m_v));
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::invert_vartime() const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, m_group->pcurve().scalar_invert_vartime(m_v));
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::add(const EC_Scalar_Data& other) const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, group()->pcurve().scalar_add(m_v, checked_ref(other).m_v));
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::sub(const EC_Scalar_Data& other) const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, group()->pcurve().scalar_sub(m_v, checked_ref(other).m_v));
}
std::unique_ptr<EC_Scalar_Data> EC_Scalar_Data_PC::mul(const EC_Scalar_Data& other) const {
return std::make_unique<EC_Scalar_Data_PC>(m_group, group()->pcurve().scalar_mul(m_v, checked_ref(other).m_v));
}
void EC_Scalar_Data_PC::serialize_to(std::span<uint8_t> bytes) const {
BOTAN_ARG_CHECK(bytes.size() == m_group->order_bytes(), "Invalid output length");
m_group->pcurve().serialize_scalar(bytes, m_v);
}
EC_AffinePoint_Data_PC::EC_AffinePoint_Data_PC(std::shared_ptr<const EC_Group_Data> group,
PCurve::PrimeOrderCurve::AffinePoint pt) :
m_group(std::move(group)), m_pt(std::move(pt)) {
auto& pcurve = m_group->pcurve();
if(!pcurve.affine_point_is_identity(m_pt)) {
m_xy.resize(1 + 2 * field_element_bytes());
pcurve.serialize_point(m_xy, m_pt);
}
}
const EC_AffinePoint_Data_PC& EC_AffinePoint_Data_PC::checked_ref(const EC_AffinePoint_Data& data) {
const auto* p = dynamic_cast<const EC_AffinePoint_Data_PC*>(&data);
if(!p) {
throw Invalid_State("Failed conversion to EC_AffinePoint_Data_PC");
}
return *p;
}
std::unique_ptr<EC_AffinePoint_Data> EC_AffinePoint_Data_PC::clone() const {
return std::make_unique<EC_AffinePoint_Data_PC>(m_group, m_pt);
}
const std::shared_ptr<const EC_Group_Data>& EC_AffinePoint_Data_PC::group() const {
return m_group;
}
std::unique_ptr<EC_AffinePoint_Data> EC_AffinePoint_Data_PC::mul(const EC_Scalar_Data& scalar,
RandomNumberGenerator& rng) const {
BOTAN_ARG_CHECK(scalar.group() == m_group, "Curve mismatch");
const auto& k = EC_Scalar_Data_PC::checked_ref(scalar).value();
auto& pcurve = m_group->pcurve();
auto pt = pcurve.point_to_affine(pcurve.mul(m_pt, k, rng));
return std::make_unique<EC_AffinePoint_Data_PC>(m_group, std::move(pt));
}
secure_vector<uint8_t> EC_AffinePoint_Data_PC::mul_x_only(const EC_Scalar_Data& scalar,
RandomNumberGenerator& rng) const {
BOTAN_ARG_CHECK(scalar.group() == m_group, "Curve mismatch");
const auto& k = EC_Scalar_Data_PC::checked_ref(scalar).value();
return m_group->pcurve().mul_x_only(m_pt, k, rng);
}
size_t EC_AffinePoint_Data_PC::field_element_bytes() const {
return m_group->pcurve().field_element_bytes();
}
bool EC_AffinePoint_Data_PC::is_identity() const {
return m_xy.empty();
}
void EC_AffinePoint_Data_PC::serialize_x_to(std::span<uint8_t> bytes) const {
BOTAN_STATE_CHECK(!this->is_identity());
const size_t fe_bytes = this->field_element_bytes();
BOTAN_ARG_CHECK(bytes.size() == fe_bytes, "Invalid output size");
copy_mem(bytes, std::span{m_xy}.subspan(1, fe_bytes));
}
void EC_AffinePoint_Data_PC::serialize_y_to(std::span<uint8_t> bytes) const {
BOTAN_STATE_CHECK(!this->is_identity());
const size_t fe_bytes = this->field_element_bytes();
BOTAN_ARG_CHECK(bytes.size() == fe_bytes, "Invalid output size");
copy_mem(bytes, std::span{m_xy}.subspan(1 + fe_bytes, fe_bytes));
}
void EC_AffinePoint_Data_PC::serialize_xy_to(std::span<uint8_t> bytes) const {
BOTAN_STATE_CHECK(!this->is_identity());
const size_t fe_bytes = this->field_element_bytes();
BOTAN_ARG_CHECK(bytes.size() == 2 * fe_bytes, "Invalid output size");
copy_mem(bytes, std::span{m_xy}.last(2 * fe_bytes));
}
void EC_AffinePoint_Data_PC::serialize_compressed_to(std::span<uint8_t> bytes) const {
BOTAN_STATE_CHECK(!this->is_identity());
const size_t fe_bytes = this->field_element_bytes();
BOTAN_ARG_CHECK(bytes.size() == 1 + fe_bytes, "Invalid output size");
const bool y_is_odd = (m_xy.back() & 0x01) == 0x01;
BufferStuffer stuffer(bytes);
stuffer.append(y_is_odd ? 0x03 : 0x02);
this->serialize_x_to(stuffer.next(fe_bytes));
}
void EC_AffinePoint_Data_PC::serialize_uncompressed_to(std::span<uint8_t> bytes) const {
BOTAN_STATE_CHECK(!this->is_identity());
const size_t fe_bytes = this->field_element_bytes();
BOTAN_ARG_CHECK(bytes.size() == 1 + 2 * fe_bytes, "Invalid output size");
copy_mem(bytes, m_xy);
}
#if defined(BOTAN_HAS_LEGACY_EC_POINT)
EC_Point EC_AffinePoint_Data_PC::to_legacy_point() const {
if(this->is_identity()) {
return EC_Point(m_group->curve());
} else {
const size_t fe_bytes = this->field_element_bytes();
return EC_Point(m_group->curve(),
BigInt::from_bytes(std::span{m_xy}.subspan(1, fe_bytes)),
BigInt::from_bytes(std::span{m_xy}.last(fe_bytes)));
}
}
#endif
EC_Mul2Table_Data_PC::EC_Mul2Table_Data_PC(const EC_AffinePoint_Data& q) : m_group(q.group()) {
BOTAN_ARG_CHECK(q.group() == m_group, "Curve mismatch");
const auto& pt_q = EC_AffinePoint_Data_PC::checked_ref(q);
m_tbl = m_group->pcurve().mul2_setup_g(pt_q.value());
}
std::unique_ptr<EC_AffinePoint_Data> EC_Mul2Table_Data_PC::mul2_vartime(const EC_Scalar_Data& xd,
const EC_Scalar_Data& yd) const {
BOTAN_ARG_CHECK(xd.group() == m_group && yd.group() == m_group, "Curve mismatch");
const auto& x = EC_Scalar_Data_PC::checked_ref(xd);
const auto& y = EC_Scalar_Data_PC::checked_ref(yd);
auto& pcurve = m_group->pcurve();
if(auto pt = pcurve.mul2_vartime(*m_tbl, x.value(), y.value())) {
return std::make_unique<EC_AffinePoint_Data_PC>(m_group, pcurve.point_to_affine(*pt));
} else {
return nullptr;
}
}
bool EC_Mul2Table_Data_PC::mul2_vartime_x_mod_order_eq(const EC_Scalar_Data& vd,
const EC_Scalar_Data& xd,
const EC_Scalar_Data& yd) const {
BOTAN_ARG_CHECK(xd.group() == m_group && yd.group() == m_group, "Curve mismatch");
const auto& v = EC_Scalar_Data_PC::checked_ref(vd);
const auto& x = EC_Scalar_Data_PC::checked_ref(xd);
const auto& y = EC_Scalar_Data_PC::checked_ref(yd);
return m_group->pcurve().mul2_vartime_x_mod_order_eq(*m_tbl, v.value(), x.value(), y.value());
}
} // namespace Botan