1*6777b538SAndroid Build Coastguard Worker // Copyright 2014 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker
5*6777b538SAndroid Build Coastguard Worker #include "net/ssl/openssl_ssl_util.h"
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker #include <errno.h>
8*6777b538SAndroid Build Coastguard Worker
9*6777b538SAndroid Build Coastguard Worker #include <utility>
10*6777b538SAndroid Build Coastguard Worker
11*6777b538SAndroid Build Coastguard Worker #include "base/lazy_instance.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/location.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
14*6777b538SAndroid Build Coastguard Worker #include "base/notreached.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/values.h"
16*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
17*6777b538SAndroid Build Coastguard Worker #include "crypto/openssl_util.h"
18*6777b538SAndroid Build Coastguard Worker #include "net/base/net_errors.h"
19*6777b538SAndroid Build Coastguard Worker #include "net/cert/x509_util.h"
20*6777b538SAndroid Build Coastguard Worker #include "net/log/net_log_with_source.h"
21*6777b538SAndroid Build Coastguard Worker #include "net/ssl/ssl_connection_status_flags.h"
22*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/include/openssl/err.h"
23*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/include/openssl/ssl.h"
24*6777b538SAndroid Build Coastguard Worker
25*6777b538SAndroid Build Coastguard Worker namespace net {
26*6777b538SAndroid Build Coastguard Worker
27*6777b538SAndroid Build Coastguard Worker SslSetClearMask::SslSetClearMask() = default;
28*6777b538SAndroid Build Coastguard Worker
ConfigureFlag(long flag,bool state)29*6777b538SAndroid Build Coastguard Worker void SslSetClearMask::ConfigureFlag(long flag, bool state) {
30*6777b538SAndroid Build Coastguard Worker (state ? set_mask : clear_mask) |= flag;
31*6777b538SAndroid Build Coastguard Worker // Make sure we haven't got any intersection in the set & clear options.
32*6777b538SAndroid Build Coastguard Worker DCHECK_EQ(0, set_mask & clear_mask) << flag << ":" << state;
33*6777b538SAndroid Build Coastguard Worker }
34*6777b538SAndroid Build Coastguard Worker
35*6777b538SAndroid Build Coastguard Worker namespace {
36*6777b538SAndroid Build Coastguard Worker
37*6777b538SAndroid Build Coastguard Worker class OpenSSLNetErrorLibSingleton {
38*6777b538SAndroid Build Coastguard Worker public:
OpenSSLNetErrorLibSingleton()39*6777b538SAndroid Build Coastguard Worker OpenSSLNetErrorLibSingleton() {
40*6777b538SAndroid Build Coastguard Worker crypto::EnsureOpenSSLInit();
41*6777b538SAndroid Build Coastguard Worker
42*6777b538SAndroid Build Coastguard Worker // Allocate a new error library value for inserting net errors into
43*6777b538SAndroid Build Coastguard Worker // OpenSSL. This does not register any ERR_STRING_DATA for the errors, so
44*6777b538SAndroid Build Coastguard Worker // stringifying error codes through OpenSSL will return NULL.
45*6777b538SAndroid Build Coastguard Worker net_error_lib_ = ERR_get_next_error_library();
46*6777b538SAndroid Build Coastguard Worker }
47*6777b538SAndroid Build Coastguard Worker
net_error_lib() const48*6777b538SAndroid Build Coastguard Worker int net_error_lib() const { return net_error_lib_; }
49*6777b538SAndroid Build Coastguard Worker
50*6777b538SAndroid Build Coastguard Worker private:
51*6777b538SAndroid Build Coastguard Worker int net_error_lib_;
52*6777b538SAndroid Build Coastguard Worker };
53*6777b538SAndroid Build Coastguard Worker
54*6777b538SAndroid Build Coastguard Worker base::LazyInstance<OpenSSLNetErrorLibSingleton>::Leaky g_openssl_net_error_lib =
55*6777b538SAndroid Build Coastguard Worker LAZY_INSTANCE_INITIALIZER;
56*6777b538SAndroid Build Coastguard Worker
OpenSSLNetErrorLib()57*6777b538SAndroid Build Coastguard Worker int OpenSSLNetErrorLib() {
58*6777b538SAndroid Build Coastguard Worker return g_openssl_net_error_lib.Get().net_error_lib();
59*6777b538SAndroid Build Coastguard Worker }
60*6777b538SAndroid Build Coastguard Worker
MapOpenSSLErrorSSL(uint32_t error_code)61*6777b538SAndroid Build Coastguard Worker int MapOpenSSLErrorSSL(uint32_t error_code) {
62*6777b538SAndroid Build Coastguard Worker DCHECK_EQ(ERR_LIB_SSL, ERR_GET_LIB(error_code));
63*6777b538SAndroid Build Coastguard Worker
64*6777b538SAndroid Build Coastguard Worker #if DCHECK_IS_ON()
65*6777b538SAndroid Build Coastguard Worker char buf[ERR_ERROR_STRING_BUF_LEN];
66*6777b538SAndroid Build Coastguard Worker ERR_error_string_n(error_code, buf, sizeof(buf));
67*6777b538SAndroid Build Coastguard Worker DVLOG(1) << "OpenSSL SSL error, reason: " << ERR_GET_REASON(error_code)
68*6777b538SAndroid Build Coastguard Worker << ", name: " << buf;
69*6777b538SAndroid Build Coastguard Worker #endif
70*6777b538SAndroid Build Coastguard Worker
71*6777b538SAndroid Build Coastguard Worker switch (ERR_GET_REASON(error_code)) {
72*6777b538SAndroid Build Coastguard Worker case SSL_R_READ_TIMEOUT_EXPIRED:
73*6777b538SAndroid Build Coastguard Worker return ERR_TIMED_OUT;
74*6777b538SAndroid Build Coastguard Worker case SSL_R_UNKNOWN_CERTIFICATE_TYPE:
75*6777b538SAndroid Build Coastguard Worker case SSL_R_UNKNOWN_CIPHER_TYPE:
76*6777b538SAndroid Build Coastguard Worker case SSL_R_UNKNOWN_KEY_EXCHANGE_TYPE:
77*6777b538SAndroid Build Coastguard Worker case SSL_R_UNKNOWN_SSL_VERSION:
78*6777b538SAndroid Build Coastguard Worker return ERR_NOT_IMPLEMENTED;
79*6777b538SAndroid Build Coastguard Worker case SSL_R_NO_CIPHER_MATCH:
80*6777b538SAndroid Build Coastguard Worker case SSL_R_NO_SHARED_CIPHER:
81*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_INSUFFICIENT_SECURITY:
82*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_PROTOCOL_VERSION:
83*6777b538SAndroid Build Coastguard Worker case SSL_R_UNSUPPORTED_PROTOCOL:
84*6777b538SAndroid Build Coastguard Worker return ERR_SSL_VERSION_OR_CIPHER_MISMATCH;
85*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_BAD_CERTIFICATE:
86*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_UNSUPPORTED_CERTIFICATE:
87*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_CERTIFICATE_REVOKED:
88*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_CERTIFICATE_EXPIRED:
89*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_CERTIFICATE_UNKNOWN:
90*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_ACCESS_DENIED:
91*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_CERTIFICATE_REQUIRED:
92*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_UNKNOWN_CA:
93*6777b538SAndroid Build Coastguard Worker return ERR_BAD_SSL_CLIENT_AUTH_CERT;
94*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_DECOMPRESSION_FAILURE:
95*6777b538SAndroid Build Coastguard Worker return ERR_SSL_DECOMPRESSION_FAILURE_ALERT;
96*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_BAD_RECORD_MAC:
97*6777b538SAndroid Build Coastguard Worker return ERR_SSL_BAD_RECORD_MAC_ALERT;
98*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_ALERT_DECRYPT_ERROR:
99*6777b538SAndroid Build Coastguard Worker return ERR_SSL_DECRYPT_ERROR_ALERT;
100*6777b538SAndroid Build Coastguard Worker case SSL_R_TLSV1_UNRECOGNIZED_NAME:
101*6777b538SAndroid Build Coastguard Worker return ERR_SSL_UNRECOGNIZED_NAME_ALERT;
102*6777b538SAndroid Build Coastguard Worker case SSL_R_SERVER_CERT_CHANGED:
103*6777b538SAndroid Build Coastguard Worker return ERR_SSL_SERVER_CERT_CHANGED;
104*6777b538SAndroid Build Coastguard Worker case SSL_R_WRONG_VERSION_ON_EARLY_DATA:
105*6777b538SAndroid Build Coastguard Worker return ERR_WRONG_VERSION_ON_EARLY_DATA;
106*6777b538SAndroid Build Coastguard Worker case SSL_R_TLS13_DOWNGRADE:
107*6777b538SAndroid Build Coastguard Worker return ERR_TLS13_DOWNGRADE_DETECTED;
108*6777b538SAndroid Build Coastguard Worker case SSL_R_ECH_REJECTED:
109*6777b538SAndroid Build Coastguard Worker return ERR_ECH_NOT_NEGOTIATED;
110*6777b538SAndroid Build Coastguard Worker // SSL_R_SSLV3_ALERT_HANDSHAKE_FAILURE may be returned from the server after
111*6777b538SAndroid Build Coastguard Worker // receiving ClientHello if there's no common supported cipher. Map that
112*6777b538SAndroid Build Coastguard Worker // specific case to ERR_SSL_VERSION_OR_CIPHER_MISMATCH to match the NSS
113*6777b538SAndroid Build Coastguard Worker // implementation. See https://goo.gl/oMtZW and https://crbug.com/446505.
114*6777b538SAndroid Build Coastguard Worker case SSL_R_SSLV3_ALERT_HANDSHAKE_FAILURE: {
115*6777b538SAndroid Build Coastguard Worker uint32_t previous = ERR_peek_error();
116*6777b538SAndroid Build Coastguard Worker if (previous != 0 && ERR_GET_LIB(previous) == ERR_LIB_SSL &&
117*6777b538SAndroid Build Coastguard Worker ERR_GET_REASON(previous) == SSL_R_HANDSHAKE_FAILURE_ON_CLIENT_HELLO) {
118*6777b538SAndroid Build Coastguard Worker return ERR_SSL_VERSION_OR_CIPHER_MISMATCH;
119*6777b538SAndroid Build Coastguard Worker }
120*6777b538SAndroid Build Coastguard Worker return ERR_SSL_PROTOCOL_ERROR;
121*6777b538SAndroid Build Coastguard Worker }
122*6777b538SAndroid Build Coastguard Worker case SSL_R_KEY_USAGE_BIT_INCORRECT:
123*6777b538SAndroid Build Coastguard Worker return ERR_SSL_KEY_USAGE_INCOMPATIBLE;
124*6777b538SAndroid Build Coastguard Worker default:
125*6777b538SAndroid Build Coastguard Worker return ERR_SSL_PROTOCOL_ERROR;
126*6777b538SAndroid Build Coastguard Worker }
127*6777b538SAndroid Build Coastguard Worker }
128*6777b538SAndroid Build Coastguard Worker
NetLogOpenSSLErrorParams(int net_error,int ssl_error,const OpenSSLErrorInfo & error_info)129*6777b538SAndroid Build Coastguard Worker base::Value::Dict NetLogOpenSSLErrorParams(int net_error,
130*6777b538SAndroid Build Coastguard Worker int ssl_error,
131*6777b538SAndroid Build Coastguard Worker const OpenSSLErrorInfo& error_info) {
132*6777b538SAndroid Build Coastguard Worker base::Value::Dict dict;
133*6777b538SAndroid Build Coastguard Worker dict.Set("net_error", net_error);
134*6777b538SAndroid Build Coastguard Worker dict.Set("ssl_error", ssl_error);
135*6777b538SAndroid Build Coastguard Worker if (error_info.error_code != 0) {
136*6777b538SAndroid Build Coastguard Worker dict.Set("error_lib", ERR_GET_LIB(error_info.error_code));
137*6777b538SAndroid Build Coastguard Worker dict.Set("error_reason", ERR_GET_REASON(error_info.error_code));
138*6777b538SAndroid Build Coastguard Worker }
139*6777b538SAndroid Build Coastguard Worker if (error_info.file != nullptr)
140*6777b538SAndroid Build Coastguard Worker dict.Set("file", error_info.file);
141*6777b538SAndroid Build Coastguard Worker if (error_info.line != 0)
142*6777b538SAndroid Build Coastguard Worker dict.Set("line", error_info.line);
143*6777b538SAndroid Build Coastguard Worker return dict;
144*6777b538SAndroid Build Coastguard Worker }
145*6777b538SAndroid Build Coastguard Worker
146*6777b538SAndroid Build Coastguard Worker } // namespace
147*6777b538SAndroid Build Coastguard Worker
OpenSSLPutNetError(const base::Location & location,int err)148*6777b538SAndroid Build Coastguard Worker void OpenSSLPutNetError(const base::Location& location, int err) {
149*6777b538SAndroid Build Coastguard Worker // Net error codes are negative. Encode them as positive numbers.
150*6777b538SAndroid Build Coastguard Worker err = -err;
151*6777b538SAndroid Build Coastguard Worker if (err < 0 || err > 0xfff) {
152*6777b538SAndroid Build Coastguard Worker // OpenSSL reserves 12 bits for the reason code.
153*6777b538SAndroid Build Coastguard Worker NOTREACHED();
154*6777b538SAndroid Build Coastguard Worker err = ERR_INVALID_ARGUMENT;
155*6777b538SAndroid Build Coastguard Worker }
156*6777b538SAndroid Build Coastguard Worker ERR_put_error(OpenSSLNetErrorLib(), 0 /* unused */, err, location.file_name(),
157*6777b538SAndroid Build Coastguard Worker location.line_number());
158*6777b538SAndroid Build Coastguard Worker }
159*6777b538SAndroid Build Coastguard Worker
MapOpenSSLError(int err,const crypto::OpenSSLErrStackTracer & tracer)160*6777b538SAndroid Build Coastguard Worker int MapOpenSSLError(int err, const crypto::OpenSSLErrStackTracer& tracer) {
161*6777b538SAndroid Build Coastguard Worker OpenSSLErrorInfo error_info;
162*6777b538SAndroid Build Coastguard Worker return MapOpenSSLErrorWithDetails(err, tracer, &error_info);
163*6777b538SAndroid Build Coastguard Worker }
164*6777b538SAndroid Build Coastguard Worker
MapOpenSSLErrorWithDetails(int err,const crypto::OpenSSLErrStackTracer & tracer,OpenSSLErrorInfo * out_error_info)165*6777b538SAndroid Build Coastguard Worker int MapOpenSSLErrorWithDetails(int err,
166*6777b538SAndroid Build Coastguard Worker const crypto::OpenSSLErrStackTracer& tracer,
167*6777b538SAndroid Build Coastguard Worker OpenSSLErrorInfo* out_error_info) {
168*6777b538SAndroid Build Coastguard Worker *out_error_info = OpenSSLErrorInfo();
169*6777b538SAndroid Build Coastguard Worker
170*6777b538SAndroid Build Coastguard Worker switch (err) {
171*6777b538SAndroid Build Coastguard Worker case SSL_ERROR_WANT_READ:
172*6777b538SAndroid Build Coastguard Worker case SSL_ERROR_WANT_WRITE:
173*6777b538SAndroid Build Coastguard Worker return ERR_IO_PENDING;
174*6777b538SAndroid Build Coastguard Worker case SSL_ERROR_EARLY_DATA_REJECTED:
175*6777b538SAndroid Build Coastguard Worker return ERR_EARLY_DATA_REJECTED;
176*6777b538SAndroid Build Coastguard Worker case SSL_ERROR_SYSCALL:
177*6777b538SAndroid Build Coastguard Worker PLOG(ERROR) << "OpenSSL SYSCALL error, earliest error code in "
178*6777b538SAndroid Build Coastguard Worker "error queue: "
179*6777b538SAndroid Build Coastguard Worker << ERR_peek_error();
180*6777b538SAndroid Build Coastguard Worker return ERR_FAILED;
181*6777b538SAndroid Build Coastguard Worker case SSL_ERROR_SSL:
182*6777b538SAndroid Build Coastguard Worker // Walk down the error stack to find an SSL or net error.
183*6777b538SAndroid Build Coastguard Worker while (true) {
184*6777b538SAndroid Build Coastguard Worker OpenSSLErrorInfo error_info;
185*6777b538SAndroid Build Coastguard Worker error_info.error_code =
186*6777b538SAndroid Build Coastguard Worker ERR_get_error_line(&error_info.file, &error_info.line);
187*6777b538SAndroid Build Coastguard Worker if (error_info.error_code == 0) {
188*6777b538SAndroid Build Coastguard Worker // Map errors to ERR_SSL_PROTOCOL_ERROR by default, reporting the most
189*6777b538SAndroid Build Coastguard Worker // recent error in |*out_error_info|.
190*6777b538SAndroid Build Coastguard Worker return ERR_SSL_PROTOCOL_ERROR;
191*6777b538SAndroid Build Coastguard Worker }
192*6777b538SAndroid Build Coastguard Worker
193*6777b538SAndroid Build Coastguard Worker *out_error_info = error_info;
194*6777b538SAndroid Build Coastguard Worker if (ERR_GET_LIB(error_info.error_code) == ERR_LIB_SSL) {
195*6777b538SAndroid Build Coastguard Worker return MapOpenSSLErrorSSL(error_info.error_code);
196*6777b538SAndroid Build Coastguard Worker }
197*6777b538SAndroid Build Coastguard Worker if (ERR_GET_LIB(error_info.error_code) == OpenSSLNetErrorLib()) {
198*6777b538SAndroid Build Coastguard Worker // Net error codes are negative but encoded in OpenSSL as positive
199*6777b538SAndroid Build Coastguard Worker // numbers.
200*6777b538SAndroid Build Coastguard Worker return -ERR_GET_REASON(error_info.error_code);
201*6777b538SAndroid Build Coastguard Worker }
202*6777b538SAndroid Build Coastguard Worker }
203*6777b538SAndroid Build Coastguard Worker default:
204*6777b538SAndroid Build Coastguard Worker // TODO(joth): Implement full mapping.
205*6777b538SAndroid Build Coastguard Worker LOG(WARNING) << "Unknown OpenSSL error " << err;
206*6777b538SAndroid Build Coastguard Worker return ERR_SSL_PROTOCOL_ERROR;
207*6777b538SAndroid Build Coastguard Worker }
208*6777b538SAndroid Build Coastguard Worker }
209*6777b538SAndroid Build Coastguard Worker
NetLogOpenSSLError(const NetLogWithSource & net_log,NetLogEventType type,int net_error,int ssl_error,const OpenSSLErrorInfo & error_info)210*6777b538SAndroid Build Coastguard Worker void NetLogOpenSSLError(const NetLogWithSource& net_log,
211*6777b538SAndroid Build Coastguard Worker NetLogEventType type,
212*6777b538SAndroid Build Coastguard Worker int net_error,
213*6777b538SAndroid Build Coastguard Worker int ssl_error,
214*6777b538SAndroid Build Coastguard Worker const OpenSSLErrorInfo& error_info) {
215*6777b538SAndroid Build Coastguard Worker net_log.AddEvent(type, [&] {
216*6777b538SAndroid Build Coastguard Worker return NetLogOpenSSLErrorParams(net_error, ssl_error, error_info);
217*6777b538SAndroid Build Coastguard Worker });
218*6777b538SAndroid Build Coastguard Worker }
219*6777b538SAndroid Build Coastguard Worker
GetNetSSLVersion(SSL * ssl)220*6777b538SAndroid Build Coastguard Worker int GetNetSSLVersion(SSL* ssl) {
221*6777b538SAndroid Build Coastguard Worker switch (SSL_version(ssl)) {
222*6777b538SAndroid Build Coastguard Worker case TLS1_VERSION:
223*6777b538SAndroid Build Coastguard Worker return SSL_CONNECTION_VERSION_TLS1;
224*6777b538SAndroid Build Coastguard Worker case TLS1_1_VERSION:
225*6777b538SAndroid Build Coastguard Worker return SSL_CONNECTION_VERSION_TLS1_1;
226*6777b538SAndroid Build Coastguard Worker case TLS1_2_VERSION:
227*6777b538SAndroid Build Coastguard Worker return SSL_CONNECTION_VERSION_TLS1_2;
228*6777b538SAndroid Build Coastguard Worker case TLS1_3_VERSION:
229*6777b538SAndroid Build Coastguard Worker return SSL_CONNECTION_VERSION_TLS1_3;
230*6777b538SAndroid Build Coastguard Worker default:
231*6777b538SAndroid Build Coastguard Worker NOTREACHED();
232*6777b538SAndroid Build Coastguard Worker return SSL_CONNECTION_VERSION_UNKNOWN;
233*6777b538SAndroid Build Coastguard Worker }
234*6777b538SAndroid Build Coastguard Worker }
235*6777b538SAndroid Build Coastguard Worker
SetSSLChainAndKey(SSL * ssl,X509Certificate * cert,EVP_PKEY * pkey,const SSL_PRIVATE_KEY_METHOD * custom_key)236*6777b538SAndroid Build Coastguard Worker bool SetSSLChainAndKey(SSL* ssl,
237*6777b538SAndroid Build Coastguard Worker X509Certificate* cert,
238*6777b538SAndroid Build Coastguard Worker EVP_PKEY* pkey,
239*6777b538SAndroid Build Coastguard Worker const SSL_PRIVATE_KEY_METHOD* custom_key) {
240*6777b538SAndroid Build Coastguard Worker std::vector<CRYPTO_BUFFER*> chain_raw;
241*6777b538SAndroid Build Coastguard Worker chain_raw.reserve(1 + cert->intermediate_buffers().size());
242*6777b538SAndroid Build Coastguard Worker chain_raw.push_back(cert->cert_buffer());
243*6777b538SAndroid Build Coastguard Worker for (const auto& handle : cert->intermediate_buffers())
244*6777b538SAndroid Build Coastguard Worker chain_raw.push_back(handle.get());
245*6777b538SAndroid Build Coastguard Worker
246*6777b538SAndroid Build Coastguard Worker if (!SSL_set_chain_and_key(ssl, chain_raw.data(), chain_raw.size(), pkey,
247*6777b538SAndroid Build Coastguard Worker custom_key)) {
248*6777b538SAndroid Build Coastguard Worker LOG(WARNING) << "Failed to set client certificate";
249*6777b538SAndroid Build Coastguard Worker return false;
250*6777b538SAndroid Build Coastguard Worker }
251*6777b538SAndroid Build Coastguard Worker
252*6777b538SAndroid Build Coastguard Worker return true;
253*6777b538SAndroid Build Coastguard Worker }
254*6777b538SAndroid Build Coastguard Worker
255*6777b538SAndroid Build Coastguard Worker } // namespace net
256