Alternative TLS implementation in Go
Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

154 рядки
4.5 KiB

  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.
  4. package tls
  5. import (
  6. "crypto/hmac"
  7. "crypto/md5"
  8. "crypto/sha1"
  9. "hash"
  10. "os"
  11. )
  12. // Split a premaster secret in two as specified in RFC 4346, section 5.
  13. func splitPreMasterSecret(secret []byte) (s1, s2 []byte) {
  14. s1 = secret[0 : (len(secret)+1)/2]
  15. s2 = secret[len(secret)/2:]
  16. return
  17. }
  18. // pHash implements the P_hash function, as defined in RFC 4346, section 5.
  19. func pHash(result, secret, seed []byte, hash func() hash.Hash) {
  20. h := hmac.New(hash, secret)
  21. h.Write(seed)
  22. a := h.Sum()
  23. j := 0
  24. for j < len(result) {
  25. h.Reset()
  26. h.Write(a)
  27. h.Write(seed)
  28. b := h.Sum()
  29. todo := len(b)
  30. if j+todo > len(result) {
  31. todo = len(result) - j
  32. }
  33. copy(result[j:j+todo], b)
  34. j += todo
  35. h.Reset()
  36. h.Write(a)
  37. a = h.Sum()
  38. }
  39. }
  40. // pRF10 implements the TLS 1.0 pseudo-random function, as defined in RFC 2246, section 5.
  41. func pRF10(result, secret, label, seed []byte) {
  42. hashSHA1 := sha1.New
  43. hashMD5 := md5.New
  44. labelAndSeed := make([]byte, len(label)+len(seed))
  45. copy(labelAndSeed, label)
  46. copy(labelAndSeed[len(label):], seed)
  47. s1, s2 := splitPreMasterSecret(secret)
  48. pHash(result, s1, labelAndSeed, hashMD5)
  49. result2 := make([]byte, len(result))
  50. pHash(result2, s2, labelAndSeed, hashSHA1)
  51. for i, b := range result2 {
  52. result[i] ^= b
  53. }
  54. }
  55. const (
  56. tlsRandomLength = 32 // Length of a random nonce in TLS 1.1.
  57. masterSecretLength = 48 // Length of a master secret in TLS 1.1.
  58. finishedVerifyLength = 12 // Length of verify_data in a Finished message.
  59. )
  60. var masterSecretLabel = []byte("master secret")
  61. var keyExpansionLabel = []byte("key expansion")
  62. var clientFinishedLabel = []byte("client finished")
  63. var serverFinishedLabel = []byte("server finished")
  64. // keysFromPreMasterSecret generates the connection keys from the pre master
  65. // secret, given the lengths of the MAC key, cipher key and IV, as defined in
  66. // RFC 2246, section 6.3.
  67. func keysFromPreMasterSecret10(preMasterSecret, clientRandom, serverRandom []byte, macLen, keyLen, ivLen int) (masterSecret, clientMAC, serverMAC, clientKey, serverKey, clientIV, serverIV []byte) {
  68. var seed [tlsRandomLength * 2]byte
  69. copy(seed[0:len(clientRandom)], clientRandom)
  70. copy(seed[len(clientRandom):], serverRandom)
  71. masterSecret = make([]byte, masterSecretLength)
  72. pRF10(masterSecret, preMasterSecret, masterSecretLabel, seed[0:])
  73. copy(seed[0:len(clientRandom)], serverRandom)
  74. copy(seed[len(serverRandom):], clientRandom)
  75. n := 2*macLen + 2*keyLen + 2*ivLen
  76. keyMaterial := make([]byte, n)
  77. pRF10(keyMaterial, masterSecret, keyExpansionLabel, seed[0:])
  78. clientMAC = keyMaterial[:macLen]
  79. keyMaterial = keyMaterial[macLen:]
  80. serverMAC = keyMaterial[:macLen]
  81. keyMaterial = keyMaterial[macLen:]
  82. clientKey = keyMaterial[:keyLen]
  83. keyMaterial = keyMaterial[keyLen:]
  84. serverKey = keyMaterial[:keyLen]
  85. keyMaterial = keyMaterial[keyLen:]
  86. clientIV = keyMaterial[:ivLen]
  87. keyMaterial = keyMaterial[ivLen:]
  88. serverIV = keyMaterial[:ivLen]
  89. return
  90. }
  91. // A finishedHash calculates the hash of a set of handshake messages suitable
  92. // for including in a Finished message.
  93. type finishedHash struct {
  94. clientMD5 hash.Hash
  95. clientSHA1 hash.Hash
  96. serverMD5 hash.Hash
  97. serverSHA1 hash.Hash
  98. }
  99. func newFinishedHash() finishedHash {
  100. return finishedHash{md5.New(), sha1.New(), md5.New(), sha1.New()}
  101. }
  102. func (h finishedHash) Write(msg []byte) (n int, err os.Error) {
  103. h.clientMD5.Write(msg)
  104. h.clientSHA1.Write(msg)
  105. h.serverMD5.Write(msg)
  106. h.serverSHA1.Write(msg)
  107. return len(msg), nil
  108. }
  109. // finishedSum calculates the contents of the verify_data member of a Finished
  110. // message given the MD5 and SHA1 hashes of a set of handshake messages.
  111. func finishedSum(md5, sha1, label, masterSecret []byte) []byte {
  112. seed := make([]byte, len(md5)+len(sha1))
  113. copy(seed, md5)
  114. copy(seed[len(md5):], sha1)
  115. out := make([]byte, finishedVerifyLength)
  116. pRF10(out, masterSecret, label, seed)
  117. return out
  118. }
  119. // clientSum returns the contents of the verify_data member of a client's
  120. // Finished message.
  121. func (h finishedHash) clientSum(masterSecret []byte) []byte {
  122. md5 := h.clientMD5.Sum()
  123. sha1 := h.clientSHA1.Sum()
  124. return finishedSum(md5, sha1, clientFinishedLabel, masterSecret)
  125. }
  126. // serverSum returns the contents of the verify_data member of a server's
  127. // Finished message.
  128. func (h finishedHash) serverSum(masterSecret []byte) []byte {
  129. md5 := h.serverMD5.Sum()
  130. sha1 := h.serverSHA1.Sum()
  131. return finishedSum(md5, sha1, serverFinishedLabel, masterSecret)
  132. }