...
Source file src/pkg/vendor/golang.org/x/crypto/hkdf/hkdf.go
1
2
3
4
5
6
7
8
9
10
11 package hkdf
12
13 import (
14 "crypto/hmac"
15 "errors"
16 "hash"
17 "io"
18 )
19
20
21
22
23
24
25
26 func Extract(hash func() hash.Hash, secret, salt []byte) []byte {
27 if salt == nil {
28 salt = make([]byte, hash().Size())
29 }
30 extractor := hmac.New(hash, salt)
31 extractor.Write(secret)
32 return extractor.Sum(nil)
33 }
34
35 type hkdf struct {
36 expander hash.Hash
37 size int
38
39 info []byte
40 counter byte
41
42 prev []byte
43 buf []byte
44 }
45
46 func (f *hkdf) Read(p []byte) (int, error) {
47
48 need := len(p)
49 remains := len(f.buf) + int(255-f.counter+1)*f.size
50 if remains < need {
51 return 0, errors.New("hkdf: entropy limit reached")
52 }
53
54 n := copy(p, f.buf)
55 p = p[n:]
56
57
58 for len(p) > 0 {
59 f.expander.Reset()
60 f.expander.Write(f.prev)
61 f.expander.Write(f.info)
62 f.expander.Write([]byte{f.counter})
63 f.prev = f.expander.Sum(f.prev[:0])
64 f.counter++
65
66
67 f.buf = f.prev
68 n = copy(p, f.buf)
69 p = p[n:]
70 }
71
72 f.buf = f.buf[n:]
73
74 return need, nil
75 }
76
77
78
79
80
81
82
83 func Expand(hash func() hash.Hash, pseudorandomKey, info []byte) io.Reader {
84 expander := hmac.New(hash, pseudorandomKey)
85 return &hkdf{expander, expander.Size(), info, 1, nil, nil}
86 }
87
88
89
90 func New(hash func() hash.Hash, secret, salt, info []byte) io.Reader {
91 prk := Extract(hash, secret, salt)
92 return Expand(hash, prk, info)
93 }
94
View as plain text