| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576 |
- //
- // CryptoSwift
- //
- // Copyright (C) 2014-2017 Marcin Krzyżanowski <marcin@krzyzanowskim.com>
- // This software is provided 'as-is', without any express or implied warranty.
- //
- // In no event will the authors be held liable for any damages arising from the use of this software.
- //
- // Permission is granted to anyone to use this software for any purpose,including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions:
- //
- // - The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation is required.
- // - Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
- // - This notice may not be removed or altered from any source or binary distribution.
- //
- // Cipher-block chaining (CBC)
- //
- public struct CBC: BlockMode {
- public enum Error: Swift.Error {
- /// Invalid IV
- case invalidInitializationVector
- }
- public let options: BlockModeOption = [.initializationVectorRequired, .paddingRequired]
-
- private let iv: Array<UInt8>
- public let customBlockSize: Int? = nil
- public init(iv: Array<UInt8>) {
- self.iv = iv
- }
- public func worker(blockSize: Int, cipherOperation: @escaping CipherOperationOnBlock, encryptionOperation: @escaping CipherOperationOnBlock) throws -> CipherModeWorker {
- if self.iv.count != blockSize {
- throw Error.invalidInitializationVector
- }
- return CBCModeWorker(blockSize: blockSize, iv: self.iv.slice, cipherOperation: cipherOperation)
- }
- }
- struct CBCModeWorker: BlockModeWorker {
- let cipherOperation: CipherOperationOnBlock
- var blockSize: Int
- let additionalBufferSize: Int = 0
- private let iv: ArraySlice<UInt8>
- private var prev: ArraySlice<UInt8>?
- @inlinable
- init(blockSize: Int, iv: ArraySlice<UInt8>, cipherOperation: @escaping CipherOperationOnBlock) {
- self.blockSize = blockSize
- self.iv = iv
- self.cipherOperation = cipherOperation
- }
- @inlinable
- mutating func encrypt(block plaintext: ArraySlice<UInt8>) -> Array<UInt8> {
- guard let ciphertext = cipherOperation(xor(prev ?? iv, plaintext)) else {
- return Array(plaintext)
- }
- self.prev = ciphertext.slice
- return ciphertext
- }
- @inlinable
- mutating func decrypt(block ciphertext: ArraySlice<UInt8>) -> Array<UInt8> {
- guard let plaintext = cipherOperation(ciphertext) else {
- return Array(ciphertext)
- }
- let result: Array<UInt8> = xor(prev ?? self.iv, plaintext)
- self.prev = ciphertext
- return result
- }
- }
|