mirror of https://github.com/PurpleI2P/i2pd.git
I2P: End-to-End encrypted and anonymous Internet
https://i2pd.website/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
88 lines
2.6 KiB
88 lines
2.6 KiB
11 years ago
|
#ifndef EL_GAMAL_H__
|
||
|
#define EL_GAMAL_H__
|
||
|
|
||
|
#include <inttypes.h>
|
||
|
#include <cryptopp/integer.h>
|
||
|
#include <cryptopp/osrng.h>
|
||
10 years ago
|
#include <cryptopp/dh.h>
|
||
11 years ago
|
#include <cryptopp/sha.h>
|
||
|
#include "CryptoConst.h"
|
||
|
#include "Log.h"
|
||
|
|
||
|
namespace i2p
|
||
|
{
|
||
|
namespace crypto
|
||
|
{
|
||
11 years ago
|
|
||
10 years ago
|
class ElGamalEncryption
|
||
|
{
|
||
|
public:
|
||
11 years ago
|
|
||
10 years ago
|
ElGamalEncryption (const uint8_t * key)
|
||
|
{
|
||
|
CryptoPP::AutoSeededRandomPool rnd;
|
||
|
CryptoPP::Integer y (key, 256), k (rnd, CryptoPP::Integer::One(), elgp-1);
|
||
|
a = a_exp_b_mod_c (elgg, k, elgp);
|
||
|
b1 = a_exp_b_mod_c (y, k, elgp);
|
||
|
}
|
||
11 years ago
|
|
||
10 years ago
|
void Encrypt (const uint8_t * data, int len, uint8_t * encrypted, bool zeroPadding = false) const
|
||
|
{
|
||
|
// calculate b = b1*m mod p
|
||
|
uint8_t m[255];
|
||
|
m[0] = 0xFF;
|
||
|
memcpy (m+33, data, len);
|
||
|
CryptoPP::SHA256().CalculateDigest(m+1, m+33, 222);
|
||
|
CryptoPP::Integer b (a_times_b_mod_c (b1, CryptoPP::Integer (m, 255), elgp));
|
||
11 years ago
|
|
||
10 years ago
|
// copy a and b
|
||
|
if (zeroPadding)
|
||
|
{
|
||
|
encrypted[0] = 0;
|
||
|
a.Encode (encrypted + 1, 256);
|
||
|
encrypted[257] = 0;
|
||
|
b.Encode (encrypted + 258, 256);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
a.Encode (encrypted, 256);
|
||
|
b.Encode (encrypted + 256, 256);
|
||
|
}
|
||
|
}
|
||
11 years ago
|
|
||
10 years ago
|
private:
|
||
11 years ago
|
|
||
10 years ago
|
CryptoPP::Integer a, b1;
|
||
|
};
|
||
11 years ago
|
|
||
10 years ago
|
inline bool ElGamalDecrypt (const uint8_t * key, const uint8_t * encrypted,
|
||
|
uint8_t * data, bool zeroPadding = false)
|
||
|
{
|
||
|
CryptoPP::Integer x(key, 256), a(zeroPadding? encrypted +1 : encrypted, 256),
|
||
|
b(zeroPadding? encrypted + 258 :encrypted + 256, 256);
|
||
|
uint8_t m[255];
|
||
|
a_times_b_mod_c (b, a_exp_b_mod_c (a, elgp - x - 1, elgp), elgp).Encode (m, 255);
|
||
|
if (!CryptoPP::SHA256().VerifyDigest (m + 1, m + 33, 222))
|
||
|
{
|
||
|
LogPrint ("ElGamal decrypt hash doesn't match");
|
||
|
return false;
|
||
|
}
|
||
|
memcpy (data, m + 33, 222);
|
||
|
return true;
|
||
|
}
|
||
10 years ago
|
|
||
10 years ago
|
inline void GenerateElGamalKeyPair (CryptoPP::RandomNumberGenerator& rnd, uint8_t * priv, uint8_t * pub)
|
||
|
{
|
||
|
#if defined(__x86_64__) || defined(__i386__) || defined(_MSC_VER)
|
||
|
rnd.GenerateBlock (priv, 256);
|
||
|
a_exp_b_mod_c (elgg, CryptoPP::Integer (priv, 256), elgp).Encode (pub, 256);
|
||
10 years ago
|
#else
|
||
10 years ago
|
CryptoPP::DH dh (elgp, elgg);
|
||
|
dh.GenerateKeyPair(rnd, priv, pub);
|
||
|
#endif
|
||
|
}
|
||
11 years ago
|
}
|
||
10 years ago
|
}
|
||
11 years ago
|
|
||
|
#endif
|