Crypto++ 8.9
Free C++ class library of cryptographic schemes
strciphr.cpp
1// strciphr.cpp - originally written and placed in the public domain by Wei Dai.
2
3#include "pch.h"
4
5#ifndef CRYPTOPP_IMPORTS
6
7#include "strciphr.h"
8
9// Squash MS LNK4221 and libtool warnings
10#ifndef CRYPTOPP_MANUALLY_INSTANTIATE_TEMPLATES
11extern const char STRCIPHER_FNAME[] = __FILE__;
12#endif
13
14NAMESPACE_BEGIN(CryptoPP)
15
16template <class S>
17void AdditiveCipherTemplate<S>::UncheckedSetKey(const byte *key, unsigned int length, const NameValuePairs &params)
18{
19 PolicyInterface &policy = this->AccessPolicy();
20 policy.CipherSetKey(params, key, length);
21 m_leftOver = 0;
22 unsigned int bufferByteSize = policy.CanOperateKeystream() ? GetBufferByteSize(policy) : RoundUpToMultipleOf(1024U, GetBufferByteSize(policy));
23 m_buffer.New(bufferByteSize);
24
25 if (this->IsResynchronizable())
26 {
27 size_t ivLength;
28 const byte *iv = this->GetIVAndThrowIfInvalid(params, ivLength);
29 policy.CipherResynchronize(m_buffer, iv, ivLength);
30 }
31}
32
33template <class S>
34void AdditiveCipherTemplate<S>::GenerateBlock(byte *outString, size_t length)
35{
36 if (m_leftOver > 0)
37 {
38 const size_t len = STDMIN(m_leftOver, length);
39 std::memcpy(outString, PtrSub(KeystreamBufferEnd(), m_leftOver), len);
40
41 length -= len; m_leftOver -= len;
42 outString = PtrAdd(outString, len);
43 if (!length) {return;}
44 }
45
46 PolicyInterface &policy = this->AccessPolicy();
47 size_t bytesPerIteration = policy.GetBytesPerIteration();
48
49 if (length >= bytesPerIteration)
50 {
51 const size_t iterations = length / bytesPerIteration;
52 policy.WriteKeystream(outString, iterations);
53 length -= iterations * bytesPerIteration;
54 outString = PtrAdd(outString, iterations * bytesPerIteration);
55 }
56
57 if (length > 0)
58 {
59 size_t bufferByteSize = RoundUpToMultipleOf(length, bytesPerIteration);
60 size_t bufferIterations = bufferByteSize / bytesPerIteration;
61
62 policy.WriteKeystream(PtrSub(KeystreamBufferEnd(), bufferByteSize), bufferIterations);
63 std::memcpy(outString, PtrSub(KeystreamBufferEnd(), bufferByteSize), length);
64 m_leftOver = bufferByteSize - length;
65 }
66}
67
68template <class S>
69void AdditiveCipherTemplate<S>::ProcessData(byte *outString, const byte *inString, size_t length)
70{
71 CRYPTOPP_ASSERT(outString); CRYPTOPP_ASSERT(inString);
72 CRYPTOPP_ASSERT(length % this->MandatoryBlockSize() == 0);
73
74 PolicyInterface &policy = this->AccessPolicy();
75 size_t bytesPerIteration = policy.GetBytesPerIteration();
76
77 if (m_leftOver > 0)
78 {
79 const size_t len = STDMIN(m_leftOver, length);
80 xorbuf(outString, inString, PtrSub(KeystreamBufferEnd(), m_leftOver), len);
81
82 inString = PtrAdd(inString, len);
83 outString = PtrAdd(outString, len);
84 length -= len; m_leftOver -= len;
85 }
86
87 if (!length) { return; }
88
89 const word32 alignment = policy.GetAlignment();
90 const bool inAligned = IsAlignedOn(inString, alignment);
91 const bool outAligned = IsAlignedOn(outString, alignment);
92 CRYPTOPP_UNUSED(inAligned); CRYPTOPP_UNUSED(outAligned);
93
94 if (policy.CanOperateKeystream() && length >= bytesPerIteration)
95 {
96 const size_t iterations = length / bytesPerIteration;
98 (inAligned ? EnumToInt(INPUT_ALIGNED) : 0) | (outAligned ? EnumToInt(OUTPUT_ALIGNED) : 0));
99 KeystreamOperation operation = KeystreamOperation(flags);
100 policy.OperateKeystream(operation, outString, inString, iterations);
101
102 inString = PtrAdd(inString, iterations * bytesPerIteration);
103 outString = PtrAdd(outString, iterations * bytesPerIteration);
104 length -= iterations * bytesPerIteration;
105 }
106
107 size_t bufferByteSize = m_buffer.size();
108 size_t bufferIterations = bufferByteSize / bytesPerIteration;
109
110 while (length >= bufferByteSize)
111 {
112 policy.WriteKeystream(m_buffer, bufferIterations);
113 xorbuf(outString, inString, KeystreamBufferBegin(), bufferByteSize);
114
115 inString = PtrAdd(inString, bufferByteSize);
116 outString = PtrAdd(outString, bufferByteSize);
117 length -= bufferByteSize;
118 }
119
120 if (length > 0)
121 {
122 bufferByteSize = RoundUpToMultipleOf(length, bytesPerIteration);
123 bufferIterations = bufferByteSize / bytesPerIteration;
124
125 policy.WriteKeystream(PtrSub(KeystreamBufferEnd(), bufferByteSize), bufferIterations);
126 xorbuf(outString, inString, PtrSub(KeystreamBufferEnd(), bufferByteSize), length);
127
128 m_leftOver = bufferByteSize - length;
129 }
130}
131
132template <class S>
133void AdditiveCipherTemplate<S>::Resynchronize(const byte *iv, int length)
134{
135 PolicyInterface &policy = this->AccessPolicy();
136 m_leftOver = 0;
137 m_buffer.New(GetBufferByteSize(policy));
138 policy.CipherResynchronize(m_buffer, iv, this->ThrowIfInvalidIVLength(length));
139}
140
141template <class BASE>
143{
144 PolicyInterface &policy = this->AccessPolicy();
145 unsigned int bytesPerIteration = policy.GetBytesPerIteration();
146
147 policy.SeekToIteration(position / bytesPerIteration);
148 position %= bytesPerIteration;
149
150 if (position > 0)
151 {
152 policy.WriteKeystream(PtrSub(KeystreamBufferEnd(), bytesPerIteration), 1);
153 m_leftOver = bytesPerIteration - static_cast<unsigned int>(position);
154 }
155 else
156 m_leftOver = 0;
157}
158
159template <class BASE>
160void CFB_CipherTemplate<BASE>::UncheckedSetKey(const byte *key, unsigned int length, const NameValuePairs &params)
161{
162 PolicyInterface &policy = this->AccessPolicy();
163 policy.CipherSetKey(params, key, length);
164
165 if (this->IsResynchronizable())
166 {
167 size_t ivLength;
168 const byte *iv = this->GetIVAndThrowIfInvalid(params, ivLength);
169 policy.CipherResynchronize(iv, ivLength);
170 }
171
172 m_leftOver = policy.GetBytesPerIteration();
173}
174
175template <class BASE>
176void CFB_CipherTemplate<BASE>::Resynchronize(const byte *iv, int length)
177{
178 PolicyInterface &policy = this->AccessPolicy();
179 policy.CipherResynchronize(iv, this->ThrowIfInvalidIVLength(length));
180 m_leftOver = policy.GetBytesPerIteration();
181}
182
183template <class BASE>
184void CFB_CipherTemplate<BASE>::ProcessData(byte *outString, const byte *inString, size_t length)
185{
186 CRYPTOPP_ASSERT(outString); CRYPTOPP_ASSERT(inString);
187 CRYPTOPP_ASSERT(length % this->MandatoryBlockSize() == 0);
188
189 PolicyInterface &policy = this->AccessPolicy();
190 unsigned int bytesPerIteration = policy.GetBytesPerIteration();
191 byte *reg = policy.GetRegisterBegin();
192
193 if (m_leftOver)
194 {
195 const size_t len = STDMIN(m_leftOver, length);
196 CombineMessageAndShiftRegister(outString, PtrAdd(reg, bytesPerIteration - m_leftOver), inString, len);
197
198 inString = PtrAdd(inString, len);
199 outString = PtrAdd(outString, len);
200 m_leftOver -= len; length -= len;
201 }
202
203 if (!length) { return; }
204
205 const word32 alignment = policy.GetAlignment();
206 const bool inAligned = IsAlignedOn(inString, alignment);
207 const bool outAligned = IsAlignedOn(outString, alignment);
208 CRYPTOPP_UNUSED(inAligned); CRYPTOPP_UNUSED(outAligned);
209
210 if (policy.CanIterate() && length >= bytesPerIteration && outAligned)
211 {
212 CipherDir cipherDir = GetCipherDir(*this);
213 policy.Iterate(outString, inString, cipherDir, length / bytesPerIteration);
214
215 const size_t remainder = length % bytesPerIteration;
216 inString = PtrAdd(inString, length - remainder);
217 outString = PtrAdd(outString, length - remainder);
218 length = remainder;
219 }
220
221 while (length >= bytesPerIteration)
222 {
223 policy.TransformRegister();
224 CombineMessageAndShiftRegister(outString, reg, inString, bytesPerIteration);
225
226 inString = PtrAdd(inString, bytesPerIteration);
227 outString = PtrAdd(outString, bytesPerIteration);
228 length -= bytesPerIteration;
229 }
230
231 if (length > 0)
232 {
233 policy.TransformRegister();
234 CombineMessageAndShiftRegister(outString, reg, inString, length);
235 m_leftOver = bytesPerIteration - length;
236 }
237}
238
239template <class BASE>
240void CFB_EncryptionTemplate<BASE>::CombineMessageAndShiftRegister(byte *output, byte *reg, const byte *message, size_t length)
241{
242 xorbuf(reg, message, length);
243 std::memcpy(output, reg, length);
244}
245
246template <class BASE>
247void CFB_DecryptionTemplate<BASE>::CombineMessageAndShiftRegister(byte *output, byte *reg, const byte *message, size_t length)
248{
249 for (size_t i=0; i<length; i++)
250 {
251 byte b = message[i];
252 output[i] = reg[i] ^ b;
253 reg[i] = b;
254 }
255}
256
257NAMESPACE_END
258
259#endif
Base class for additive stream ciphers with SymmetricCipher interface.
Definition strciphr.h:298
void ProcessData(byte *outString, const byte *inString, size_t length)
Apply keystream to data.
void GenerateBlock(byte *output, size_t size)
Generate random array of bytes.
void Seek(lword position)
Seeks to a random position in the stream.
void Resynchronize(const byte *iv, int length=-1)
Resynchronize the cipher.
Base class for feedback based stream ciphers with SymmetricCipher interface.
Definition strciphr.h:568
void Resynchronize(const byte *iv, int length=-1)
Resynchronize the cipher.
void ProcessData(byte *outString, const byte *inString, size_t length)
Apply keystream to data.
Base class for feedback based stream ciphers in the reverse direction with SymmetricCipher interface.
Definition strciphr.h:664
Base class for feedback based stream ciphers in the forward direction with SymmetricCipher interface.
Definition strciphr.h:655
Interface for retrieving values given their names.
Definition cryptlib.h:327
unsigned int word32
32-bit unsigned datatype
Definition config_int.h:72
word64 lword
Large word type.
Definition config_int.h:168
CipherDir
Specifies a direction for a cipher to operate.
Definition cryptlib.h:128
T1 RoundUpToMultipleOf(const T1 &n, const T2 &m)
Rounds a value up to a multiple of a second value.
Definition misc.h:1384
PTR PtrSub(PTR pointer, OFF offset)
Create a pointer with an offset.
Definition misc.h:401
const T & STDMIN(const T &a, const T &b)
Replacement function for std::min.
Definition misc.h:657
bool IsAlignedOn(const void *ptr, unsigned int alignment)
Determines whether ptr is aligned to a minimum value.
Definition misc.h:1436
PTR PtrAdd(PTR pointer, OFF offset)
Create a pointer with an offset.
Definition misc.h:388
#define EnumToInt(v)
Integer value.
Definition misc.h:504
CipherDir GetCipherDir(const T &obj)
Returns the direction the cipher is being operated.
Definition misc.h:1497
CRYPTOPP_DLL void xorbuf(byte *buf, const byte *mask, size_t count)
Performs an XOR of a buffer with a mask.
Crypto++ library namespace.
Precompiled header file.
Classes for implementing stream ciphers.
KeystreamOperation
Keystream operation flags.
Definition strciphr.h:88
KeystreamOperationFlags
Keystream operation flags.
Definition strciphr.h:76
@ INPUT_ALIGNED
Input buffer is aligned.
Definition strciphr.h:80
@ OUTPUT_ALIGNED
Output buffer is aligned.
Definition strciphr.h:78
#define CRYPTOPP_ASSERT(exp)
Debugging and diagnostic assertion.
Definition trap.h:68