cipher_suite.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
  2. // SPDX-License-Identifier: MIT
  3. package dtls
  4. import (
  5. "crypto/ecdsa"
  6. "crypto/ed25519"
  7. "crypto/rsa"
  8. "crypto/tls"
  9. "fmt"
  10. "hash"
  11. "github.com/pion/dtls/v2/internal/ciphersuite"
  12. "github.com/pion/dtls/v2/pkg/crypto/clientcertificate"
  13. "github.com/pion/dtls/v2/pkg/protocol/recordlayer"
  14. )
  15. // CipherSuiteID is an ID for our supported CipherSuites
  16. type CipherSuiteID = ciphersuite.ID
  17. // Supported Cipher Suites
  18. const (
  19. // AES-128-CCM
  20. TLS_ECDHE_ECDSA_WITH_AES_128_CCM CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_CCM //nolint:revive,stylecheck
  21. TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8 CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8 //nolint:revive,stylecheck
  22. // AES-128-GCM-SHA256
  23. TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 //nolint:revive,stylecheck
  24. TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 //nolint:revive,stylecheck
  25. TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 //nolint:revive,stylecheck
  26. TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 CipherSuiteID = ciphersuite.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 //nolint:revive,stylecheck
  27. // AES-256-CBC-SHA
  28. TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA //nolint:revive,stylecheck
  29. TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA CipherSuiteID = ciphersuite.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA //nolint:revive,stylecheck
  30. TLS_PSK_WITH_AES_128_CCM CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CCM //nolint:revive,stylecheck
  31. TLS_PSK_WITH_AES_128_CCM_8 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CCM_8 //nolint:revive,stylecheck
  32. TLS_PSK_WITH_AES_256_CCM_8 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_256_CCM_8 //nolint:revive,stylecheck
  33. TLS_PSK_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_GCM_SHA256 //nolint:revive,stylecheck
  34. TLS_PSK_WITH_AES_128_CBC_SHA256 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CBC_SHA256 //nolint:revive,stylecheck
  35. TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 CipherSuiteID = ciphersuite.TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 //nolint:revive,stylecheck
  36. )
  37. // CipherSuiteAuthenticationType controls what authentication method is using during the handshake for a CipherSuite
  38. type CipherSuiteAuthenticationType = ciphersuite.AuthenticationType
  39. // AuthenticationType Enums
  40. const (
  41. CipherSuiteAuthenticationTypeCertificate CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypeCertificate
  42. CipherSuiteAuthenticationTypePreSharedKey CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypePreSharedKey
  43. CipherSuiteAuthenticationTypeAnonymous CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypeAnonymous
  44. )
  45. // CipherSuiteKeyExchangeAlgorithm controls what exchange algorithm is using during the handshake for a CipherSuite
  46. type CipherSuiteKeyExchangeAlgorithm = ciphersuite.KeyExchangeAlgorithm
  47. // CipherSuiteKeyExchangeAlgorithm Bitmask
  48. const (
  49. CipherSuiteKeyExchangeAlgorithmNone CipherSuiteKeyExchangeAlgorithm = ciphersuite.KeyExchangeAlgorithmNone
  50. CipherSuiteKeyExchangeAlgorithmPsk CipherSuiteKeyExchangeAlgorithm = ciphersuite.KeyExchangeAlgorithmPsk
  51. CipherSuiteKeyExchangeAlgorithmEcdhe CipherSuiteKeyExchangeAlgorithm = ciphersuite.KeyExchangeAlgorithmEcdhe
  52. )
  53. var _ = allCipherSuites() // Necessary until this function isn't only used by Go 1.14
  54. // CipherSuite is an interface that all DTLS CipherSuites must satisfy
  55. type CipherSuite interface {
  56. // String of CipherSuite, only used for logging
  57. String() string
  58. // ID of CipherSuite.
  59. ID() CipherSuiteID
  60. // What type of Certificate does this CipherSuite use
  61. CertificateType() clientcertificate.Type
  62. // What Hash function is used during verification
  63. HashFunc() func() hash.Hash
  64. // AuthenticationType controls what authentication method is using during the handshake
  65. AuthenticationType() CipherSuiteAuthenticationType
  66. // KeyExchangeAlgorithm controls what exchange algorithm is using during the handshake
  67. KeyExchangeAlgorithm() CipherSuiteKeyExchangeAlgorithm
  68. // ECC (Elliptic Curve Cryptography) determines whether ECC extesions will be send during handshake.
  69. // https://datatracker.ietf.org/doc/html/rfc4492#page-10
  70. ECC() bool
  71. // Called when keying material has been generated, should initialize the internal cipher
  72. Init(masterSecret, clientRandom, serverRandom []byte, isClient bool) error
  73. IsInitialized() bool
  74. Encrypt(pkt *recordlayer.RecordLayer, raw []byte) ([]byte, error)
  75. Decrypt(in []byte) ([]byte, error)
  76. }
  77. // CipherSuiteName provides the same functionality as tls.CipherSuiteName
  78. // that appeared first in Go 1.14.
  79. //
  80. // Our implementation differs slightly in that it takes in a CiperSuiteID,
  81. // like the rest of our library, instead of a uint16 like crypto/tls.
  82. func CipherSuiteName(id CipherSuiteID) string {
  83. suite := cipherSuiteForID(id, nil)
  84. if suite != nil {
  85. return suite.String()
  86. }
  87. return fmt.Sprintf("0x%04X", uint16(id))
  88. }
  89. // Taken from https://www.iana.org/assignments/tls-parameters/tls-parameters.xml
  90. // A cipherSuite is a specific combination of key agreement, cipher and MAC
  91. // function.
  92. func cipherSuiteForID(id CipherSuiteID, customCiphers func() []CipherSuite) CipherSuite {
  93. switch id { //nolint:exhaustive
  94. case TLS_ECDHE_ECDSA_WITH_AES_128_CCM:
  95. return ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm()
  96. case TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8:
  97. return ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm8()
  98. case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256:
  99. return &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{}
  100. case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256:
  101. return &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{}
  102. case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA:
  103. return &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{}
  104. case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA:
  105. return &ciphersuite.TLSEcdheRsaWithAes256CbcSha{}
  106. case TLS_PSK_WITH_AES_128_CCM:
  107. return ciphersuite.NewTLSPskWithAes128Ccm()
  108. case TLS_PSK_WITH_AES_128_CCM_8:
  109. return ciphersuite.NewTLSPskWithAes128Ccm8()
  110. case TLS_PSK_WITH_AES_256_CCM_8:
  111. return ciphersuite.NewTLSPskWithAes256Ccm8()
  112. case TLS_PSK_WITH_AES_128_GCM_SHA256:
  113. return &ciphersuite.TLSPskWithAes128GcmSha256{}
  114. case TLS_PSK_WITH_AES_128_CBC_SHA256:
  115. return &ciphersuite.TLSPskWithAes128CbcSha256{}
  116. case TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384:
  117. return &ciphersuite.TLSEcdheEcdsaWithAes256GcmSha384{}
  118. case TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384:
  119. return &ciphersuite.TLSEcdheRsaWithAes256GcmSha384{}
  120. case TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256:
  121. return ciphersuite.NewTLSEcdhePskWithAes128CbcSha256()
  122. }
  123. if customCiphers != nil {
  124. for _, c := range customCiphers() {
  125. if c.ID() == id {
  126. return c
  127. }
  128. }
  129. }
  130. return nil
  131. }
  132. // CipherSuites we support in order of preference
  133. func defaultCipherSuites() []CipherSuite {
  134. return []CipherSuite{
  135. &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{},
  136. &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{},
  137. &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{},
  138. &ciphersuite.TLSEcdheRsaWithAes256CbcSha{},
  139. &ciphersuite.TLSEcdheEcdsaWithAes256GcmSha384{},
  140. &ciphersuite.TLSEcdheRsaWithAes256GcmSha384{},
  141. }
  142. }
  143. func allCipherSuites() []CipherSuite {
  144. return []CipherSuite{
  145. ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm(),
  146. ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm8(),
  147. &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{},
  148. &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{},
  149. &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{},
  150. &ciphersuite.TLSEcdheRsaWithAes256CbcSha{},
  151. ciphersuite.NewTLSPskWithAes128Ccm(),
  152. ciphersuite.NewTLSPskWithAes128Ccm8(),
  153. ciphersuite.NewTLSPskWithAes256Ccm8(),
  154. &ciphersuite.TLSPskWithAes128GcmSha256{},
  155. &ciphersuite.TLSEcdheEcdsaWithAes256GcmSha384{},
  156. &ciphersuite.TLSEcdheRsaWithAes256GcmSha384{},
  157. }
  158. }
  159. func cipherSuiteIDs(cipherSuites []CipherSuite) []uint16 {
  160. rtrn := []uint16{}
  161. for _, c := range cipherSuites {
  162. rtrn = append(rtrn, uint16(c.ID()))
  163. }
  164. return rtrn
  165. }
  166. func parseCipherSuites(userSelectedSuites []CipherSuiteID, customCipherSuites func() []CipherSuite, includeCertificateSuites, includePSKSuites bool) ([]CipherSuite, error) {
  167. cipherSuitesForIDs := func(ids []CipherSuiteID) ([]CipherSuite, error) {
  168. cipherSuites := []CipherSuite{}
  169. for _, id := range ids {
  170. c := cipherSuiteForID(id, nil)
  171. if c == nil {
  172. return nil, &invalidCipherSuiteError{id}
  173. }
  174. cipherSuites = append(cipherSuites, c)
  175. }
  176. return cipherSuites, nil
  177. }
  178. var (
  179. cipherSuites []CipherSuite
  180. err error
  181. i int
  182. )
  183. if userSelectedSuites != nil {
  184. cipherSuites, err = cipherSuitesForIDs(userSelectedSuites)
  185. if err != nil {
  186. return nil, err
  187. }
  188. } else {
  189. cipherSuites = defaultCipherSuites()
  190. }
  191. // Put CustomCipherSuites before ID selected suites
  192. if customCipherSuites != nil {
  193. cipherSuites = append(customCipherSuites(), cipherSuites...)
  194. }
  195. var foundCertificateSuite, foundPSKSuite, foundAnonymousSuite bool
  196. for _, c := range cipherSuites {
  197. switch {
  198. case includeCertificateSuites && c.AuthenticationType() == CipherSuiteAuthenticationTypeCertificate:
  199. foundCertificateSuite = true
  200. case includePSKSuites && c.AuthenticationType() == CipherSuiteAuthenticationTypePreSharedKey:
  201. foundPSKSuite = true
  202. case c.AuthenticationType() == CipherSuiteAuthenticationTypeAnonymous:
  203. foundAnonymousSuite = true
  204. default:
  205. continue
  206. }
  207. cipherSuites[i] = c
  208. i++
  209. }
  210. switch {
  211. case includeCertificateSuites && !foundCertificateSuite && !foundAnonymousSuite:
  212. return nil, errNoAvailableCertificateCipherSuite
  213. case includePSKSuites && !foundPSKSuite:
  214. return nil, errNoAvailablePSKCipherSuite
  215. case i == 0:
  216. return nil, errNoAvailableCipherSuites
  217. }
  218. return cipherSuites[:i], nil
  219. }
  220. func filterCipherSuitesForCertificate(cert *tls.Certificate, cipherSuites []CipherSuite) []CipherSuite {
  221. if cert == nil || cert.PrivateKey == nil {
  222. return cipherSuites
  223. }
  224. var certType clientcertificate.Type
  225. switch cert.PrivateKey.(type) {
  226. case ed25519.PrivateKey, *ecdsa.PrivateKey:
  227. certType = clientcertificate.ECDSASign
  228. case *rsa.PrivateKey:
  229. certType = clientcertificate.RSASign
  230. }
  231. filtered := []CipherSuite{}
  232. for _, c := range cipherSuites {
  233. if c.AuthenticationType() != CipherSuiteAuthenticationTypeCertificate || certType == c.CertificateType() {
  234. filtered = append(filtered, c)
  235. }
  236. }
  237. return filtered
  238. }