mirror of
https://github.com/refraction-networking/utls.git
synced 2025-04-03 03:57:36 +03:00
Implement a basic TLS 1.3 server handshake, only enabled if explicitly requested with MaxVersion. This CL intentionally leaves for future CLs: - PSK modes and resumption - client authentication - compatibility mode ChangeCipherSpecs - early data skipping - post-handshake messages - downgrade protection - KeyLogWriter support - TLS_FALLBACK_SCSV processing It also leaves a few areas up for a wider refactor (maybe in Go 1.13): - the certificate selection logic can be significantly improved, including supporting and surfacing signature_algorithms_cert, but this isn't new in TLS 1.3 (see comment in processClientHello) - handshake_server_tls13.go can be dried up and broken into more meaningful, smaller functions, but it felt premature to do before PSK and client auth support - the monstrous ClientHello equality check in doHelloRetryRequest can get both cleaner and more complete with collaboration from the parsing layer, which can come at the same time as extension duplicates detection Updates #9671 Change-Id: Id9db2b6ecc2eea21bf9b59b6d1d9c84a7435151c Reviewed-on: https://go-review.googlesource.com/c/147017 Run-TryBot: Filippo Valsorda <filippo@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Adam Langley <agl@golang.org>
166 lines
5.3 KiB
Go
166 lines
5.3 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 (
|
|
"crypto"
|
|
"crypto/ecdsa"
|
|
"crypto/elliptic"
|
|
"crypto/rsa"
|
|
"encoding/asn1"
|
|
"errors"
|
|
"fmt"
|
|
"hash"
|
|
"io"
|
|
)
|
|
|
|
// pickSignatureAlgorithm selects a signature algorithm that is compatible with
|
|
// the given public key and the list of algorithms from the peer and this side.
|
|
// The lists of signature algorithms (peerSigAlgs and ourSigAlgs) are ignored
|
|
// for tlsVersion < VersionTLS12.
|
|
//
|
|
// The returned SignatureScheme codepoint is only meaningful for TLS 1.2,
|
|
// previous TLS versions have a fixed hash function.
|
|
func pickSignatureAlgorithm(pubkey crypto.PublicKey, peerSigAlgs, ourSigAlgs []SignatureScheme, tlsVersion uint16) (sigAlg SignatureScheme, sigType uint8, hashFunc crypto.Hash, err error) {
|
|
if tlsVersion < VersionTLS12 || len(peerSigAlgs) == 0 {
|
|
// For TLS 1.1 and before, the signature algorithm could not be
|
|
// negotiated and the hash is fixed based on the signature type. For TLS
|
|
// 1.2, if the client didn't send signature_algorithms extension then we
|
|
// can assume that it supports SHA1. See RFC 5246, Section 7.4.1.4.1.
|
|
switch pubkey.(type) {
|
|
case *rsa.PublicKey:
|
|
if tlsVersion < VersionTLS12 {
|
|
return 0, signaturePKCS1v15, crypto.MD5SHA1, nil
|
|
} else {
|
|
return PKCS1WithSHA1, signaturePKCS1v15, crypto.SHA1, nil
|
|
}
|
|
case *ecdsa.PublicKey:
|
|
return ECDSAWithSHA1, signatureECDSA, crypto.SHA1, nil
|
|
default:
|
|
return 0, 0, 0, fmt.Errorf("tls: unsupported public key: %T", pubkey)
|
|
}
|
|
}
|
|
for _, sigAlg := range peerSigAlgs {
|
|
if !isSupportedSignatureAlgorithm(sigAlg, ourSigAlgs) {
|
|
continue
|
|
}
|
|
hashAlg, err := hashFromSignatureScheme(sigAlg)
|
|
if err != nil {
|
|
panic("tls: supported signature algorithm has an unknown hash function")
|
|
}
|
|
sigType := signatureFromSignatureScheme(sigAlg)
|
|
switch pubkey.(type) {
|
|
case *rsa.PublicKey:
|
|
if sigType == signaturePKCS1v15 || sigType == signatureRSAPSS {
|
|
return sigAlg, sigType, hashAlg, nil
|
|
}
|
|
case *ecdsa.PublicKey:
|
|
if sigType == signatureECDSA {
|
|
return sigAlg, sigType, hashAlg, nil
|
|
}
|
|
default:
|
|
return 0, 0, 0, fmt.Errorf("tls: unsupported public key: %T", pubkey)
|
|
}
|
|
}
|
|
return 0, 0, 0, errors.New("tls: peer doesn't support any common signature algorithms")
|
|
}
|
|
|
|
// verifyHandshakeSignature verifies a signature against pre-hashed handshake
|
|
// contents.
|
|
func verifyHandshakeSignature(sigType uint8, pubkey crypto.PublicKey, hashFunc crypto.Hash, digest, sig []byte) error {
|
|
switch sigType {
|
|
case signatureECDSA:
|
|
pubKey, ok := pubkey.(*ecdsa.PublicKey)
|
|
if !ok {
|
|
return errors.New("tls: ECDSA signing requires a ECDSA public key")
|
|
}
|
|
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("tls: ECDSA signature contained zero or negative values")
|
|
}
|
|
if !ecdsa.Verify(pubKey, digest, ecdsaSig.R, ecdsaSig.S) {
|
|
return errors.New("tls: ECDSA verification failure")
|
|
}
|
|
case signaturePKCS1v15:
|
|
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
if !ok {
|
|
return errors.New("tls: RSA signing requires a RSA public key")
|
|
}
|
|
if err := rsa.VerifyPKCS1v15(pubKey, hashFunc, digest, sig); err != nil {
|
|
return err
|
|
}
|
|
case signatureRSAPSS:
|
|
pubKey, ok := pubkey.(*rsa.PublicKey)
|
|
if !ok {
|
|
return errors.New("tls: RSA signing requires a RSA public key")
|
|
}
|
|
signOpts := &rsa.PSSOptions{SaltLength: rsa.PSSSaltLengthEqualsHash}
|
|
if err := rsa.VerifyPSS(pubKey, hashFunc, digest, sig, signOpts); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
return errors.New("tls: unknown signature algorithm")
|
|
}
|
|
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,
|
|
}
|
|
|
|
// writeSignedMessage writes the content to be signed by certificate keys in TLS
|
|
// 1.3 to sigHash. See RFC 8446, Section 4.4.3.
|
|
func writeSignedMessage(sigHash io.Writer, context string, transcript hash.Hash) {
|
|
sigHash.Write(signaturePadding)
|
|
io.WriteString(sigHash, context)
|
|
sigHash.Write(transcript.Sum(nil))
|
|
}
|
|
|
|
// signatureSchemesForCertificate returns the list of supported SignatureSchemes
|
|
// for a given certificate, based on the public key.
|
|
func signatureSchemesForCertificate(cert *Certificate) []SignatureScheme {
|
|
priv, ok := cert.PrivateKey.(crypto.Signer)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
|
|
switch priv := priv.Public().(type) {
|
|
case *ecdsa.PublicKey:
|
|
switch priv.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:
|
|
// RSA keys with RSA-PSS OID are not supported by crypto/x509.
|
|
return []SignatureScheme{
|
|
PSSWithSHA256,
|
|
PSSWithSHA384,
|
|
PSSWithSHA512,
|
|
}
|
|
default:
|
|
return nil
|
|
}
|
|
}
|