// Copyright 2009 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Cipher feedback (CFB) mode. // CFB provides confidentiality by feeding a fraction of // the previous ciphertext in as the plaintext for the next // block operation. // See NIST SP 800-38A, pp 11-13 package block import ( "io"; ) type cfbCipher struct { c Cipher; blockSize int; // our block size (s/8) cipherSize int; // underlying cipher block size iv []byte; tmp []byte; } func newCFB(c Cipher, s int, iv []byte) *cfbCipher { if s == 0 || s%8 != 0 { panicln("crypto/block: invalid CFB mode", s) } b := c.BlockSize(); x := new(cfbCipher); x.c = c; x.blockSize = s / 8; x.cipherSize = b; x.iv = copy(iv); x.tmp = make([]byte, b); return x; } func (x *cfbCipher) BlockSize() int { return x.blockSize } func (x *cfbCipher) Encrypt(src, dst []byte) { // Encrypt old IV and xor prefix with src to make dst. x.c.Encrypt(x.iv, x.tmp); for i := 0; i < x.blockSize; i++ { dst[i] = src[i] ^ x.tmp[i] } // Slide unused IV pieces down and insert dst at end. for i := 0; i < x.cipherSize-x.blockSize; i++ { x.iv[i] = x.iv[i+x.blockSize] } off := x.cipherSize - x.blockSize; for i := off; i < x.cipherSize; i++ { x.iv[i] = dst[i-off] } } func (x *cfbCipher) Decrypt(src, dst []byte) { // Encrypt [sic] old IV and xor prefix with src to make dst. x.c.Encrypt(x.iv, x.tmp); for i := 0; i < x.blockSize; i++ { dst[i] = src[i] ^ x.tmp[i] } // Slide unused IV pieces down and insert src at top. for i := 0; i < x.cipherSize-x.blockSize; i++ { x.iv[i] = x.iv[i+x.blockSize] } off := x.cipherSize - x.blockSize; for i := off; i < x.cipherSize; i++ { // Reconstruct src = dst ^ x.tmp // in case we overwrote src (src == dst). x.iv[i] = dst[i-off] ^ x.tmp[i-off] } } // NewCFBDecrypter returns a reader that reads data from r and decrypts it using c // in s-bit cipher feedback (CFB) mode with the initialization vector iv. // The returned Reader does not buffer or read ahead except // as required by the cipher's block size. // Modes for s not a multiple of 8 are unimplemented. func NewCFBDecrypter(c Cipher, s int, iv []byte, r io.Reader) io.Reader { return NewECBDecrypter(newCFB(c, s, iv), r) } // NewCFBEncrypter returns a writer that encrypts data using c // in s-bit cipher feedback (CFB) mode with the initialization vector iv // and writes the encrypted data to w. // The returned Writer does no buffering except as required // by the cipher's block size, so there is no need for a Flush method. // Modes for s not a multiple of 8 are unimplemented. func NewCFBEncrypter(c Cipher, s int, iv []byte, w io.Writer) io.Writer { return NewECBEncrypter(newCFB(c, s, iv), w) }