@@ -71,3 +71,147 @@ void AES256Decrypt::Decrypt(unsigned char plaintext[16], const unsigned char cip
7171{
7272 AES256_decrypt (&ctx, 1 , plaintext, ciphertext);
7373}
74+
75+
76+ template <typename T>
77+ static int CBCEncrypt (const T& enc, const unsigned char iv[AES_BLOCKSIZE], const unsigned char * data, int size, bool pad, unsigned char * out)
78+ {
79+ int written = 0 ;
80+ int padsize = size % AES_BLOCKSIZE;
81+ unsigned char mixed[AES_BLOCKSIZE];
82+
83+ if (!data || !size || !out)
84+ return 0 ;
85+
86+ if (!pad && padsize != 0 )
87+ return 0 ;
88+
89+ memcpy (mixed, iv, AES_BLOCKSIZE);
90+
91+ // Write all but the last block
92+ while (written + AES_BLOCKSIZE <= size) {
93+ for (int i = 0 ; i != AES_BLOCKSIZE; i++)
94+ mixed[i] ^= *data++;
95+ enc.Encrypt (out + written, mixed);
96+ memcpy (mixed, out + written, AES_BLOCKSIZE);
97+ written += AES_BLOCKSIZE;
98+ }
99+ if (pad) {
100+ // For all that remains, pad each byte with the value of the remaining
101+ // space. If there is none, pad by a full block.
102+ for (int i = 0 ; i != padsize; i++)
103+ mixed[i] ^= *data++;
104+ for (int i = padsize; i != AES_BLOCKSIZE; i++)
105+ mixed[i] ^= AES_BLOCKSIZE - padsize;
106+ enc.Encrypt (out + written, mixed);
107+ written += AES_BLOCKSIZE;
108+ }
109+ return written;
110+ }
111+
112+ template <typename T>
113+ static int CBCDecrypt (const T& dec, const unsigned char iv[AES_BLOCKSIZE], const unsigned char * data, int size, bool pad, unsigned char * out)
114+ {
115+ unsigned char padsize = 0 ;
116+ int written = 0 ;
117+ bool fail = false ;
118+ const unsigned char * prev = iv;
119+
120+ if (!data || !size || !out)
121+ return 0 ;
122+
123+ if (size % AES_BLOCKSIZE != 0 )
124+ return 0 ;
125+
126+ // Decrypt all data. Padding will be checked in the output.
127+ while (written != size) {
128+ dec.Decrypt (out, data + written);
129+ for (int i = 0 ; i != AES_BLOCKSIZE; i++)
130+ *out++ ^= prev[i];
131+ prev = data + written;
132+ written += AES_BLOCKSIZE;
133+ }
134+
135+ // When decrypting padding, attempt to run in constant-time
136+ if (pad) {
137+ // If used, padding size is the value of the last decrypted byte. For
138+ // it to be valid, It must be between 1 and AES_BLOCKSIZE.
139+ padsize = *--out;
140+ fail = !padsize | (padsize > AES_BLOCKSIZE);
141+
142+ // If not well-formed, treat it as though there's no padding.
143+ padsize *= !fail;
144+
145+ // All padding must equal the last byte otherwise it's not well-formed
146+ for (int i = AES_BLOCKSIZE; i != 0 ; i--)
147+ fail |= ((i > AES_BLOCKSIZE - padsize) & (*out-- != padsize));
148+
149+ written -= padsize;
150+ }
151+ return written * !fail;
152+ }
153+
154+ AES256CBCEncrypt::AES256CBCEncrypt (const unsigned char key[AES256_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
155+ : enc(key), pad(padIn)
156+ {
157+ memcpy (iv, ivIn, AES_BLOCKSIZE);
158+ }
159+
160+ int AES256CBCEncrypt::Encrypt (const unsigned char * data, int size, unsigned char * out) const
161+ {
162+ return CBCEncrypt (enc, iv, data, size, pad, out);
163+ }
164+
165+ AES256CBCEncrypt::~AES256CBCEncrypt ()
166+ {
167+ memset (iv, 0 , sizeof (iv));
168+ }
169+
170+ AES256CBCDecrypt::AES256CBCDecrypt (const unsigned char key[AES256_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
171+ : dec(key), pad(padIn)
172+ {
173+ memcpy (iv, ivIn, AES_BLOCKSIZE);
174+ }
175+
176+
177+ int AES256CBCDecrypt::Decrypt (const unsigned char * data, int size, unsigned char * out) const
178+ {
179+ return CBCDecrypt (dec, iv, data, size, pad, out);
180+ }
181+
182+ AES256CBCDecrypt::~AES256CBCDecrypt ()
183+ {
184+ memset (iv, 0 , sizeof (iv));
185+ }
186+
187+ AES128CBCEncrypt::AES128CBCEncrypt (const unsigned char key[AES128_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
188+ : enc(key), pad(padIn)
189+ {
190+ memcpy (iv, ivIn, AES_BLOCKSIZE);
191+ }
192+
193+ AES128CBCEncrypt::~AES128CBCEncrypt ()
194+ {
195+ memset (iv, 0 , AES_BLOCKSIZE);
196+ }
197+
198+ int AES128CBCEncrypt::Encrypt (const unsigned char * data, int size, unsigned char * out) const
199+ {
200+ return CBCEncrypt (enc, iv, data, size, pad, out);
201+ }
202+
203+ AES128CBCDecrypt::AES128CBCDecrypt (const unsigned char key[AES128_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
204+ : dec(key), pad(padIn)
205+ {
206+ memcpy (iv, ivIn, AES_BLOCKSIZE);
207+ }
208+
209+ AES128CBCDecrypt::~AES128CBCDecrypt ()
210+ {
211+ memset (iv, 0 , AES_BLOCKSIZE);
212+ }
213+
214+ int AES128CBCDecrypt::Decrypt (const unsigned char * data, int size, unsigned char * out) const
215+ {
216+ return CBCDecrypt (dec, iv, data, size, pad, out);
217+ }
0 commit comments