1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
|
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "crypto/aes_cbc.h"
#include "base/check.h"
#include "base/check_op.h"
#include "crypto/openssl_util.h"
#include "third_party/boringssl/src/include/openssl/aes.h"
#include "third_party/boringssl/src/include/openssl/evp.h"
namespace crypto::aes_cbc {
std::vector<uint8_t> Encrypt(base::span<const uint8_t> key,
base::span<const uint8_t, kBlockSize> iv,
base::span<const uint8_t> plaintext) {
const EVP_CIPHER* cipher =
key.size() == 32 ? EVP_aes_256_cbc() : EVP_aes_128_cbc();
CHECK_EQ(EVP_CIPHER_key_length(cipher), key.size());
OpenSSLErrStackTracer err_tracer(FROM_HERE);
bssl::ScopedEVP_CIPHER_CTX ctx;
CHECK(EVP_EncryptInit_ex(ctx.get(), cipher, nullptr, key.data(), iv.data()));
std::vector<uint8_t> ciphertext(plaintext.size() + kBlockSize);
int out_len;
CHECK(EVP_EncryptUpdate(ctx.get(), ciphertext.data(), &out_len,
plaintext.data(), plaintext.size()));
int tail_len;
CHECK(EVP_EncryptFinal_ex(ctx.get(),
// SAFETY: boringssl guarantees out_len is still
// inside ciphertext.
UNSAFE_BUFFERS(ciphertext.data() + out_len),
&tail_len));
ciphertext.resize(out_len + tail_len);
return ciphertext;
}
std::optional<std::vector<uint8_t>> Decrypt(
base::span<const uint8_t> key,
base::span<const uint8_t, kBlockSize> iv,
base::span<const uint8_t> ciphertext) {
const EVP_CIPHER* cipher =
key.size() == 32 ? EVP_aes_256_cbc() : EVP_aes_128_cbc();
OpenSSLErrStackTracer err_tracer(FROM_HERE);
bssl::ScopedEVP_CIPHER_CTX ctx;
CHECK(EVP_DecryptInit_ex(ctx.get(), cipher, nullptr, key.data(), iv.data()));
std::vector<uint8_t> plaintext(ciphertext.size());
int out_len;
CHECK(EVP_DecryptUpdate(ctx.get(), plaintext.data(), &out_len,
ciphertext.data(), ciphertext.size()));
int tail_len;
if (!EVP_DecryptFinal_ex(ctx.get(),
// SAFETY: boringssl guarantees out_len is still
// inside ciphertext.
UNSAFE_BUFFERS(plaintext.data() + out_len),
&tail_len)) {
return std::nullopt;
}
plaintext.resize(out_len + tail_len);
return plaintext;
}
} // namespace crypto::aes_cbc
|