/** * Copyright (C) 2019-present MongoDB, Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the Server Side Public License, version 1, * as published by MongoDB, Inc. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Server Side Public License for more details. * * You should have received a copy of the Server Side Public License * along with this program. If not, see * . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the Server Side Public License in all respects for * all of the code used other than as permitted herein. If you modify file(s) * with this exception, you may extend this exception to your version of the * file(s), but you are not obligated to do so. If you do not wish to do so, * delete this exception statement from your version. If you delete this * exception statement from all source files in the program, then also delete * it in the license file. */ #include "mongo/platform/basic.h" #include #include #include #include #include "mongo/base/init.h" #include "mongo/base/status.h" #include "mongo/crypto/symmetric_crypto.h" #include "mongo/crypto/symmetric_key.h" #include "mongo/platform/random.h" #include "mongo/util/assert_util.h" #include "mongo/util/str.h" namespace mongo { namespace crypto { namespace { template class SymmetricImplApple : public Parent { public: SymmetricImplApple(const SymmetricKey& key, aesMode mode, ConstDataRange iv) : _ctx(nullptr, CCCryptorRelease) { static_assert( std::is_same::value || std::is_same::value, "SymmetricImplApple must inherit from SymmetricEncryptor or SymmetricDecryptor"); CCMode ccMode; if (mode == aesMode::cbc) { ccMode = kCCModeCBC; uassert(ErrorCodes::BadValue, str::stream() << "Invalid ivlen for selected algorithm, expected " << aesCBCIVSize << ", got " << static_cast(iv.length()), iv.length() == aesCBCIVSize); } else if (mode == aesMode::ctr) { ccMode = kCCModeCTR; uassert(ErrorCodes::BadValue, str::stream() << "Invalid ivlen for selected algorithm, expected " << aesCTRIVSize << ", got " << static_cast(iv.length()), iv.length() == aesCTRIVSize); } else { uassert(ErrorCodes::UnsupportedFormat, "Native crypto on this platform only supports AES256-CBC or AES256-CTR", false); } CCCryptorRef context = nullptr; constexpr auto op = std::is_same::value ? kCCEncrypt : kCCDecrypt; constexpr void* tweak = nullptr; constexpr size_t tweakLength = 0; constexpr int numRounds = 0; constexpr CCModeOptions ccModeOptions = 0; const auto status = CCCryptorCreateWithMode(op, ccMode, kCCAlgorithmAES, kCCOptionPKCS7Padding, iv.data(), key.getKey(), key.getKeySize(), tweak, tweakLength, numRounds, ccModeOptions, &context); uassert(ErrorCodes::UnknownError, str::stream() << "CCCryptorCreate failure: " << status, status == kCCSuccess); _ctx.reset(context); } StatusWith update(ConstDataRange in, DataRange out) final { std::size_t outUsed = 0; const auto status = CCCryptorUpdate( _ctx.get(), in.data(), in.length(), out.data(), out.length(), &outUsed); if (status != kCCSuccess) { return Status(ErrorCodes::UnknownError, str::stream() << "Unable to perform CCCryptorUpdate: " << status); } return outUsed; } Status addAuthenticatedData(ConstDataRange authData) final { fassert(51128, authData.length() == 0); return Status::OK(); } StatusWith finalize(DataRange out) final { size_t outUsed = 0; const auto status = CCCryptorFinal(_ctx.get(), out.data(), out.length(), &outUsed); if (status != kCCSuccess) { return Status(ErrorCodes::UnknownError, str::stream() << "Unable to perform CCCryptorFinal: " << status); } return outUsed; } private: std::unique_ptr<_CCCryptor, decltype(&CCCryptorRelease)> _ctx; }; class SymmetricEncryptorApple : public SymmetricImplApple { public: using SymmetricImplApple::SymmetricImplApple; StatusWith finalizeTag(DataRange) final { // CBC only, no tag to create. return 0; } }; class SymmetricDecryptorApple : public SymmetricImplApple { public: using SymmetricImplApple::SymmetricImplApple; Status updateTag(ConstDataRange tag) final { // CBC only, no tag to verify. if (tag.length() > 0) { return {ErrorCodes::BadValue, "Unexpected tag for non-gcm cipher"}; } return Status::OK(); } }; } // namespace std::set getSupportedSymmetricAlgorithms() { return {aes256CBCName, aes256CTRName}; } Status engineRandBytes(DataRange buffer) { auto result = SecRandomCopyBytes(kSecRandomDefault, buffer.length(), buffer.data()); if (result != errSecSuccess) { return {ErrorCodes::UnknownError, str::stream() << "Failed generating random bytes: " << result}; } else { return Status::OK(); } } StatusWith> SymmetricEncryptor::create(const SymmetricKey& key, aesMode mode, ConstDataRange iv) try { std::unique_ptr encryptor = std::make_unique(key, mode, iv); return std::move(encryptor); } catch (const DBException& e) { return e.toStatus(); } StatusWith> SymmetricDecryptor::create(const SymmetricKey& key, aesMode mode, ConstDataRange iv) try { std::unique_ptr decryptor = std::make_unique(key, mode, iv); return std::move(decryptor); } catch (const DBException& e) { return e.toStatus(); } } // namespace crypto } // namespace mongo