summaryrefslogtreecommitdiffstats
path: root/crypto
diff options
context:
space:
mode:
authoragl@chromium.org <agl@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-07-07 13:39:45 +0000
committeragl@chromium.org <agl@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-07-07 13:39:45 +0000
commit9b8c962632b7f107d346f63bfedcb028061da043 (patch)
tree8378a4535ce665b96c1414165859b393e7d4cff5 /crypto
parentf9d5d0855d0cdd62e0b1af22797d69cab5a2876d (diff)
downloadchromium_src-9b8c962632b7f107d346f63bfedcb028061da043.zip
chromium_src-9b8c962632b7f107d346f63bfedcb028061da043.tar.gz
chromium_src-9b8c962632b7f107d346f63bfedcb028061da043.tar.bz2
crypto: convert OpenPGP code to NSS
(This is a reland of r91350 which was reverted in r91355 (GYP issue), landed again in r91559 and reverted again in r91561 (Windows shared build issue).) BUG=none TEST=crypto_unittests http://codereview.chromium.org/7273080 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@91699 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'crypto')
-rw-r--r--crypto/crypto.gyp15
-rw-r--r--crypto/openpgp_symmetric_encryption.cc (renamed from crypto/openpgp_symmetric_encryption_openssl.cc)297
-rw-r--r--crypto/openpgp_symmetric_encryption_unittest.cc (renamed from crypto/openpgp_symmetric_encryption_test_openssl.cc)0
3 files changed, 200 insertions, 112 deletions
diff --git a/crypto/crypto.gyp b/crypto/crypto.gyp
index e3d6c84..53a41ef5 100644
--- a/crypto/crypto.gyp
+++ b/crypto/crypto.gyp
@@ -35,6 +35,9 @@
'sources/': [
['exclude', '_nss\.cc$'],
],
+ 'sources!': [
+ 'openpgp_symmetric_encryption.cc',
+ ],
}],
[ 'OS == "freebsd" or OS == "openbsd"', {
'link_settings': {
@@ -92,6 +95,7 @@
'hmac_nss.cc',
'nss_util.cc',
'nss_util.h',
+ 'openpgp_symmetric_encryption.cc',
'rsa_private_key_nss.cc',
'secure_hash_default.cc',
'signature_creator_nss.cc',
@@ -106,7 +110,6 @@
'sources!': [
'encryptor_openssl.cc',
'hmac_openssl.cc',
- 'openpgp_symmetric_encryption_openssl.cc',
'openssl_util.cc',
'openssl_util.h',
'rsa_private_key_openssl.cc',
@@ -141,7 +144,7 @@
'nss_util.cc',
'nss_util.h',
'nss_util_internal.h',
- 'openpgp_symmetric_encryption_openssl.cc',
+ 'openpgp_symmetric_encryption.cc',
'openpgp_symmetric_encryption.h',
'openssl_util.cc',
'openssl_util.h',
@@ -196,7 +199,7 @@
'signature_creator_unittest.cc',
'signature_verifier_unittest.cc',
'symmetric_key_unittest.cc',
- 'openpgp_symmetric_encryption_test_openssl.cc',
+ 'openpgp_symmetric_encryption_unittest.cc',
],
'dependencies': [
'crypto',
@@ -221,6 +224,7 @@
}, { # os_posix != 1 or OS == "mac"
'sources!': [
'rsa_private_key_nss_unittest.cc',
+ 'openpgp_symmetric_encryption_unittest.cc',
]
}],
[ 'OS == "mac" or OS == "win"', {
@@ -230,12 +234,9 @@
}],
[ 'use_openssl==1', {
'sources!': [
+ 'openpgp_symmetric_encryption_unittest.cc',
'rsa_private_key_nss_unittest.cc',
],
- }, {
- 'sources!': [
- 'openpgp_symmetric_encryption_test_openssl.cc',
- ],
}],
],
},
diff --git a/crypto/openpgp_symmetric_encryption_openssl.cc b/crypto/openpgp_symmetric_encryption.cc
index a1685be..3f37d4c 100644
--- a/crypto/openpgp_symmetric_encryption_openssl.cc
+++ b/crypto/openpgp_symmetric_encryption.cc
@@ -4,15 +4,17 @@
#include "crypto/openpgp_symmetric_encryption.h"
-#include <vector>
#include <stdlib.h>
-#include <openssl/evp.h>
-#include <openssl/aes.h>
-#include <openssl/sha.h>
+#include <sechash.h>
+#include <cryptohi.h>
+
+#include <vector>
-#include "base/rand_util.h"
#include "base/logging.h"
+#include "base/rand_util.h"
+#include "crypto/scoped_nss_types.h"
+#include "crypto/nss_util.h"
namespace crypto {
@@ -47,7 +49,7 @@ class Reader {
// Prefix sets |*out| to the first |n| bytes of the StringPiece and advances
// the StringPiece by |n|.
- bool Prefix(uint32 n, base::StringPiece *out) {
+ bool Prefix(size_t n, base::StringPiece *out) {
if (data_.size() < n)
return false;
*out = base::StringPiece(data_.data(), n);
@@ -73,7 +75,7 @@ class Reader {
data_ = p;
}
- bool Skip(uint32 n) {
+ bool Skip(size_t n) {
if (data_.size() < n)
return false;
data_.remove_prefix(n);
@@ -94,59 +96,86 @@ class Reader {
// SaltedIteratedS2K implements the salted and iterated string-to-key
// convertion. See RFC 4880, section 3.7.1.3.
-void SaltedIteratedS2K(uint32 cipher_key_length,
- const EVP_MD *hash_function,
+void SaltedIteratedS2K(unsigned cipher_key_length,
+ HASH_HashType hash_function,
base::StringPiece passphrase,
base::StringPiece salt,
- uint32 count,
+ unsigned count,
uint8 *out_key) {
const std::string combined = salt.as_string() + passphrase.as_string();
const size_t combined_len = combined.size();
- uint32 done = 0;
+ unsigned done = 0;
uint8 zero[1] = {0};
- EVP_MD_CTX ctx;
- EVP_MD_CTX_init(&ctx);
+ HASHContext* hash_context = HASH_Create(hash_function);
- for (uint32 i = 0; done < cipher_key_length; i++) {
- CHECK_EQ(EVP_DigestInit_ex(&ctx, hash_function, NULL), 1);
+ for (unsigned i = 0; done < cipher_key_length; i++) {
+ HASH_Begin(hash_context);
- for (uint32 j = 0; j < i; j++)
- EVP_DigestUpdate(&ctx, zero, sizeof(zero));
+ for (unsigned j = 0; j < i; j++)
+ HASH_Update(hash_context, zero, sizeof(zero));
- uint32 written = 0;
+ unsigned written = 0;
while (written < count) {
if (written + combined_len > count) {
- uint32 todo = count - written;
- EVP_DigestUpdate(&ctx, combined.data(), todo);
+ unsigned todo = count - written;
+ HASH_Update(hash_context,
+ reinterpret_cast<const uint8*>(combined.data()),
+ todo);
written = count;
} else {
- EVP_DigestUpdate(&ctx, combined.data(), combined_len);
+ HASH_Update(hash_context,
+ reinterpret_cast<const uint8*>(combined.data()),
+ combined_len);
written += combined_len;
}
}
- uint32 num_hash_bytes;
- uint8 hash[EVP_MAX_MD_SIZE];
- CHECK_EQ(EVP_DigestFinal_ex(&ctx, hash, &num_hash_bytes), 1);
+ unsigned num_hash_bytes;
+ uint8 digest[HASH_LENGTH_MAX];
+ HASH_End(hash_context, digest, &num_hash_bytes, sizeof(digest));
- uint32 todo = cipher_key_length - done;
+ unsigned todo = cipher_key_length - done;
if (todo > num_hash_bytes)
todo = num_hash_bytes;
- memcpy(out_key + done, hash, todo);
+ memcpy(out_key + done, digest, todo);
done += todo;
}
- EVP_MD_CTX_cleanup(&ctx);
+ HASH_Destroy(hash_context);
+}
+
+// CreateAESContext sets up |out_key| to be an AES context, with the given key,
+// in ECB mode and with no IV.
+bool CreateAESContext(const uint8* key, unsigned key_len,
+ ScopedPK11Context* out_decryption_context) {
+ ScopedPK11Slot slot(PK11_GetBestSlot(CKM_AES_ECB, NULL));
+ if (!slot.get())
+ return false;
+ SECItem key_item;
+ key_item.type = siBuffer;
+ key_item.data = const_cast<uint8*>(key);
+ key_item.len = key_len;
+ ScopedPK11SymKey pk11_key(PK11_ImportSymKey(
+ slot.get(), CKM_AES_ECB, PK11_OriginUnwrap, CKA_ENCRYPT, &key_item,
+ NULL));
+ if (!pk11_key.get())
+ return false;
+ ScopedSECItem iv_param(PK11_ParamFromIV(CKM_AES_ECB, NULL));
+ out_decryption_context->reset(
+ PK11_CreateContextBySymKey(CKM_AES_ECB, CKA_ENCRYPT, pk11_key.get(),
+ iv_param.get()));
+ return out_decryption_context->get() != NULL;
}
+
// These constants are the tag numbers for the various packet types that we
// use.
-static const uint32 kSymmetricKeyEncryptedTag = 3;
-static const uint32 kSymmetricallyEncryptedTag = 18;
-static const uint32 kCompressedTag = 8;
-static const uint32 kLiteralDataTag = 11;
+static const unsigned kSymmetricKeyEncryptedTag = 3;
+static const unsigned kSymmetricallyEncryptedTag = 18;
+static const unsigned kCompressedTag = 8;
+static const unsigned kLiteralDataTag = 11;
class Decrypter {
public:
@@ -162,9 +191,9 @@ class Decrypter {
base::StringPiece passphrase,
base::StringPiece *out_contents) {
Reader reader(in);
- uint32 tag;
+ unsigned tag;
base::StringPiece contents;
- AES_KEY key;
+ ScopedPK11Context decryption_context;
if (!ParsePacket(&reader, &tag, &contents))
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
@@ -172,7 +201,7 @@ class Decrypter {
return OpenPGPSymmetricEncrytion::NOT_SYMMETRICALLY_ENCRYPTED;
Reader inner(contents);
OpenPGPSymmetricEncrytion::Result result =
- ParseSymmetricKeyEncrypted(&inner, passphrase, &key);
+ ParseSymmetricKeyEncrypted(&inner, passphrase, &decryption_context);
if (result != OpenPGPSymmetricEncrytion::OK)
return result;
@@ -183,7 +212,7 @@ class Decrypter {
if (!reader.empty())
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
inner = Reader(contents);
- if (!ParseSymmetricallyEncrypted(&inner, &key, &contents))
+ if (!ParseSymmetricallyEncrypted(&inner, &decryption_context, &contents))
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
reader = Reader(contents);
@@ -204,7 +233,7 @@ class Decrypter {
// ParsePacket parses an OpenPGP packet from reader. See RFC 4880, section
// 4.2.2.
bool ParsePacket(Reader *reader,
- uint32 *out_tag,
+ unsigned *out_tag,
base::StringPiece *out_contents) {
uint8 header;
if (!reader->U8(&header))
@@ -224,9 +253,9 @@ class Decrypter {
return true;
}
- const uint32 length_bytes = 1 << length_type;
- uint32 length = 0;
- for (uint32 i = 0; i < length_bytes; i++) {
+ const unsigned length_bytes = 1 << length_type;
+ size_t length = 0;
+ for (unsigned i = 0; i < length_bytes; i++) {
uint8 length_byte;
if (!reader->U8(&length_byte))
return false;
@@ -239,7 +268,7 @@ class Decrypter {
// New format packet.
*out_tag = header & 0x3f;
- uint32 length;
+ size_t length;
bool is_partial;
if (!ParseLength(reader, &length, &is_partial))
return false;
@@ -251,17 +280,17 @@ class Decrypter {
// ParseStreamContents parses all the chunks of a partial length stream from
// reader. See http://tools.ietf.org/html/rfc4880#section-4.2.2.4
bool ParseStreamContents(Reader *reader,
- uint32 length,
+ size_t length,
base::StringPiece *out_contents) {
const Reader::Position beginning_of_stream = reader->tell();
- const uint32 first_chunk_length = length;
+ const size_t first_chunk_length = length;
// First we parse the stream to find its length.
if (!reader->Skip(length))
return false;
for (;;) {
- uint32 chunk_length;
+ size_t chunk_length;
bool is_partial;
if (!ParseLength(reader, &chunk_length, &is_partial))
@@ -278,7 +307,7 @@ class Decrypter {
// Now we have the length of the whole stream in |length|.
char* buf = reinterpret_cast<char*>(malloc(length));
arena_.push_back(buf);
- uint32 j = 0;
+ size_t j = 0;
reader->Seek(beginning_of_stream);
base::StringPiece first_chunk;
@@ -289,7 +318,7 @@ class Decrypter {
// Now we parse the stream again, this time copying into |buf|
for (;;) {
- uint32 chunk_length;
+ size_t chunk_length;
bool is_partial;
if (!ParseLength(reader, &chunk_length, &is_partial))
@@ -309,7 +338,7 @@ class Decrypter {
// ParseLength parses an OpenPGP length from reader. See RFC 4880, section
// 4.2.2.
- bool ParseLength(Reader *reader, uint32 *out_length, bool *out_is_prefix) {
+ bool ParseLength(Reader *reader, size_t *out_length, bool *out_is_prefix) {
uint8 length_spec;
if (!reader->U8(&length_spec))
return false;
@@ -331,7 +360,11 @@ class Decrypter {
*out_is_prefix = true;
return true;
} else {
- return reader->U32(out_length);
+ uint32 length32;
+ if (!reader->U32(&length32))
+ return false;
+ *out_length = length32;
+ return true;
}
}
@@ -340,7 +373,7 @@ class Decrypter {
OpenPGPSymmetricEncrytion::Result ParseSymmetricKeyEncrypted(
Reader *reader,
base::StringPiece passphrase,
- AES_KEY *out_key) {
+ ScopedPK11Context *decryption_context) {
uint8 version, cipher, s2k_type, hash_func_id;
if (!reader->U8(&version) || version != 4)
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
@@ -355,18 +388,19 @@ class Decrypter {
if (cipher_key_length == 0)
return OpenPGPSymmetricEncrytion::UNKNOWN_CIPHER;
- const EVP_MD *hash_function;
+ HASH_HashType hash_function;
switch (hash_func_id) {
case 2: // SHA-1
- hash_function = EVP_sha1();
+ hash_function = HASH_AlgSHA1;
break;
case 8: // SHA-256
- hash_function = EVP_sha256();
+ hash_function = HASH_AlgSHA256;
break;
default:
return OpenPGPSymmetricEncrytion::UNKNOWN_HASH;
}
+ // This chunk of code parses the S2K specifier. See RFC 4880, section 3.7.1.
base::StringPiece salt;
uint8 key[32];
uint8 count_spec;
@@ -374,6 +408,7 @@ class Decrypter {
case 1:
if (!reader->Prefix(8, &salt))
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
+ // Fall through.
case 0:
SaltedIteratedS2K(cipher_key_length, hash_function, passphrase, salt,
passphrase.size() + salt.size(), key);
@@ -385,14 +420,14 @@ class Decrypter {
}
SaltedIteratedS2K(
cipher_key_length, hash_function, passphrase, salt,
- static_cast<uint32>(
+ static_cast<unsigned>(
16 + (count_spec&15)) << ((count_spec >> 4) + 6), key);
break;
default:
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
}
- if (AES_set_encrypt_key(key, 8 * cipher_key_length, out_key))
+ if (!CreateAESContext(key, cipher_key_length, decryption_context))
return OpenPGPSymmetricEncrytion::INTERNAL_ERROR;
if (reader->empty()) {
@@ -409,30 +444,50 @@ class Decrypter {
malloc(encrypted_key.size()));
arena_.push_back(plaintext_key);
- int num = 0;
- uint8 iv[16] = {0};
-
- AES_cfb128_encrypt(reinterpret_cast<const uint8*>(encrypted_key.data()),
- plaintext_key,
- encrypted_key.size(),
- out_key,
- iv,
- &num,
- AES_DECRYPT);
+ CFBDecrypt(encrypted_key, decryption_context, plaintext_key);
cipher_key_length = OpenPGPCipherIdToKeyLength(plaintext_key[0]);
if (cipher_key_length == 0)
return OpenPGPSymmetricEncrytion::UNKNOWN_CIPHER;
if (encrypted_key.size() != 1u + cipher_key_length)
return OpenPGPSymmetricEncrytion::PARSE_ERROR;
- if (AES_set_encrypt_key(plaintext_key + 1, 8 * cipher_key_length,
- out_key)) {
+ if (!CreateAESContext(plaintext_key + 1, cipher_key_length,
+ decryption_context)) {
return OpenPGPSymmetricEncrytion::INTERNAL_ERROR;
}
return OpenPGPSymmetricEncrytion::OK;
}
- uint32 OpenPGPCipherIdToKeyLength(uint8 cipher) {
+ // CFBDecrypt decrypts the cipher-feedback encrypted data in |in| to |out|
+ // using |decryption_context| and assumes an IV of all zeros.
+ void CFBDecrypt(base::StringPiece in, ScopedPK11Context* decryption_context,
+ uint8* out) {
+ // We need this for PK11_CipherOp to write to, but we never check it as we
+ // work in ECB mode, one block at a time.
+ int out_len;
+
+ uint8 mask[AES_BLOCK_SIZE];
+ memset(mask, 0, sizeof(mask));
+
+ unsigned used = AES_BLOCK_SIZE;
+
+ for (size_t i = 0; i < in.size(); i++) {
+ if (used == AES_BLOCK_SIZE) {
+ PK11_CipherOp(decryption_context->get(), mask, &out_len, sizeof(mask),
+ mask, AES_BLOCK_SIZE);
+ used = 0;
+ }
+
+ uint8 t = in[i];
+ out[i] = t ^ mask[used];
+ mask[used] = t;
+ used++;
+ }
+ }
+
+ // OpenPGPCipherIdToKeyLength converts an OpenPGP cipher id (see RFC 4880,
+ // section 9.2) to the key length of that cipher. It returns 0 on error.
+ unsigned OpenPGPCipherIdToKeyLength(uint8 cipher) {
switch (cipher) {
case 7: // AES-128
return 16;
@@ -448,8 +503,12 @@ class Decrypter {
// ParseSymmetricallyEncrypted parses a Symmetrically Encrypted packet. See
// RFC 4880, sections 5.7 and 5.13.
bool ParseSymmetricallyEncrypted(Reader *reader,
- AES_KEY *key,
+ ScopedPK11Context *decryption_context,
base::StringPiece *out_plaintext) {
+ // We need this for PK11_CipherOp to write to, but we never check it as we
+ // work in ECB mode, one block at a time.
+ int out_len;
+
uint8 version;
if (!reader->U8(&version) || version != 1)
return false;
@@ -464,10 +523,12 @@ class Decrypter {
uint8 fre[AES_BLOCK_SIZE];
memset(prefix_copy, 0, AES_BLOCK_SIZE);
- AES_ecb_encrypt(prefix_copy, fre, key, AES_ENCRYPT);
- for (uint32 i = 0; i < AES_BLOCK_SIZE; i++)
+ PK11_CipherOp(decryption_context->get(), fre, &out_len, sizeof(fre),
+ prefix_copy, AES_BLOCK_SIZE);
+ for (unsigned i = 0; i < AES_BLOCK_SIZE; i++)
prefix_copy[i] = fre[i] ^ prefix[i];
- AES_ecb_encrypt(prefix, fre, key, AES_ENCRYPT);
+ PK11_CipherOp(decryption_context->get(), fre, &out_len, sizeof(fre), prefix,
+ AES_BLOCK_SIZE);
prefix_copy[AES_BLOCK_SIZE] = prefix[AES_BLOCK_SIZE] ^ fre[0];
prefix_copy[AES_BLOCK_SIZE + 1] = prefix[AES_BLOCK_SIZE + 1] ^ fre[1];
@@ -479,10 +540,10 @@ class Decrypter {
fre[0] = prefix[AES_BLOCK_SIZE];
fre[1] = prefix[AES_BLOCK_SIZE + 1];
- uint32 out_used = 2;
+ unsigned out_used = 2;
- const uint32 plaintext_size = reader->size();
- if (plaintext_size < SHA_DIGEST_LENGTH + 2) {
+ const size_t plaintext_size = reader->size();
+ if (plaintext_size < SHA1_LENGTH + 2) {
// Too small to contain an MDC trailer.
return false;
}
@@ -490,12 +551,13 @@ class Decrypter {
uint8* plaintext = reinterpret_cast<uint8*>(malloc(plaintext_size));
arena_.push_back(plaintext);
- for (uint32 i = 0; i < plaintext_size; i++) {
+ for (size_t i = 0; i < plaintext_size; i++) {
uint8 b;
if (!reader->U8(&b))
return false;
if (out_used == AES_BLOCK_SIZE) {
- AES_ecb_encrypt(fre, fre, key, AES_ENCRYPT);
+ PK11_CipherOp(decryption_context->get(), fre, &out_len, sizeof(fre),
+ fre, AES_BLOCK_SIZE);
out_used = 0;
}
@@ -506,25 +568,27 @@ class Decrypter {
// The plaintext should be followed by a Modification Detection Code
// packet. This packet is specified such that the header is always
// serialized as exactly these two bytes:
- if (plaintext[plaintext_size - SHA_DIGEST_LENGTH - 2] != 0xd3 ||
- plaintext[plaintext_size - SHA_DIGEST_LENGTH - 1] != 0x14) {
+ if (plaintext[plaintext_size - SHA1_LENGTH - 2] != 0xd3 ||
+ plaintext[plaintext_size - SHA1_LENGTH - 1] != 0x14) {
return false;
}
- SHA_CTX sha1;
- SHA1_Init(&sha1);
- SHA1_Update(&sha1, prefix_copy, sizeof(prefix_copy));
- SHA1_Update(&sha1, plaintext, plaintext_size - SHA_DIGEST_LENGTH);
- uint8 digest[SHA_DIGEST_LENGTH];
- SHA1_Final(digest, &sha1);
-
- if (memcmp(digest, &plaintext[plaintext_size - SHA_DIGEST_LENGTH],
- SHA_DIGEST_LENGTH) != 0) {
+ HASHContext* hash_context = HASH_Create(HASH_AlgSHA1);
+ HASH_Begin(hash_context);
+ HASH_Update(hash_context, prefix_copy, sizeof(prefix_copy));
+ HASH_Update(hash_context, plaintext, plaintext_size - SHA1_LENGTH);
+ uint8 digest[SHA1_LENGTH];
+ unsigned num_hash_bytes;
+ HASH_End(hash_context, digest, &num_hash_bytes, sizeof(digest));
+ HASH_Destroy(hash_context);
+
+ if (memcmp(digest, &plaintext[plaintext_size - SHA1_LENGTH],
+ SHA1_LENGTH) != 0) {
return false;
}
*out_plaintext = base::StringPiece(reinterpret_cast<char*>(plaintext),
- plaintext_size - SHA_DIGEST_LENGTH);
+ plaintext_size - SHA1_LENGTH);
return true;
}
@@ -564,7 +628,9 @@ class Encrypter {
}
private:
- static ByteString MakePacket(uint32 tag, const ByteString& contents) {
+ // MakePacket returns an OpenPGP packet tagged as type |tag|. It always uses
+ // new-format headers. See RFC 4880, section 4.2.
+ static ByteString MakePacket(unsigned tag, const ByteString& contents) {
ByteString header;
header.push_back(0x80 | 0x40 | tag);
@@ -587,6 +653,9 @@ class Encrypter {
return header + contents;
}
+ // SerializeLiteralData returns a Literal Data packet containing |contents|
+ // as binary data with no filename nor mtime specified. See RFC 4880, section
+ // 5.9.
static ByteString SerializeLiteralData(base::StringPiece contents) {
ByteString literal_data;
literal_data.push_back(0x74); // text mode
@@ -600,6 +669,9 @@ class Encrypter {
return MakePacket(kLiteralDataTag, literal_data);
}
+ // SerializeSymmetricKeyEncrypted generates a random AES-128 key from
+ // |passphrase|, sets |out_key| to it and returns a Symmetric Key Encrypted
+ // packet. See RFC 4880, section 5.3.
static ByteString SerializeSymmetricKeyEncrypted(base::StringPiece passphrase,
ByteString *out_key) {
ByteString ske;
@@ -617,30 +689,39 @@ class Encrypter {
uint8 key[16];
SaltedIteratedS2K(
- sizeof(key), EVP_sha1(), passphrase,
+ sizeof(key), HASH_AlgSHA1, passphrase,
base::StringPiece(reinterpret_cast<char*>(&salt64), sizeof(salt64)),
65536, key);
*out_key = ByteString(key, sizeof(key));
return MakePacket(kSymmetricKeyEncryptedTag, ske);
}
+ // SerializeSymmetricallyEncrypted encrypts |plaintext| with |key| and
+ // returns a Symmetrically Encrypted packet containing the ciphertext. See
+ // RFC 4880, section 5.7.
static ByteString SerializeSymmetricallyEncrypted(ByteString plaintext,
const ByteString& key) {
+ // We need this for PK11_CipherOp to write to, but we never check it as we
+ // work in ECB mode, one block at a time.
+ int out_len;
+
ByteString packet;
packet.push_back(1); // version 1
- static const uint32 kBlockSize = 16; // AES block size
+ static const unsigned kBlockSize = 16; // AES block size
uint8 prefix[kBlockSize + 2], fre[kBlockSize], iv[kBlockSize];
base::RandBytes(iv, kBlockSize);
memset(fre, 0, sizeof(fre));
- AES_KEY aes_key;
- AES_set_encrypt_key(key.data(), 8 * key.size(), &aes_key);
+ ScopedPK11Context aes_context;
+ CHECK(CreateAESContext(key.data(), key.size(), &aes_context));
- AES_ecb_encrypt(fre, fre, &aes_key, AES_ENCRYPT);
- for (uint32 i = 0; i < 16; i++)
+ PK11_CipherOp(aes_context.get(), fre, &out_len, sizeof(fre), fre,
+ AES_BLOCK_SIZE);
+ for (unsigned i = 0; i < 16; i++)
prefix[i] = iv[i] ^ fre[i];
- AES_ecb_encrypt(prefix, fre, &aes_key, AES_ENCRYPT);
+ PK11_CipherOp(aes_context.get(), fre, &out_len, sizeof(fre), prefix,
+ AES_BLOCK_SIZE);
prefix[kBlockSize] = iv[kBlockSize - 2] ^ fre[0];
prefix[kBlockSize + 1] = iv[kBlockSize - 1] ^ fre[1];
@@ -650,23 +731,26 @@ class Encrypter {
plaintext_copy.push_back(0xd3); // MDC packet
plaintext_copy.push_back(20); // packet length (20 bytes)
- SHA_CTX sha1;
- SHA1_Init(&sha1);
- SHA1_Update(&sha1, iv, sizeof(iv));
- SHA1_Update(&sha1, iv + kBlockSize - 2, 2);
- SHA1_Update(&sha1, plaintext_copy.data(), plaintext_copy.size());
- uint8 digest[SHA_DIGEST_LENGTH];
- SHA1_Final(digest, &sha1);
+ HASHContext* hash_context = HASH_Create(HASH_AlgSHA1);
+ HASH_Begin(hash_context);
+ HASH_Update(hash_context, iv, sizeof(iv));
+ HASH_Update(hash_context, iv + kBlockSize - 2, 2);
+ HASH_Update(hash_context, plaintext_copy.data(), plaintext_copy.size());
+ uint8 digest[SHA1_LENGTH];
+ unsigned num_hash_bytes;
+ HASH_End(hash_context, digest, &num_hash_bytes, sizeof(digest));
+ HASH_Destroy(hash_context);
plaintext_copy += ByteString(digest, sizeof(digest));
fre[0] = prefix[kBlockSize];
fre[1] = prefix[kBlockSize+1];
- uint32 out_used = 2;
+ unsigned out_used = 2;
for (size_t i = 0; i < plaintext_copy.size(); i++) {
if (out_used == kBlockSize) {
- AES_ecb_encrypt(fre, fre, &aes_key, AES_ENCRYPT);
+ PK11_CipherOp(aes_context.get(), fre, &out_len, sizeof(fre), fre,
+ AES_BLOCK_SIZE);
out_used = 0;
}
@@ -686,8 +770,9 @@ OpenPGPSymmetricEncrytion::Result OpenPGPSymmetricEncrytion::Decrypt(
base::StringPiece encrypted,
base::StringPiece passphrase,
std::string *out) {
- Decrypter decrypter;
+ EnsureNSSInit();
+ Decrypter decrypter;
base::StringPiece result;
Result reader = decrypter.Decrypt(encrypted, passphrase, &result);
if (reader == OK)
@@ -699,6 +784,8 @@ OpenPGPSymmetricEncrytion::Result OpenPGPSymmetricEncrytion::Decrypt(
std::string OpenPGPSymmetricEncrytion::Encrypt(
base::StringPiece plaintext,
base::StringPiece passphrase) {
+ EnsureNSSInit();
+
Encrypter::ByteString b =
Encrypter::Encrypt(plaintext, passphrase);
return std::string(reinterpret_cast<const char*>(b.data()), b.size());
diff --git a/crypto/openpgp_symmetric_encryption_test_openssl.cc b/crypto/openpgp_symmetric_encryption_unittest.cc
index 6185719..6185719 100644
--- a/crypto/openpgp_symmetric_encryption_test_openssl.cc
+++ b/crypto/openpgp_symmetric_encryption_unittest.cc