mirror of
https://github.com/autc04/Retro68.git
synced 2024-11-24 23:32:06 +00:00
67 lines
1.3 KiB
Go
67 lines
1.3 KiB
Go
// Copyright 2011 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.
|
|
|
|
// OFB (Output Feedback) Mode.
|
|
|
|
package cipher
|
|
|
|
type ofb struct {
|
|
b Block
|
|
cipher []byte
|
|
out []byte
|
|
outUsed int
|
|
}
|
|
|
|
// NewOFB returns a Stream that encrypts or decrypts using the block cipher b
|
|
// in output feedback mode. The initialization vector iv's length must be equal
|
|
// to b's block size.
|
|
func NewOFB(b Block, iv []byte) Stream {
|
|
blockSize := b.BlockSize()
|
|
if len(iv) != blockSize {
|
|
return nil
|
|
}
|
|
bufSize := streamBufferSize
|
|
if bufSize < blockSize {
|
|
bufSize = blockSize
|
|
}
|
|
x := &ofb{
|
|
b: b,
|
|
cipher: make([]byte, blockSize),
|
|
out: make([]byte, 0, bufSize),
|
|
outUsed: 0,
|
|
}
|
|
|
|
copy(x.cipher, iv)
|
|
return x
|
|
}
|
|
|
|
func (x *ofb) refill() {
|
|
bs := x.b.BlockSize()
|
|
remain := len(x.out) - x.outUsed
|
|
if remain > x.outUsed {
|
|
return
|
|
}
|
|
copy(x.out, x.out[x.outUsed:])
|
|
x.out = x.out[:cap(x.out)]
|
|
for remain < len(x.out)-bs {
|
|
x.b.Encrypt(x.cipher, x.cipher)
|
|
copy(x.out[remain:], x.cipher)
|
|
remain += bs
|
|
}
|
|
x.out = x.out[:remain]
|
|
x.outUsed = 0
|
|
}
|
|
|
|
func (x *ofb) XORKeyStream(dst, src []byte) {
|
|
for len(src) > 0 {
|
|
if x.outUsed >= len(x.out)-x.b.BlockSize() {
|
|
x.refill()
|
|
}
|
|
n := xorBytes(dst, src, x.out[x.outUsed:])
|
|
dst = dst[n:]
|
|
src = src[n:]
|
|
x.outUsed += n
|
|
}
|
|
}
|