radare2/libr/hash/hash.c
2017-04-11 22:47:16 +02:00

234 lines
5.0 KiB
C

/* radare - LGPL - Copyright 2007-2017 pancake */
#include <r_hash.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 },
{ "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 },
{ "crc8smbus", R_HASH_CRC8_SMBUS },
{ "crc15can", R_HASH_CRC15_CAN },
{ "crc16", R_HASH_CRC16 },
{ "crc16hdlc", R_HASH_CRC16_HDLC },
{ "crc16usb", R_HASH_CRC16_USB },
{ "crc16citt", R_HASH_CRC16_CITT },
{ "crc24", R_HASH_CRC24 },
{ "crc32", R_HASH_CRC32 },
{ "crc32c", R_HASH_CRC32C },
{ "crc32ecma267", R_HASH_CRC32_ECMA_267 },
{ 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) {
#define ALGOBIT(x) if (algo & R_HASH_ ## x) { return R_HASH_SIZE_ ## x; }
ALGOBIT (MD4);
ALGOBIT (MD5);
ALGOBIT (SHA1);
ALGOBIT (SHA256);
ALGOBIT (SHA384);
ALGOBIT (SHA512);
ALGOBIT (CRC16);
ALGOBIT (CRC32);
ALGOBIT (XXHASH);
ALGOBIT (ADLER32);
ALGOBIT (PARITY);
ALGOBIT (ENTROPY);
ALGOBIT (HAMDIST);
ALGOBIT (XOR);
ALGOBIT (XORPAIR);
ALGOBIT (MOD255);
ALGOBIT (PCPRINT);
ALGOBIT (LUHN);
ALGOBIT (CRC8_SMBUS);
ALGOBIT (CRC15_CAN);
ALGOBIT (CRC16_HDLC);
ALGOBIT (CRC16_USB);
ALGOBIT (CRC16_CITT);
ALGOBIT (CRC24);
ALGOBIT (CRC32C);
ALGOBIT (CRC32_ECMA_267);
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;
}