1 // Copyright 2009 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
14 // Split a premaster secret in two as specified in RFC 4346, section 5.
15 func splitPreMasterSecret(secret []byte) (s1, s2 []byte) {
16 s1 = secret[0 : (len(secret)+1)/2]
17 s2 = secret[len(secret)/2:]
21 // pHash implements the P_hash function, as defined in RFC 4346, section 5.
22 func pHash(result, secret, seed []byte, hash func() hash.Hash) {
23 h := hmac.New(hash, secret)
34 if j+todo > len(result) {
35 todo = len(result) - j
37 copy(result[j:j+todo], b)
46 // pRF10 implements the TLS 1.0 pseudo-random function, as defined in RFC 2246, section 5.
47 func pRF10(result, secret, label, seed []byte) {
51 labelAndSeed := make([]byte, len(label)+len(seed))
52 copy(labelAndSeed, label)
53 copy(labelAndSeed[len(label):], seed)
55 s1, s2 := splitPreMasterSecret(secret)
56 pHash(result, s1, labelAndSeed, hashMD5)
57 result2 := make([]byte, len(result))
58 pHash(result2, s2, labelAndSeed, hashSHA1)
60 for i, b := range result2 {
65 // pRF30 implements the SSL 3.0 pseudo-random function, as defined in
66 // www.mozilla.org/projects/security/pki/nss/ssl/draft302.txt section 6.
67 func pRF30(result, secret, label, seed []byte) {
68 hashSHA1 := sha1.New()
73 // RFC5246 section 6.3 says that the largest PRF output needed is 128
74 // bytes. Since no more ciphersuites will be added to SSLv3, this will
75 // remain true. Each iteration gives us 16 bytes so 10 iterations will
78 for done < len(result) {
79 for j := 0; j <= i; j++ {
84 hashSHA1.Write(b[:i+1])
85 hashSHA1.Write(secret)
87 digest := hashSHA1.Sum(nil)
93 done += copy(result[done:], hashMD5.Sum(nil))
99 tlsRandomLength = 32 // Length of a random nonce in TLS 1.1.
100 masterSecretLength = 48 // Length of a master secret in TLS 1.1.
101 finishedVerifyLength = 12 // Length of verify_data in a Finished message.
104 var masterSecretLabel = []byte("master secret")
105 var keyExpansionLabel = []byte("key expansion")
106 var clientFinishedLabel = []byte("client finished")
107 var serverFinishedLabel = []byte("server finished")
109 // masterFromPreMasterSecret generates the master secret from the pre-master
110 // secret. See http://tools.ietf.org/html/rfc5246#section-8.1
111 func masterFromPreMasterSecret(version uint16, preMasterSecret, clientRandom, serverRandom []byte) []byte {
113 if version == versionSSL30 {
117 var seed [tlsRandomLength * 2]byte
118 copy(seed[0:len(clientRandom)], clientRandom)
119 copy(seed[len(clientRandom):], serverRandom)
120 masterSecret := make([]byte, masterSecretLength)
121 prf(masterSecret, preMasterSecret, masterSecretLabel, seed[0:])
125 // keysFromMasterSecret generates the connection keys from the master
126 // secret, given the lengths of the MAC key, cipher key and IV, as defined in
127 // RFC 2246, section 6.3.
128 func keysFromMasterSecret(version uint16, masterSecret, clientRandom, serverRandom []byte, macLen, keyLen, ivLen int) (clientMAC, serverMAC, clientKey, serverKey, clientIV, serverIV []byte) {
130 if version == versionSSL30 {
134 var seed [tlsRandomLength * 2]byte
135 copy(seed[0:len(clientRandom)], serverRandom)
136 copy(seed[len(serverRandom):], clientRandom)
138 n := 2*macLen + 2*keyLen + 2*ivLen
139 keyMaterial := make([]byte, n)
140 prf(keyMaterial, masterSecret, keyExpansionLabel, seed[0:])
141 clientMAC = keyMaterial[:macLen]
142 keyMaterial = keyMaterial[macLen:]
143 serverMAC = keyMaterial[:macLen]
144 keyMaterial = keyMaterial[macLen:]
145 clientKey = keyMaterial[:keyLen]
146 keyMaterial = keyMaterial[keyLen:]
147 serverKey = keyMaterial[:keyLen]
148 keyMaterial = keyMaterial[keyLen:]
149 clientIV = keyMaterial[:ivLen]
150 keyMaterial = keyMaterial[ivLen:]
151 serverIV = keyMaterial[:ivLen]
155 func newFinishedHash(version uint16) finishedHash {
156 return finishedHash{md5.New(), sha1.New(), md5.New(), sha1.New(), version}
159 // A finishedHash calculates the hash of a set of handshake messages suitable
160 // for including in a Finished message.
161 type finishedHash struct {
169 func (h finishedHash) Write(msg []byte) (n int, err error) {
170 h.clientMD5.Write(msg)
171 h.clientSHA1.Write(msg)
172 h.serverMD5.Write(msg)
173 h.serverSHA1.Write(msg)
177 // finishedSum10 calculates the contents of the verify_data member of a TLSv1
178 // Finished message given the MD5 and SHA1 hashes of a set of handshake
180 func finishedSum10(md5, sha1, label, masterSecret []byte) []byte {
181 seed := make([]byte, len(md5)+len(sha1))
183 copy(seed[len(md5):], sha1)
184 out := make([]byte, finishedVerifyLength)
185 pRF10(out, masterSecret, label, seed)
189 // finishedSum30 calculates the contents of the verify_data member of a SSLv3
190 // Finished message given the MD5 and SHA1 hashes of a set of handshake
192 func finishedSum30(md5, sha1 hash.Hash, masterSecret []byte, magic [4]byte) []byte {
194 md5.Write(masterSecret)
195 md5.Write(ssl30Pad1[:])
196 md5Digest := md5.Sum(nil)
199 md5.Write(masterSecret)
200 md5.Write(ssl30Pad2[:])
202 md5Digest = md5.Sum(nil)
205 sha1.Write(masterSecret)
206 sha1.Write(ssl30Pad1[:40])
207 sha1Digest := sha1.Sum(nil)
210 sha1.Write(masterSecret)
211 sha1.Write(ssl30Pad2[:40])
212 sha1.Write(sha1Digest)
213 sha1Digest = sha1.Sum(nil)
215 ret := make([]byte, len(md5Digest)+len(sha1Digest))
217 copy(ret[len(md5Digest):], sha1Digest)
221 var ssl3ClientFinishedMagic = [4]byte{0x43, 0x4c, 0x4e, 0x54}
222 var ssl3ServerFinishedMagic = [4]byte{0x53, 0x52, 0x56, 0x52}
224 // clientSum returns the contents of the verify_data member of a client's
226 func (h finishedHash) clientSum(masterSecret []byte) []byte {
227 if h.version == versionSSL30 {
228 return finishedSum30(h.clientMD5, h.clientSHA1, masterSecret, ssl3ClientFinishedMagic)
231 md5 := h.clientMD5.Sum(nil)
232 sha1 := h.clientSHA1.Sum(nil)
233 return finishedSum10(md5, sha1, clientFinishedLabel, masterSecret)
236 // serverSum returns the contents of the verify_data member of a server's
238 func (h finishedHash) serverSum(masterSecret []byte) []byte {
239 if h.version == versionSSL30 {
240 return finishedSum30(h.serverMD5, h.serverSHA1, masterSecret, ssl3ServerFinishedMagic)
243 md5 := h.serverMD5.Sum(nil)
244 sha1 := h.serverSHA1.Sum(nil)
245 return finishedSum10(md5, sha1, serverFinishedLabel, masterSecret)