aboutsummaryrefslogtreecommitdiff
path: root/src/hash/fnv.c
blob: e0fc98120df0b08cb09665f2d4fd6953e7ecee36 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include "../util.h"
#include "../crypto.h"
#include <stdio.h>
#include <stdint.h>

struct fnv_1_hash fnv_1_init(enum fnv_version A){
  return (struct fnv_1_hash){.A = A, .hash = (A != v_0) * 0xcbf29ce484222325};
}

void fnv_1_update(uint8_t* in, size_t len, struct fnv_1_hash* hash){
  uint64_t prime = 0x100000001b3;

  for(int i = 0; i != len; i++){
    switch(hash->A){
      case v_1:
      case v_0:
        hash->hash *= prime;
        hash->hash ^= in[i];
        break;
      case v_a:
        hash->hash ^= in[i];
        hash->hash *= prime;
        break;
    }
  }
}

uint64_t fnv_1_final(struct fnv_1_hash* hash){
  return hash->hash;
}

uint64_t fnv_1(uint8_t* in, size_t len, enum fnv_version A){
  struct fnv_1_hash a = fnv_1_init(A);
  fnv_1_update(in, len, &a);
  return fnv_1_final(&a);
}

lua_common_hash_update(fnv_1, fnv_1);
lua_common_hash_update(fnv_1, fnv_0);
lua_common_hash_update(fnv_1, fnv_a);

lua_common_hash_init_warg(fnv_1, fnv_1, ;, v_1);
lua_common_hash_init_warg(fnv_1, fnv_0, ;, v_0);
lua_common_hash_init_warg(fnv_1, fnv_a, ;, v_a);

#define aaa(v)\
int l_fnv_##v##_final(lua_State* L){\
  lua_pushstring(L, "ud");\
  lua_gettable(L, 1);\
\
  struct fnv_1_hash* a = (struct fnv_1_hash*)lua_touserdata(L, -1);\
  uint64_t u = fnv_1_final(a);\
  char digest[64];\
  sprintf(digest,"%16llx",u);\
  lua_pushstring(L, digest);\
  return 1;\
}

aaa(0);
aaa(1);
aaa(a);

int l_fnv_0(lua_State* L){
  if(lua_gettop(L) == 0) return l_fnv_0_init(L);
  size_t len = 0;
  uint8_t* a = (uint8_t*)luaL_checklstring(L, 1, &len);

  char digest[64];

  uint64_t u = fnv_1(a, len, v_0);
  sprintf(digest,"%16llx",u);
  lua_pushstring(L, digest);
  return 1;
}

int l_fnv_1(lua_State* L){ 
  if(lua_gettop(L) == 0) return l_fnv_1_init(L);
  size_t len = 0;
  uint8_t* a = (uint8_t*)luaL_checklstring(L, 1, &len);

  char digest[64];

  uint64_t u = fnv_1(a, len, v_1);
  sprintf(digest,"%16llx",u);
  lua_pushstring(L, digest);
  return 1;
}

int l_fnv_a(lua_State* L){ 
  if(lua_gettop(L) == 0) return l_fnv_a_init(L);
  size_t len = 0;
  uint8_t* a = (uint8_t*)luaL_checklstring(L, 1, &len);

  char digest[64];

  uint64_t u = fnv_1(a, len, v_a);
  sprintf(digest,"%16llx",u);
  lua_pushstring(L, digest);
  return 1;
}