45da7a6fef
* refactored Perceptionhash for performance * update AUTHORS.md * replaced PerceptionHash algorithm * Update hashcompute_test.go Co-authored-by: Dong-hee Na <donghee.na92@gmail.com> * Update hashcompute_test.go Co-authored-by: Evan Oberholster <eroberholster@gmail.com> Co-authored-by: Dong-hee Na <donghee.na92@gmail.com>
204 lines
5.7 KiB
Go
204 lines
5.7 KiB
Go
package transforms
|
|
|
|
import "math"
|
|
|
|
// DCT1DFast64 function returns result of DCT-II.
|
|
// DCT type II, unscaled. Algorithm by Byeong Gi Lee, 1984.
|
|
// Static implementation by Evan Oberholster, 2022.
|
|
func DCT1DFast64(input []float64) {
|
|
var temp [64]float64
|
|
for i := 0; i < 32; i++ {
|
|
x, y := input[i], input[63-i]
|
|
temp[i] = x + y
|
|
temp[i+32] = (x - y) / dct64[i]
|
|
}
|
|
forwardTransformStatic32(temp[:32])
|
|
forwardTransformStatic32(temp[32:])
|
|
for i := 0; i < 32-1; i++ {
|
|
input[i*2+0] = temp[i]
|
|
input[i*2+1] = temp[i+32] + temp[i+32+1]
|
|
}
|
|
input[62], input[63] = temp[31], temp[63]
|
|
}
|
|
|
|
func forwardTransformStatic32(input []float64) {
|
|
var temp [32]float64
|
|
for i := 0; i < 16; i++ {
|
|
x, y := input[i], input[31-i]
|
|
temp[i] = x + y
|
|
temp[i+16] = (x - y) / dct32[i]
|
|
}
|
|
forwardTransformStatic16(temp[:16])
|
|
forwardTransformStatic16(temp[16:])
|
|
for i := 0; i < 16-1; i++ {
|
|
input[i*2+0] = temp[i]
|
|
input[i*2+1] = temp[i+16] + temp[i+16+1]
|
|
}
|
|
|
|
input[30], input[31] = temp[15], temp[31]
|
|
}
|
|
|
|
func forwardTransformStatic16(input []float64) {
|
|
var temp [16]float64
|
|
for i := 0; i < 8; i++ {
|
|
x, y := input[i], input[15-i]
|
|
temp[i] = x + y
|
|
temp[i+8] = (x - y) / dct16[i]
|
|
}
|
|
forwardTransformStatic8(temp[:8])
|
|
forwardTransformStatic8(temp[8:])
|
|
for i := 0; i < 8-1; i++ {
|
|
input[i*2+0] = temp[i]
|
|
input[i*2+1] = temp[i+8] + temp[i+8+1]
|
|
}
|
|
|
|
input[14], input[15] = temp[7], temp[15]
|
|
}
|
|
|
|
func forwardTransformStatic8(input []float64) {
|
|
var temp [8]float64
|
|
x0, y0 := input[0], input[7]
|
|
x1, y1 := input[1], input[6]
|
|
x2, y2 := input[2], input[5]
|
|
x3, y3 := input[3], input[4]
|
|
|
|
temp[0] = x0 + y0
|
|
temp[1] = x1 + y1
|
|
temp[2] = x2 + y2
|
|
temp[3] = x3 + y3
|
|
temp[4] = (x0 - y0) / dct8[0]
|
|
temp[5] = (x1 - y1) / dct8[1]
|
|
temp[6] = (x2 - y2) / dct8[2]
|
|
temp[7] = (x3 - y3) / dct8[3]
|
|
|
|
forwardTransformStatic4(temp[:4])
|
|
forwardTransformStatic4(temp[4:])
|
|
|
|
input[0] = temp[0]
|
|
input[1] = temp[4] + temp[5]
|
|
input[2] = temp[1]
|
|
input[3] = temp[5] + temp[6]
|
|
input[4] = temp[2]
|
|
input[5] = temp[6] + temp[7]
|
|
input[6] = temp[3]
|
|
input[7] = temp[7]
|
|
}
|
|
|
|
func forwardTransformStatic4(input []float64) {
|
|
var (
|
|
t0, t1, t2, t3 float64
|
|
)
|
|
x0, y0 := input[0], input[3]
|
|
x1, y1 := input[1], input[2]
|
|
|
|
t0 = x0 + y0
|
|
t1 = x1 + y1
|
|
t2 = (x0 - y0) / dct4[0]
|
|
t3 = (x1 - y1) / dct4[1]
|
|
|
|
x, y := t0, t1
|
|
t0 += t1
|
|
t1 = (x - y) / dct2[0]
|
|
|
|
x, y = t2, t3
|
|
t2 += t3
|
|
t3 = (x - y) / dct2[0]
|
|
|
|
input[0] = t0
|
|
input[1] = t2 + t3
|
|
input[2] = t1
|
|
input[3] = t3
|
|
}
|
|
|
|
func init() {
|
|
// dct256
|
|
for i := 0; i < 128; i++ {
|
|
dct256[i] = (math.Cos((float64(i)+0.5)*math.Pi/256) * 2)
|
|
}
|
|
// dct128
|
|
for i := 0; i < 64; i++ {
|
|
dct128[i] = (math.Cos((float64(i)+0.5)*math.Pi/128) * 2)
|
|
}
|
|
}
|
|
|
|
// Static DCT Tables
|
|
var (
|
|
dct256 = [128]float64{}
|
|
dct128 = [64]float64{}
|
|
dct64 = [32]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(1)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(2)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(3)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(4)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(5)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(6)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(7)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(8)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(9)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(10)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(11)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(12)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(13)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(14)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(15)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(16)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(17)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(18)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(19)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(20)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(21)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(22)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(23)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(24)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(25)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(26)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(27)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(28)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(29)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(30)+0.5)*math.Pi/64) * 2),
|
|
(math.Cos((float64(31)+0.5)*math.Pi/64) * 2),
|
|
}
|
|
dct32 = [16]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(1)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(2)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(3)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(4)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(5)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(6)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(7)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(8)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(9)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(10)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(11)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(12)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(13)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(14)+0.5)*math.Pi/32) * 2),
|
|
(math.Cos((float64(15)+0.5)*math.Pi/32) * 2),
|
|
}
|
|
dct16 = [8]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(1)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(2)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(3)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(4)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(5)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(6)+0.5)*math.Pi/16) * 2),
|
|
(math.Cos((float64(7)+0.5)*math.Pi/16) * 2),
|
|
}
|
|
dct8 = [4]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/8) * 2),
|
|
(math.Cos((float64(1)+0.5)*math.Pi/8) * 2),
|
|
(math.Cos((float64(2)+0.5)*math.Pi/8) * 2),
|
|
(math.Cos((float64(3)+0.5)*math.Pi/8) * 2),
|
|
}
|
|
dct4 = [2]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/4) * 2),
|
|
(math.Cos((float64(1)+0.5)*math.Pi/4) * 2),
|
|
}
|
|
dct2 = [1]float64{
|
|
(math.Cos((float64(0)+0.5)*math.Pi/2) * 2),
|
|
}
|
|
)
|