mirror of
https://github.com/refraction-networking/utls.git
synced 2025-04-03 20:17:36 +03:00
This refactors a lot of the certificate support logic to make it cleaner and reusable where possible. These changes will make the following CLs much simpler. In particular, the heavily overloaded pickSignatureAlgorithm is gone. That function used to cover both signing and verifying side, would work both for pre-signature_algorithms TLS 1.0/1.1 and TLS 1.2, and returned sigalg, type and hash. Now, TLS 1.0/1.1 and 1.2 are differentiated at the caller, as they have effectively completely different logic. TLS 1.0/1.1 simply use legacyTypeAndHashFromPublicKey as they employ a fixed hash function and signature algorithm for each public key type. TLS 1.2 is instead routed through selectSignatureScheme (on the signing side) or isSupportedSignatureAlgorithm (on the verifying side) and typeAndHashFromSignatureScheme, like TLS 1.3. On the signing side, signatureSchemesForCertificate was already version aware (for PKCS#1 v1.5 vs PSS support), so selectSignatureScheme just had to learn the Section 7.4.1.4.1 defaults for a missing signature_algorithms to replace pickSignatureAlgorithm. On the verifying side, pickSignatureAlgorithm was also checking the public key type, while isSupportedSignatureAlgorithm + typeAndHashFromSignatureScheme are not, but that check was redundant with the one in verifyHandshakeSignature. There should be no major change in behavior so far. A few minor changes came from the refactor: we now correctly require signature_algorithms in TLS 1.3 when using a certificate; we won't use Ed25519 in TLS 1.2 if the client didn't send signature_algorithms; and we don't send ec_points_format in the ServerHello (a compatibility measure) if we are not doing ECDHE anyway because there are no mutually supported curves. The tests also got simpler because they test simpler functions. The caller logic switching between TLS 1.0/1.1 and 1.2 is tested by the transcript tests. Updates #32426 Change-Id: Ice9dcaea78d204718f661f8d60efdb408ba41577 Reviewed-on: https://go-review.googlesource.com/c/go/+/205061 Reviewed-by: Katie Hockman <katie@golang.org>
270 lines
8.7 KiB
Go
270 lines
8.7 KiB
Go
// Copyright 2017 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package tls
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto"
|
|
"crypto/ecdsa"
|
|
"crypto/ed25519"
|
|
"crypto/elliptic"
|
|
"crypto/rsa"
|
|
"encoding/asn1"
|
|
"errors"
|
|
"fmt"
|
|
"hash"
|
|
"io"
|
|
)
|
|
|
|
// verifyHandshakeSignature verifies a signature against pre-hashed
|
|
// (if required) handshake contents.
|
|
func verifyHandshakeSignature(sigType uint8, pubkey crypto.PublicKey, hashFunc crypto.Hash, signed, sig []byte) error {
|
|
switch sigType {
|
|
case signatureECDSA:
|
|
pubKey, ok := pubkey.(*ecdsa.PublicKey)
|
|
if !ok {
|
|
return fmt.Errorf("expected an ECDSA public key, got %T", pubkey)
|
|
}
|
|
ecdsaSig := new(ecdsaSignature)
|
|
if _, err := asn1.Unmarshal(sig, ecdsaSig); err != nil {
|
|
return err
|
|
}
|
|
if ecdsaSig.R.Sign() <= 0 || ecdsaSig.S.Sign() <= 0 {
|
|
return errors.New("ECDSA signature contained zero or negative values")
|
|
}
|
|
if !ecdsa.Verify(pubKey, signed, ecdsaSig.R, ecdsaSig.S) {
|
|
return errors.New("ECDSA verification failure")
|
|
}
|
|
case signatureEd25519:
|
|
pubKey, ok := pubkey.(ed25519.PublicKey)
|
|
if !ok {
|
|
return fmt.Errorf("expected an Ed25519 public key, got %T", pubkey)
|
|
}
|
|
if !ed25519.Verify(pubKey, signed, sig) {
|
|
return errors.New("Ed25519 verification failure")
|
|
}
|
|
case signaturePKCS1v15:
|
|
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
if !ok {
|
|
return fmt.Errorf("expected an RSA public key, got %T", pubkey)
|
|
}
|
|
if err := rsa.VerifyPKCS1v15(pubKey, hashFunc, signed, sig); err != nil {
|
|
return err
|
|
}
|
|
case signatureRSAPSS:
|
|
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
if !ok {
|
|
return fmt.Errorf("expected an RSA public key, got %T", pubkey)
|
|
}
|
|
signOpts := &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash}
|
|
if err := rsa.VerifyPSS(pubKey, hashFunc, signed, sig, signOpts); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
return errors.New("internal error: unknown signature type")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
const (
|
|
serverSignatureContext = "TLS 1.3, server CertificateVerify\x00"
|
|
clientSignatureContext = "TLS 1.3, client CertificateVerify\x00"
|
|
)
|
|
|
|
var signaturePadding = []byte{
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
|
|
}
|
|
|
|
// signedMessage returns the pre-hashed (if necessary) message to be signed by
|
|
// certificate keys in TLS 1.3. See RFC 8446, Section 4.4.3.
|
|
func signedMessage(sigHash crypto.Hash, context string, transcript hash.Hash) []byte {
|
|
if sigHash == directSigning {
|
|
b := &bytes.Buffer{}
|
|
b.Write(signaturePadding)
|
|
io.WriteString(b, context)
|
|
b.Write(transcript.Sum(nil))
|
|
return b.Bytes()
|
|
}
|
|
h := sigHash.New()
|
|
h.Write(signaturePadding)
|
|
io.WriteString(h, context)
|
|
h.Write(transcript.Sum(nil))
|
|
return h.Sum(nil)
|
|
}
|
|
|
|
// typeAndHashFromSignatureScheme returns the corresponding signature type and
|
|
// crypto.Hash for a given TLS SignatureScheme.
|
|
func typeAndHashFromSignatureScheme(signatureAlgorithm SignatureScheme) (sigType uint8, hash crypto.Hash, err error) {
|
|
switch signatureAlgorithm {
|
|
case PKCS1WithSHA1, PKCS1WithSHA256, PKCS1WithSHA384, PKCS1WithSHA512:
|
|
sigType = signaturePKCS1v15
|
|
case PSSWithSHA256, PSSWithSHA384, PSSWithSHA512:
|
|
sigType = signatureRSAPSS
|
|
case ECDSAWithSHA1, ECDSAWithP256AndSHA256, ECDSAWithP384AndSHA384, ECDSAWithP521AndSHA512:
|
|
sigType = signatureECDSA
|
|
case Ed25519:
|
|
sigType = signatureEd25519
|
|
default:
|
|
return 0, 0, fmt.Errorf("unsupported signature algorithm: %#04x", signatureAlgorithm)
|
|
}
|
|
switch signatureAlgorithm {
|
|
case PKCS1WithSHA1, ECDSAWithSHA1:
|
|
hash = crypto.SHA1
|
|
case PKCS1WithSHA256, PSSWithSHA256, ECDSAWithP256AndSHA256:
|
|
hash = crypto.SHA256
|
|
case PKCS1WithSHA384, PSSWithSHA384, ECDSAWithP384AndSHA384:
|
|
hash = crypto.SHA384
|
|
case PKCS1WithSHA512, PSSWithSHA512, ECDSAWithP521AndSHA512:
|
|
hash = crypto.SHA512
|
|
case Ed25519:
|
|
hash = directSigning
|
|
default:
|
|
return 0, 0, fmt.Errorf("unsupported signature algorithm: %#04x", signatureAlgorithm)
|
|
}
|
|
return sigType, hash, nil
|
|
}
|
|
|
|
// legacyTypeAndHashFromPublicKey returns the fixed signature type and crypto.Hash for
|
|
// a given public key used with TLS 1.0 and 1.1, before the introduction of
|
|
// signature algorithm negotiation.
|
|
func legacyTypeAndHashFromPublicKey(pub crypto.PublicKey) (sigType uint8, hash crypto.Hash, err error) {
|
|
switch pub.(type) {
|
|
case *rsa.PublicKey:
|
|
return signaturePKCS1v15, crypto.MD5SHA1, nil
|
|
case *ecdsa.PublicKey:
|
|
return signatureECDSA, crypto.SHA1, nil
|
|
case ed25519.PublicKey:
|
|
// RFC 8422 specifies support for Ed25519 in TLS 1.0 and 1.1,
|
|
// but it requires holding on to a handshake transcript to do a
|
|
// full signature, and not even OpenSSL bothers with the
|
|
// complexity, so we can't even test it properly.
|
|
return 0, 0, fmt.Errorf("tls: Ed25519 public keys are not supported before TLS 1.2")
|
|
default:
|
|
return 0, 0, fmt.Errorf("tls: unsupported public key: %T", pub)
|
|
}
|
|
}
|
|
|
|
// signatureSchemesForCertificate returns the list of supported SignatureSchemes
|
|
// for a given certificate, based on the public key and the protocol version.
|
|
//
|
|
// This function must be kept in sync with supportedSignatureAlgorithms.
|
|
func signatureSchemesForCertificate(version uint16, cert *Certificate) []SignatureScheme {
|
|
priv, ok := cert.PrivateKey.(crypto.Signer)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
|
|
switch pub := priv.Public().(type) {
|
|
case *ecdsa.PublicKey:
|
|
if version != VersionTLS13 {
|
|
// In TLS 1.2 and earlier, ECDSA algorithms are not
|
|
// constrained to a single curve.
|
|
return []SignatureScheme{
|
|
ECDSAWithP256AndSHA256,
|
|
ECDSAWithP384AndSHA384,
|
|
ECDSAWithP521AndSHA512,
|
|
ECDSAWithSHA1,
|
|
}
|
|
}
|
|
switch pub.Curve {
|
|
case elliptic.P256():
|
|
return []SignatureScheme{ECDSAWithP256AndSHA256}
|
|
case elliptic.P384():
|
|
return []SignatureScheme{ECDSAWithP384AndSHA384}
|
|
case elliptic.P521():
|
|
return []SignatureScheme{ECDSAWithP521AndSHA512}
|
|
default:
|
|
return nil
|
|
}
|
|
case *rsa.PublicKey:
|
|
if version != VersionTLS13 {
|
|
return []SignatureScheme{
|
|
// Temporarily disable RSA-PSS in TLS 1.2, see Issue 32425.
|
|
// PSSWithSHA256,
|
|
// PSSWithSHA384,
|
|
// PSSWithSHA512,
|
|
PKCS1WithSHA256,
|
|
PKCS1WithSHA384,
|
|
PKCS1WithSHA512,
|
|
PKCS1WithSHA1,
|
|
}
|
|
}
|
|
// TLS 1.3 dropped support for PKCS#1 v1.5 in favor of RSA-PSS.
|
|
return []SignatureScheme{
|
|
PSSWithSHA256,
|
|
PSSWithSHA384,
|
|
PSSWithSHA512,
|
|
}
|
|
case ed25519.PublicKey:
|
|
return []SignatureScheme{Ed25519}
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// selectSignatureScheme picks a SignatureScheme from the peer's preference list
|
|
// that works with the selected certificate. It's only called for protocol
|
|
// versions that support signature algorithms, so TLS 1.2 and 1.3.
|
|
func selectSignatureScheme(vers uint16, c *Certificate, peerAlgs []SignatureScheme) (SignatureScheme, error) {
|
|
supportedAlgs := signatureSchemesForCertificate(vers, c)
|
|
if supportedAlgs == nil {
|
|
return 0, unsupportedCertificateError(c)
|
|
}
|
|
if len(peerAlgs) == 0 && vers == VersionTLS12 {
|
|
// For TLS 1.2, if the client didn't send signature_algorithms then we
|
|
// can assume that it supports SHA1. See RFC 5246, Section 7.4.1.4.1.
|
|
peerAlgs = []SignatureScheme{PKCS1WithSHA1, ECDSAWithSHA1}
|
|
}
|
|
// Pick signature scheme in the peer's preference order, as our
|
|
// preference order is not configurable.
|
|
for _, preferredAlg := range peerAlgs {
|
|
if isSupportedSignatureAlgorithm(preferredAlg, supportedAlgs) {
|
|
return preferredAlg, nil
|
|
}
|
|
}
|
|
return 0, errors.New("tls: peer doesn't support any of the certificate's signature algorithms")
|
|
}
|
|
|
|
// unsupportedCertificateError returns a helpful error for certificates with
|
|
// an unsupported private key.
|
|
func unsupportedCertificateError(cert *Certificate) error {
|
|
switch cert.PrivateKey.(type) {
|
|
case rsa.PrivateKey, ecdsa.PrivateKey:
|
|
return fmt.Errorf("tls: unsupported certificate: private key is %T, expected *%T",
|
|
cert.PrivateKey, cert.PrivateKey)
|
|
case *ed25519.PrivateKey:
|
|
return fmt.Errorf("tls: unsupported certificate: private key is *ed25519.PrivateKey, expected ed25519.PrivateKey")
|
|
}
|
|
|
|
signer, ok := cert.PrivateKey.(crypto.Signer)
|
|
if !ok {
|
|
return fmt.Errorf("tls: certificate private key (%T) does not implement crypto.Signer",
|
|
cert.PrivateKey)
|
|
}
|
|
|
|
switch pub := signer.Public().(type) {
|
|
case *ecdsa.PublicKey:
|
|
switch pub.Curve {
|
|
case elliptic.P256():
|
|
case elliptic.P384():
|
|
case elliptic.P521():
|
|
default:
|
|
return fmt.Errorf("tls: unsupported certificate curve (%s)", pub.Curve.Params().Name)
|
|
}
|
|
case *rsa.PublicKey:
|
|
case ed25519.PublicKey:
|
|
default:
|
|
return fmt.Errorf("tls: unsupported certificate key (%T)", pub)
|
|
}
|
|
|
|
return fmt.Errorf("tls: internal error: unsupported key (%T)", cert.PrivateKey)
|
|
}
|