aboutsummaryrefslogtreecommitdiff
path: root/src/hash/sysvchecksum.c
blob: 527bcc76e1613ec26ebf9fcf99f8bc281dd2ef3c (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
#include "../crypto.h"
#include <stdio.h>
#include <stdint.h>
#include <math.h>

struct sysvchecksum_hash sysvchecksum_init(){
  return (struct sysvchecksum_hash){.check = 0};
}

int sysvchecksum_free_l(lua_State* L){
  return 0;
}

void sysvchecksum_update(uint8_t* aa, size_t len, struct sysvchecksum_hash* hash){
  for(int i = 0; i != len; i++)
    hash->check += aa[i];
}

uint16_t sysvchecksum_final(struct sysvchecksum_hash* hash){
  //uint16_t r = hash->check % (int)pow(2,16) + (hash->check % (int)pow(2,32)) / (int)pow(2,16);
  //return (r % (int)pow(2,16)) + (r / (int)pow(2,16));
  //rewrote from https://github.com/cxmcc/unixsums/blob/master/sysvsum/sysvsum.go, pow breaks with clang -O3 somehow
  uint32_t r = (hash->check & 0xffff) + (hash->check >> 16);
  return r & 0xffff + (r >> 16);
}

uint16_t sysvchecksum(uint8_t* aa, size_t len){
  struct sysvchecksum_hash a = sysvchecksum_init();
  sysvchecksum_update(aa, len, &a);
  return sysvchecksum_final(&a);
}

common_hash_clone(sysvchecksum);
common_hash_init_update(sysvchecksum);

int l_sysvchecksum_final(lua_State* L){
  struct sysvchecksum_hash* a = (struct sysvchecksum_hash*)lua_touserdata(L, 1);
  uint16_t u = sysvchecksum_final(a);
  char digest[16];
  sprintf(digest,"%x",u);
  lua_pushstring(L, digest);
  return 1;
}
 
int l_sysvchecksum(lua_State* L){
  if(lua_gettop(L) == 0) return l_sysvchecksum_init(L);
  size_t len = 0;
  uint8_t* a = (uint8_t*)luaL_checklstring(L, 1, &len);
  
  char digest[16];

  uint16_t u = sysvchecksum(a, len);
  sprintf(digest,"%x",u);
  lua_pushstring(L, digest);

  return 1;
}