radare2/libr/hash/hash.c

208 lines
5.0 KiB
C

/* radare - LGPL - Copyright 2007-2016 pancake */
#include "r_hash.h"
#include "r_util.h"
R_LIB_VERSION(r_hash);
struct { const char *name; ut64 bit; }
static const hash_name_bytes[] = {
{"all", UT64_MAX},
{"xor", R_HASH_XOR},
{"xorpair", R_HASH_XORPAIR},
{"md4", R_HASH_MD4},
{"md5", R_HASH_MD5},
{"sha1", R_HASH_SHA1},
{"sha256", R_HASH_SHA256},
{"sha384", R_HASH_SHA384},
{"sha512", R_HASH_SHA512},
{"crc16", R_HASH_CRC16},
{"crc32", R_HASH_CRC32},
{"adler32", R_HASH_ADLER32},
{"xxhash", R_HASH_XXHASH},
{"parity", R_HASH_PARITY},
{"entropy", R_HASH_ENTROPY},
{"hamdist", R_HASH_HAMDIST},
{"pcprint", R_HASH_PCPRINT},
{"mod255", R_HASH_MOD255},
// {"base64", R_HASH_BASE64},
// {"base91", R_HASH_BASE91},
// {"punycode", R_HASH_PUNYCODE},
{"luhn", R_HASH_LUHN},
{NULL, 0}
};
/* returns 0-100 */
R_API int r_hash_pcprint(const ut8 *buffer, ut64 len) {
const ut8 *end = buffer + len;
int n;
if (len < 1) {
return 0;
}
for (n = 0; buffer < end; buffer++) {
if (IS_PRINTABLE (*buffer)) {
n++;
}
}
return ((100 * n) / len);
}
R_API int r_hash_parity(const ut8 *buf, ut64 len) {
const ut8 *end = buf+len;
ut32 ones = 0;
for (;buf<end;buf++) {
ut8 x = buf[0];
ones += ((x&128)?1:0) + ((x&64)?1:0) + ((x&32)?1:0) + ((x&16)?1:0) +
((x&8)?1:0) + ((x&4)?1:0) + ((x&2)?1:0) + ((x&1)?1:0);
}
return ones%2;
}
/* These functions comes from 0xFFFF */
/* fmi: nopcode.org/0xFFFF */
R_API ut16 r_hash_xorpair(const ut8 *a, ut64 len) {
ut16 result = 0, *b = (ut16 *)a;
for (len >>= 1; len--; b++) {
result ^= *b;
}
return result;
}
R_API ut8 r_hash_xor(const ut8 *b, ut64 len) {
ut8 res = 0;
for (; len--; b++)
res ^= *b;
return res;
}
R_API ut8 r_hash_mod255(const ut8 *b, ut64 len) {
int i, c = 0;
/* from gdb */
for (i = 0; i < len; i++)
c += b[i];
return c%255;
}
R_API ut8 r_hash_deviation(const ut8 *b, ut64 len) {
int i, c;
for (c = i = 0, --len; i < len; i++)
c += R_ABS (b[i+1] - b[i]);
return c;
}
R_API const char *r_hash_name(ut64 bit) {
int i;
for (i=1; hash_name_bytes[i].bit; i++)
if (bit & hash_name_bytes[i].bit)
return hash_name_bytes[i].name;
return "";
}
R_API int r_hash_size(ut64 algo) {
if (algo & R_HASH_MD4) return R_HASH_SIZE_MD4;
if (algo & R_HASH_MD5) return R_HASH_SIZE_MD5;
if (algo & R_HASH_SHA1) return R_HASH_SIZE_SHA1;
if (algo & R_HASH_SHA256) return R_HASH_SIZE_SHA256;
if (algo & R_HASH_SHA384) return R_HASH_SIZE_SHA384;
if (algo & R_HASH_SHA512) return R_HASH_SIZE_SHA512;
if (algo & R_HASH_CRC16) return R_HASH_SIZE_CRC16;
if (algo & R_HASH_CRC32) return R_HASH_SIZE_CRC32;
if (algo & R_HASH_XXHASH) return R_HASH_SIZE_XXHASH;
if (algo & R_HASH_ADLER32) return R_HASH_SIZE_ADLER32;
if (algo & R_HASH_PARITY) return R_HASH_SIZE_PARITY;
if (algo & R_HASH_ENTROPY) return R_HASH_SIZE_ENTROPY;
if (algo & R_HASH_HAMDIST) return R_HASH_SIZE_HAMDIST;
if (algo & R_HASH_XOR) return R_HASH_SIZE_XOR;
if (algo & R_HASH_XORPAIR) return R_HASH_SIZE_XORPAIR;
if (algo & R_HASH_MOD255) return R_HASH_SIZE_MOD255;
if (algo & R_HASH_PCPRINT) return R_HASH_SIZE_PCPRINT;
if (algo & R_HASH_LUHN) return R_HASH_SIZE_LUHN;
return 0;
}
/* Converts a comma separated list of names to the respective bit combination */
R_API ut64 r_hash_name_to_bits(const char *name) {
char tmp[128];
int i;
const char *ptr;
ut64 ret;
ret = 0;
ptr = name;
if (!ptr)
return ret;
do {
/* Eat everything up to the comma */
for (i = 0; *ptr && *ptr != ',' && i < sizeof (tmp) - 1; i++)
tmp[i] = *ptr++;
/* Safety net */
tmp[i] = '\0';
for (i = 0; hash_name_bytes[i].name; i++) {
if (!strcasecmp(tmp, hash_name_bytes[i].name)) {
ret |= hash_name_bytes[i].bit;
break;
}
}
/* Skip the trailing comma, if any */
if (*ptr)
ptr++;
} while (*ptr);
return ret;
}
R_API void r_hash_do_spice(RHash *ctx, int algo, int loops, RHashSeed *seed) {
ut8 buf[1024];
int i, len, hlen = r_hash_size (algo);
for (i = 0; i< loops; i++) {
if (seed) {
if (seed->prefix) {
memcpy (buf, seed->buf, seed->len);
memcpy (buf+seed->len, ctx->digest, hlen);
} else {
memcpy (buf, ctx->digest, hlen);
memcpy (buf+hlen, seed->buf, seed->len);
}
len = hlen + seed->len;
} else {
memcpy (buf, ctx->digest, hlen);
len = hlen;
}
(void)r_hash_calculate (ctx, algo, buf, len);
}
}
R_API char *r_hash_to_string(RHash *ctx, const char *name, const ut8 *data, int len) {
ut64 algo = r_hash_name_to_bits (name);
char *digest_hex = NULL;
RHash *myctx = NULL;
int i, digest_size;
if (!algo || !data) {
return NULL;
}
if (!ctx) {
myctx = ctx = r_hash_new (true, algo);
}
r_hash_do_begin (ctx, algo);
digest_size = r_hash_calculate (ctx, algo, data, len);
r_hash_do_end (ctx, algo);
if (digest_size > 0) {
if (digest_size * 2 < digest_size) {
digest_hex = NULL;
} else {
digest_hex = malloc ((digest_size * 2) + 1);
for (i = 0; i < digest_size; i++) {
sprintf (digest_hex + (i * 2), "%02x", ctx->digest[i]);
}
digest_hex[digest_size * 2] = 0;
}
}
r_hash_free (myctx);
return digest_hex;
}