summaryrefslogtreecommitdiffstats
path: root/tune.go
blob: 6624f8826a2f8da30a81c350afc9465e39c4797b (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
package luksy

import (
	"hash"
	"time"

	"golang.org/x/crypto/argon2"
	"golang.org/x/crypto/pbkdf2"
)

func durationOf(f func()) time.Duration {
	start := time.Now()
	f()
	return time.Since(start)
}

func IterationsPBKDF2(salt []byte, keyLen int, h func() hash.Hash) int {
	iterations := 2
	var d time.Duration
	for d < time.Second {
		d = durationOf(func() {
			_ = pbkdf2.Key([]byte{}, salt, iterations, keyLen, h)
		})
		if d < time.Second/10 {
			iterations *= 2
		} else {
			return iterations * int(time.Second) / int(d)
		}
	}
	return iterations
}

func memoryCostArgon2(salt []byte, keyLen, timeCost, threadsCost int, kdf func([]byte, []byte, uint32, uint32, uint8, uint32) []byte) int {
	memoryCost := 2
	var d time.Duration
	for d < time.Second {
		d = durationOf(func() {
			_ = kdf([]byte{}, salt, uint32(timeCost), uint32(memoryCost), uint8(threadsCost), uint32(keyLen))
		})
		if d < time.Second/10 {
			memoryCost *= 2
		} else {
			return memoryCost * int(float64(time.Second)/float64(d))
		}
	}
	return memoryCost
}

func MemoryCostArgon2(salt []byte, keyLen, timeCost, threadsCost int) int {
	return memoryCostArgon2(salt, keyLen, timeCost, threadsCost, argon2.Key)
}

func MemoryCostArgon2i(salt []byte, keyLen, timeCost, threadsCost int) int {
	return memoryCostArgon2(salt, keyLen, timeCost, threadsCost, argon2.IDKey)
}