Skip to content

Commit

Permalink
Fix segfault in PKCS7 test (aws#2025)
Browse files Browse the repository at this point in the history
### Description of changes: 
Fix Segfault in PKCS7 test


By submitting this pull request, I confirm that my contribution is made
under the terms of the Apache 2.0 license and the ISC license.
  • Loading branch information
justsmth authored Dec 2, 2024
1 parent a47d28b commit beb8846
Showing 1 changed file with 16 additions and 18 deletions.
34 changes: 16 additions & 18 deletions crypto/pkcs7/pkcs7_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1699,7 +1699,7 @@ TEST(PKCS7Test, TestEnveloped) {
// NOTE: we make |buf| larger than |pt_len| in case padding gets added.
// without the extra room, we sometimes overflow into the next variable on the
// stack.
uint8_t buf[pt_len + EVP_MAX_BLOCK_LENGTH], decrypted[pt_len];
uint8_t buf[pt_len + EVP_MAX_BLOCK_LENGTH], decrypted[pt_len + EVP_MAX_BLOCK_LENGTH];

OPENSSL_cleanse(buf, sizeof(buf));
OPENSSL_memset(buf, 'A', pt_len);
Expand Down Expand Up @@ -1731,11 +1731,10 @@ TEST(PKCS7Test, TestEnveloped) {
bio.reset(BIO_new(BIO_s_mem()));
EXPECT_TRUE(PKCS7_decrypt(p7.get(), rsa_pkey.get(), rsa_x509.get(), bio.get(),
/*flags*/ 0));
EXPECT_EQ(sizeof(decrypted), BIO_pending(bio.get()));
EXPECT_EQ(pt_len, BIO_pending(bio.get()));
OPENSSL_cleanse(decrypted, sizeof(decrypted));
ASSERT_EQ((int)sizeof(decrypted),
BIO_read(bio.get(), decrypted, sizeof(decrypted)));
EXPECT_EQ(Bytes(buf, pt_len), Bytes(decrypted, sizeof(decrypted)));
ASSERT_EQ(pt_len, (size_t)BIO_read(bio.get(), decrypted, sizeof(decrypted)));
EXPECT_EQ(Bytes(buf, pt_len), Bytes(decrypted, pt_len));

// no certs provided for decryption
bio.reset(BIO_new_mem_buf(buf, pt_len));
Expand All @@ -1747,11 +1746,10 @@ TEST(PKCS7Test, TestEnveloped) {
EXPECT_TRUE(PKCS7_decrypt(p7.get(), rsa_pkey.get(), /*certs*/ nullptr,
bio.get(),
/*flags*/ 0));
EXPECT_EQ(sizeof(decrypted), BIO_pending(bio.get()));
EXPECT_EQ(pt_len, BIO_pending(bio.get()));
OPENSSL_cleanse(decrypted, sizeof(decrypted));
ASSERT_EQ((int)sizeof(decrypted),
BIO_read(bio.get(), decrypted, sizeof(decrypted)));
EXPECT_EQ(Bytes(buf, pt_len), Bytes(decrypted, sizeof(decrypted)));
ASSERT_EQ(pt_len, (size_t)BIO_read(bio.get(), decrypted, sizeof(decrypted)));
EXPECT_EQ(Bytes(buf, pt_len), Bytes(decrypted, pt_len));

// empty plaintext
bio.reset(BIO_new(BIO_s_mem()));
Expand Down Expand Up @@ -1814,7 +1812,7 @@ TEST(PKCS7Test, TestEnveloped) {
int decrypt_ok =
PKCS7_decrypt(p7.get(), rsa_pkey.get(), /*certs*/ nullptr, bio.get(),
/*flags*/ 0);
EXPECT_LE(sizeof(decrypted), BIO_pending(bio.get()));
EXPECT_LE(pt_len, BIO_pending(bio.get()));
OPENSSL_cleanse(decrypted, sizeof(decrypted));
// There's a fun edge case here for block ciphers using conventional PKCS#7
// padding. In this padding scheme, the last byte of the padded plaintext
Expand All @@ -1830,15 +1828,16 @@ TEST(PKCS7Test, TestEnveloped) {
// expectation. Ideally we'd find a way to access the padded plaintext and
// account for this deterministically by checking the random "padding" and
// adusting accordingly.
int max_decrypt =
sizeof(decrypted) + EVP_CIPHER_block_size(EVP_aes_128_cbc());
int decrypted_len = BIO_read(bio.get(), decrypted, max_decrypt);
if (decrypted_len > (int)pt_len) {
const size_t max_decrypt =
pt_len + EVP_CIPHER_block_size(EVP_aes_128_cbc());
const size_t decrypted_len = (size_t)BIO_read(bio.get(), decrypted, sizeof(decrypted));
ASSERT_LE(decrypted_len, sizeof(decrypted));
if (decrypted_len > pt_len) {
EXPECT_LT(max_decrypt - 4, decrypted_len);
EXPECT_TRUE(decrypt_ok);
EXPECT_FALSE(ERR_GET_REASON(ERR_peek_error()));
} else {
EXPECT_EQ((int)sizeof(decrypted), decrypted_len);
EXPECT_EQ(pt_len, decrypted_len);
EXPECT_FALSE(decrypt_ok);
EXPECT_EQ(CIPHER_R_BAD_DECRYPT, ERR_GET_REASON(ERR_peek_error()));
}
Expand All @@ -1858,10 +1857,9 @@ TEST(PKCS7Test, TestEnveloped) {
EXPECT_TRUE(PKCS7_decrypt(p7.get(), rsa_pkey.get(), /*certs*/ nullptr,
bio.get(),
/*flags*/ 0));
EXPECT_EQ(sizeof(decrypted), BIO_pending(bio.get()));
EXPECT_EQ(pt_len, BIO_pending(bio.get()));
OPENSSL_cleanse(decrypted, sizeof(decrypted));
ASSERT_EQ((int)sizeof(decrypted),
BIO_read(bio.get(), decrypted, sizeof(decrypted)));
ASSERT_EQ(pt_len, (size_t)BIO_read(bio.get(), decrypted, sizeof(decrypted)));
// ...but it produces pseudo-random nonsense
EXPECT_NE(Bytes(buf, pt_len), Bytes(decrypted, sizeof(decrypted)));
EXPECT_FALSE(ERR_GET_REASON(ERR_peek_error()));
Expand Down

0 comments on commit beb8846

Please sign in to comment.