freeswitch/libs/libks/crypt/twofish_cfb.c

99 lines
2.3 KiB
C
Executable File

#include <stdint.h>
#include <stdlib.h>
#include "twofish.h"
#ifdef ANDROID
void Two_debugDummy(Twofish_Byte* in, Twofish_Byte* out, Twofish_Byte* ivec);
#endif
void Twofish_cfb128_encrypt(Twofish_key* keyCtx, Twofish_Byte* in,
Twofish_Byte* out, size_t len,
Twofish_Byte* ivec, int32_t *num)
{
uint32_t n;
n = *num;
do {
while (n && len) {
*(out++) = ivec[n] ^= *(in++);
--len;
n = (n+1) % 16;
}
while (len>=16) {
Twofish_encrypt(keyCtx, ivec, ivec);
for (n=0; n<16; n+=sizeof(size_t)) {
/*
* Some GCC version(s) of Android's NDK produce code that leads to a crash (SIGBUS). The
* offending line if the line that produces the output by xor'ing the ivec. Somehow the
* compiler/optimizer seems to incorrectly setup the pointers. Adding a call to an
* external function that uses the pointer disabled or modifies this optimzing
* behaviour. This debug functions as such does nothing, it just disables some
* optimization. Don't use a local (static) function - the compiler sees that it does
* nothing and optimizes again :-) .
*/
#ifdef ANDROID
Two_debugDummy(in, out, ivec);
#endif
*(size_t*)(out+n) = *(size_t*)(ivec+n) ^= *(size_t*)(in+n);;
}
len -= 16;
out += 16;
in += 16;
}
n = 0;
if (len) {
Twofish_encrypt(keyCtx, ivec, ivec);
while (len--) {
out[n] = ivec[n] ^= in[n];
++n;
}
}
*num = n;
return;
} while (0);
}
void Twofish_cfb128_decrypt(Twofish_key* keyCtx, Twofish_Byte* in,
Twofish_Byte* out, size_t len,
Twofish_Byte* ivec, int32_t *num)
{
uint32_t n;
n = *num;
do {
while (n && len) {
unsigned char c;
*(out++) = ivec[n] ^ (c = *(in++)); ivec[n] = c;
--len;
n = (n+1) % 16;
}
while (len>=16) {
Twofish_encrypt(keyCtx, ivec, ivec);
for (n=0; n<16; n+=sizeof(size_t)) {
size_t t = *(size_t*)(in+n);
*(size_t*)(out+n) = *(size_t*)(ivec+n) ^ t;
*(size_t*)(ivec+n) = t;
}
len -= 16;
out += 16;
in += 16;
}
n = 0;
if (len) {
Twofish_encrypt(keyCtx, ivec, ivec);
while (len--) {
unsigned char c;
out[n] = ivec[n] ^ (c = in[n]); ivec[n] = c;
++n;
}
}
*num = n;
return;
} while (0);
}