Path: blob/a-new-beginning/SharedDependencies/Sources/cryptopp/ec2n.cpp
2 views
// ec2n.cpp - originally written and placed in the public domain by Wei Dai12#include "pch.h"34#ifndef CRYPTOPP_IMPORTS56#include "ec2n.h"7#include "asn.h"8#include "integer.h"9#include "filters.h"10#include "algebra.cpp"11#include "eprecomp.cpp"1213ANONYMOUS_NAMESPACE_BEGIN1415using CryptoPP::EC2N;1617#if defined(HAVE_GCC_INIT_PRIORITY)18#define INIT_ATTRIBUTE __attribute__ ((init_priority (CRYPTOPP_INIT_PRIORITY + 51)))19const EC2N::Point g_identity INIT_ATTRIBUTE = EC2N::Point();20#elif defined(HAVE_MSC_INIT_PRIORITY)21#pragma warning(disable: 4075)22#pragma init_seg(".CRT$XCU")23const EC2N::Point g_identity;24#pragma warning(default: 4075)25#elif defined(HAVE_XLC_INIT_PRIORITY)26#pragma priority(290)27const EC2N::Point g_identity;28#endif2930ANONYMOUS_NAMESPACE_END3132NAMESPACE_BEGIN(CryptoPP)3334EC2N::EC2N(BufferedTransformation &bt)35: m_field(BERDecodeGF2NP(bt))36{37BERSequenceDecoder seq(bt);38m_field->BERDecodeElement(seq, m_a);39m_field->BERDecodeElement(seq, m_b);40// skip optional seed41if (!seq.EndReached())42{43SecByteBlock seed;44unsigned int unused;45BERDecodeBitString(seq, seed, unused);46}47seq.MessageEnd();48}4950void EC2N::DEREncode(BufferedTransformation &bt) const51{52m_field->DEREncode(bt);53DERSequenceEncoder seq(bt);54m_field->DEREncodeElement(seq, m_a);55m_field->DEREncodeElement(seq, m_b);56seq.MessageEnd();57}5859bool EC2N::DecodePoint(EC2N::Point &P, const byte *encodedPoint, size_t encodedPointLen) const60{61StringStore store(encodedPoint, encodedPointLen);62return DecodePoint(P, store, encodedPointLen);63}6465bool EC2N::DecodePoint(EC2N::Point &P, BufferedTransformation &bt, size_t encodedPointLen) const66{67byte type;68if (encodedPointLen < 1 || !bt.Get(type))69return false;7071switch (type)72{73case 0:74P.identity = true;75return true;76case 2:77case 3:78{79if (encodedPointLen != EncodedPointSize(true))80return false;8182P.identity = false;83P.x.Decode(bt, m_field->MaxElementByteLength());8485if (P.x.IsZero())86{87P.y = m_field->SquareRoot(m_b);88return true;89}9091FieldElement z = m_field->Square(P.x);92CRYPTOPP_ASSERT(P.x == m_field->SquareRoot(z));93P.y = m_field->Divide(m_field->Add(m_field->Multiply(z, m_field->Add(P.x, m_a)), m_b), z);94CRYPTOPP_ASSERT(P.x == m_field->Subtract(m_field->Divide(m_field->Subtract(m_field->Multiply(P.y, z), m_b), z), m_a));95z = m_field->SolveQuadraticEquation(P.y);96CRYPTOPP_ASSERT(m_field->Add(m_field->Square(z), z) == P.y);97z.SetCoefficient(0, type & 1);9899P.y = m_field->Multiply(z, P.x);100return true;101}102case 4:103{104if (encodedPointLen != EncodedPointSize(false))105return false;106107unsigned int len = m_field->MaxElementByteLength();108P.identity = false;109P.x.Decode(bt, len);110P.y.Decode(bt, len);111return true;112}113default:114return false;115}116}117118void EC2N::EncodePoint(BufferedTransformation &bt, const Point &P, bool compressed) const119{120if (P.identity)121NullStore().TransferTo(bt, EncodedPointSize(compressed));122else if (compressed)123{124bt.Put((byte)(2U + (!P.x ? 0U : m_field->Divide(P.y, P.x).GetBit(0))));125P.x.Encode(bt, m_field->MaxElementByteLength());126}127else128{129unsigned int len = m_field->MaxElementByteLength();130bt.Put(4); // uncompressed131P.x.Encode(bt, len);132P.y.Encode(bt, len);133}134}135136void EC2N::EncodePoint(byte *encodedPoint, const Point &P, bool compressed) const137{138ArraySink sink(encodedPoint, EncodedPointSize(compressed));139EncodePoint(sink, P, compressed);140CRYPTOPP_ASSERT(sink.TotalPutLength() == EncodedPointSize(compressed));141}142143EC2N::Point EC2N::BERDecodePoint(BufferedTransformation &bt) const144{145SecByteBlock str;146BERDecodeOctetString(bt, str);147Point P;148if (!DecodePoint(P, str, str.size()))149BERDecodeError();150return P;151}152153void EC2N::DEREncodePoint(BufferedTransformation &bt, const Point &P, bool compressed) const154{155SecByteBlock str(EncodedPointSize(compressed));156EncodePoint(str, P, compressed);157DEREncodeOctetString(bt, str);158}159160bool EC2N::ValidateParameters(RandomNumberGenerator &rng, unsigned int level) const161{162CRYPTOPP_UNUSED(rng);163bool pass = !!m_b;164pass = pass && m_a.CoefficientCount() <= m_field->MaxElementBitLength();165pass = pass && m_b.CoefficientCount() <= m_field->MaxElementBitLength();166167if (level >= 1)168pass = pass && m_field->GetModulus().IsIrreducible();169170return pass;171}172173bool EC2N::VerifyPoint(const Point &P) const174{175const FieldElement &x = P.x, &y = P.y;176return P.identity ||177(x.CoefficientCount() <= m_field->MaxElementBitLength()178&& y.CoefficientCount() <= m_field->MaxElementBitLength()179&& !(((x+m_a)*x*x+m_b-(x+y)*y)%m_field->GetModulus()));180}181182bool EC2N::Equal(const Point &P, const Point &Q) const183{184if (P.identity && Q.identity)185return true;186187if (P.identity && !Q.identity)188return false;189190if (!P.identity && Q.identity)191return false;192193return (m_field->Equal(P.x,Q.x) && m_field->Equal(P.y,Q.y));194}195196const EC2N::Point& EC2N::Identity() const197{198#if defined(HAVE_GCC_INIT_PRIORITY) || defined(HAVE_MSC_INIT_PRIORITY) || defined(HAVE_XLC_INIT_PRIORITY)199return g_identity;200#elif defined(CRYPTOPP_CXX11_STATIC_INIT)201static const EC2N::Point g_identity;202return g_identity;203#else204return Singleton<Point>().Ref();205#endif206}207208const EC2N::Point& EC2N::Inverse(const Point &P) const209{210if (P.identity)211return P;212else213{214m_R.identity = false;215m_R.y = m_field->Add(P.x, P.y);216m_R.x = P.x;217return m_R;218}219}220221const EC2N::Point& EC2N::Add(const Point &P, const Point &Q) const222{223if (P.identity) return Q;224if (Q.identity) return P;225if (Equal(P, Q)) return Double(P);226if (m_field->Equal(P.x, Q.x) && m_field->Equal(P.y, m_field->Add(Q.x, Q.y))) return Identity();227228FieldElement t = m_field->Add(P.y, Q.y);229t = m_field->Divide(t, m_field->Add(P.x, Q.x));230FieldElement x = m_field->Square(t);231m_field->Accumulate(x, t);232m_field->Accumulate(x, Q.x);233m_field->Accumulate(x, m_a);234m_R.y = m_field->Add(P.y, m_field->Multiply(t, x));235m_field->Accumulate(x, P.x);236m_field->Accumulate(m_R.y, x);237238m_R.x.swap(x);239m_R.identity = false;240return m_R;241}242243const EC2N::Point& EC2N::Double(const Point &P) const244{245if (P.identity) return P;246if (!m_field->IsUnit(P.x)) return Identity();247248FieldElement t = m_field->Divide(P.y, P.x);249m_field->Accumulate(t, P.x);250m_R.y = m_field->Square(P.x);251m_R.x = m_field->Square(t);252m_field->Accumulate(m_R.x, t);253m_field->Accumulate(m_R.x, m_a);254m_field->Accumulate(m_R.y, m_field->Multiply(t, m_R.x));255m_field->Accumulate(m_R.y, m_R.x);256257m_R.identity = false;258return m_R;259}260261// ********************************************************262263#if 0264EcPrecomputation<EC2N>& EcPrecomputation<EC2N>::operator=(const EcPrecomputation<EC2N> &rhs)265{266m_ec = rhs.m_ec;267m_ep = rhs.m_ep;268m_ep.m_group = m_ec.get();269return *this;270}271272void EcPrecomputation<EC2N>::SetCurveAndBase(const EC2N &ec, const EC2N::Point &base)273{274m_ec.reset(new EC2N(ec));275m_ep.SetGroupAndBase(*m_ec, base);276}277278void EcPrecomputation<EC2N>::Precompute(unsigned int maxExpBits, unsigned int storage)279{280m_ep.Precompute(maxExpBits, storage);281}282283void EcPrecomputation<EC2N>::Load(BufferedTransformation &bt)284{285BERSequenceDecoder seq(bt);286word32 version;287BERDecodeUnsigned<word32>(seq, version, INTEGER, 1, 1);288m_ep.m_exponentBase.BERDecode(seq);289m_ep.m_windowSize = m_ep.m_exponentBase.BitCount() - 1;290m_ep.m_bases.clear();291while (!seq.EndReached())292m_ep.m_bases.push_back(m_ec->BERDecodePoint(seq));293seq.MessageEnd();294}295296void EcPrecomputation<EC2N>::Save(BufferedTransformation &bt) const297{298DERSequenceEncoder seq(bt);299DEREncodeUnsigned<word32>(seq, 1); // version300m_ep.m_exponentBase.DEREncode(seq);301for (unsigned i=0; i<m_ep.m_bases.size(); i++)302m_ec->DEREncodePoint(seq, m_ep.m_bases[i]);303seq.MessageEnd();304}305306EC2N::Point EcPrecomputation<EC2N>::Exponentiate(const Integer &exponent) const307{308return m_ep.Exponentiate(exponent);309}310311EC2N::Point EcPrecomputation<EC2N>::CascadeExponentiate(const Integer &exponent, const DL_FixedBasePrecomputation<Element> &pc2, const Integer &exponent2) const312{313return m_ep.CascadeExponentiate(exponent, static_cast<const EcPrecomputation<EC2N> &>(pc2).m_ep, exponent2);314}315#endif316317NAMESPACE_END318319#endif320321322