-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathhamming_distance.c
47 lines (36 loc) · 1.62 KB
/
hamming_distance.c
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
/*
(c) Bharanee Rathna 2011
CC BY-SA 3.0
http://creativecommons.org/licenses/by-sa/3.0/
Free for every type of use. The author cannot be legally held responsible for
any damages resulting from the use of this work. All modifications or derivatives
need to be attributed.
*/
#include <ctype.h>
#include "postgres.h"
#include "fmgr.h"
#ifdef PG_MODULE_MAGIC
PG_MODULE_MAGIC;
#endif
int popcount(uint64_t);
PG_FUNCTION_INFO_V1(hamming_distance);
Datum hamming_distance(PG_FUNCTION_ARGS) {
uint64_t v1 = PG_GETARG_INT64(0);
uint64_t v2 = PG_GETARG_INT64(1);
PG_RETURN_INT16(popcount(v1^v2));
}
// http://en.wikipedia.org/wiki/Hamming_weight
const uint64_t m1 = 0x5555555555555555; //binary: 0101...
const uint64_t m2 = 0x3333333333333333; //binary: 00110011..
const uint64_t m4 = 0x0f0f0f0f0f0f0f0f; //binary: 4 zeros, 4 ones ...
const uint64_t m8 = 0x00ff00ff00ff00ff; //binary: 8 zeros, 8 ones ...
const uint64_t m16 = 0x0000ffff0000ffff; //binary: 16 zeros, 16 ones ...
const uint64_t m32 = 0x00000000ffffffff; //binary: 32 zeros, 32 ones
const uint64_t hff = 0xffffffffffffffff; //binary: all ones
const uint64_t h01 = 0x0101010101010101; //the sum of 256 to the power of 0,1,2,3...
int popcount(uint64_t x) {
x -= (x >> 1) & m1; //put count of each 2 bits into those 2 bits
x = (x & m2) + ((x >> 2) & m2); //put count of each 4 bits into those 4 bits
x = (x + (x >> 4)) & m4; //put count of each 8 bits into those 8 bits
return (x * h01)>>56; //returns left 8 bits of x + (x<<8) + (x<<16) + (x<<24) + ...
}