// Copyright 2016 Google Inc. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd #include "packager/media/base/aes_pattern_cryptor.h" #include #include #include "packager/base/logging.h" namespace shaka { namespace media { AesPatternCryptor::AesPatternCryptor(uint8_t crypt_byte_block, uint8_t skip_byte_block, PatternEncryptionMode encryption_mode, ConstantIvFlag constant_iv_flag, std::unique_ptr cryptor) : AesCryptor(constant_iv_flag), crypt_byte_block_(crypt_byte_block), skip_byte_block_(skip_byte_block), encryption_mode_(encryption_mode), cryptor_(std::move(cryptor)) { // Treat pattern 0:0 as 1:0. if (crypt_byte_block_ == 0 && skip_byte_block_ == 0) crypt_byte_block_ = 1; DCHECK(cryptor_); DCHECK(!cryptor_->use_constant_iv()); } AesPatternCryptor::~AesPatternCryptor() {} bool AesPatternCryptor::InitializeWithIv(const std::vector& key, const std::vector& iv) { return SetIv(iv) && cryptor_->InitializeWithIv(key, iv); } bool AesPatternCryptor::CryptInternal(const uint8_t* text, size_t text_size, uint8_t* crypt_text, size_t* crypt_text_size) { // |crypt_text_size| is always the same as |text_size| for pattern encryption. if (*crypt_text_size < text_size) { LOG(ERROR) << "Expecting output size of at least " << text_size << " bytes."; return false; } *crypt_text_size = text_size; while (text_size > 0) { const size_t crypt_byte_size = crypt_byte_block_ * AES_BLOCK_SIZE; if (text_size <= crypt_byte_size) { const bool need_encrypt = encryption_mode_ != kSkipIfCryptByteBlockRemaining && text_size >= AES_BLOCK_SIZE; if (need_encrypt) { // The partial pattern SHALL be followed with the partial 16-byte block // remains unencrypted. const size_t aligned_crypt_byte_size = text_size / AES_BLOCK_SIZE * AES_BLOCK_SIZE; if (!cryptor_->Crypt(text, aligned_crypt_byte_size, crypt_text)) return false; text += aligned_crypt_byte_size; text_size -= aligned_crypt_byte_size; crypt_text += aligned_crypt_byte_size; } // The remaining bytes are not encrypted. memcpy(crypt_text, text, text_size); return true; } if (!cryptor_->Crypt(text, crypt_byte_size, crypt_text)) return false; text += crypt_byte_size; text_size -= crypt_byte_size; crypt_text += crypt_byte_size; const size_t skip_byte_size = std::min( static_cast(skip_byte_block_ * AES_BLOCK_SIZE), text_size); memcpy(crypt_text, text, skip_byte_size); text += skip_byte_size; text_size -= skip_byte_size; crypt_text += skip_byte_size; } return true; } void AesPatternCryptor::SetIvInternal() { CHECK(cryptor_->SetIv(iv())); } } // namespace media } // namespace shaka