Blame src/hash.c

Packit Service 8eee21
/**
Packit Service 8eee21
 * Seccomp Library hash code
Packit Service 8eee21
 *
Packit Service 8eee21
 */
Packit Service 8eee21
Packit Service 8eee21
/*
Packit Service 8eee21
 * This code is based on MurmurHash3.cpp from Austin Appleby and is placed in
Packit Service 8eee21
 * the public domain.
Packit Service 8eee21
 *
Packit Service 8eee21
 * https://github.com/aappleby/smhasher
Packit Service 8eee21
 *
Packit Service 8eee21
 */
Packit Service 8eee21
Packit Service 8eee21
#include <stdlib.h>
Packit Service 8eee21
#include <inttypes.h>
Packit Service 8eee21
Packit Service 8eee21
#include "hash.h"
Packit Service 8eee21
Packit Service 8eee21
static inline uint32_t getblock32(const uint32_t *p, int i)
Packit Service 8eee21
{
Packit Service 8eee21
	return p[i];
Packit Service 8eee21
}
Packit Service 8eee21
Packit Service 8eee21
static inline uint32_t rotl32(uint32_t x, int8_t r)
Packit Service 8eee21
{
Packit Service 8eee21
	return (x << r) | (x >> (32 - r));
Packit Service 8eee21
}
Packit Service 8eee21
Packit Service 8eee21
static inline uint32_t fmix32(uint32_t h)
Packit Service 8eee21
{
Packit Service 8eee21
	h ^= h >> 16;
Packit Service 8eee21
	h *= 0x85ebca6b;
Packit Service 8eee21
	h ^= h >> 13;
Packit Service 8eee21
	h *= 0xc2b2ae35;
Packit Service 8eee21
	h ^= h >> 16;
Packit Service 8eee21
Packit Service 8eee21
	return h;
Packit Service 8eee21
}
Packit Service 8eee21
Packit Service 8eee21
/* NOTE: this is an implementation of MurmurHash3_x86_32 */
Packit Service 8eee21
uint32_t hash(const void *key, size_t length)
Packit Service 8eee21
{
Packit Service 8eee21
	const uint8_t *data = (const uint8_t *)key;
Packit Service 8eee21
	const uint32_t *blocks;
Packit Service 8eee21
	const uint8_t *tail;
Packit Service 8eee21
	const int nblocks = length / 4;
Packit Service 8eee21
	const uint32_t c1 = 0xcc9e2d51;
Packit Service 8eee21
	const uint32_t c2 = 0x1b873593;
Packit Service 8eee21
	uint32_t k1;
Packit Service 8eee21
	uint32_t k2 = 0;
Packit Service 8eee21
	int i;
Packit Service 8eee21
Packit Service 8eee21
	/* NOTE: we always force a seed of 0 */
Packit Service 8eee21
	uint32_t h1 = 0;
Packit Service 8eee21
Packit Service 8eee21
	/* body */
Packit Service 8eee21
	blocks = (const uint32_t *)(data + nblocks * 4);
Packit Service 8eee21
	for(i = -nblocks; i; i++) {
Packit Service 8eee21
		k1 = getblock32(blocks, i);
Packit Service 8eee21
Packit Service 8eee21
		k1 *= c1;
Packit Service 8eee21
		k1 = rotl32(k1, 15);
Packit Service 8eee21
		k1 *= c2;
Packit Service 8eee21
Packit Service 8eee21
		h1 ^= k1;
Packit Service 8eee21
		h1 = rotl32(h1, 13);
Packit Service 8eee21
		h1 = h1 * 5 + 0xe6546b64;
Packit Service 8eee21
	}
Packit Service 8eee21
Packit Service 8eee21
	/* tail */
Packit Service 8eee21
	tail = (const uint8_t *)(data + nblocks * 4);
Packit Service 8eee21
	switch(length & 3) {
Packit Service 8eee21
	case 3:
Packit Service 8eee21
		k2 ^= tail[2] << 16;
Packit Service 8eee21
	case 2:
Packit Service 8eee21
		k2 ^= tail[1] << 8;
Packit Service 8eee21
	case 1:
Packit Service 8eee21
		k2 ^= tail[0];
Packit Service 8eee21
		k2 *= c1;
Packit Service 8eee21
		k2 = rotl32(k2, 15);
Packit Service 8eee21
		k2 *= c2;
Packit Service 8eee21
		h1 ^= k2;
Packit Service 8eee21
	};
Packit Service 8eee21
Packit Service 8eee21
	/* finalization */
Packit Service 8eee21
	h1 ^= length;
Packit Service 8eee21
	h1 = fmix32(h1);
Packit Service 8eee21
Packit Service 8eee21
	return h1;
Packit Service 8eee21
}