xref: /aosp_15_r20/external/cronet/net/cert/x509_certificate.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 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/cert/x509_certificate.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <limits.h>
8*6777b538SAndroid Build Coastguard Worker #include <stdlib.h>
9*6777b538SAndroid Build Coastguard Worker 
10*6777b538SAndroid Build Coastguard Worker #include <memory>
11*6777b538SAndroid Build Coastguard Worker #include <string>
12*6777b538SAndroid Build Coastguard Worker #include <string_view>
13*6777b538SAndroid Build Coastguard Worker #include <vector>
14*6777b538SAndroid Build Coastguard Worker 
15*6777b538SAndroid Build Coastguard Worker #include "base/containers/contains.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/containers/span.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/notreached.h"
19*6777b538SAndroid Build Coastguard Worker #include "base/numerics/safe_conversions.h"
20*6777b538SAndroid Build Coastguard Worker #include "base/pickle.h"
21*6777b538SAndroid Build Coastguard Worker #include "base/strings/strcat.h"
22*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_util.h"
23*6777b538SAndroid Build Coastguard Worker #include "base/time/time.h"
24*6777b538SAndroid Build Coastguard Worker #include "crypto/openssl_util.h"
25*6777b538SAndroid Build Coastguard Worker #include "net/base/registry_controlled_domains/registry_controlled_domain.h"
26*6777b538SAndroid Build Coastguard Worker #include "net/base/tracing.h"
27*6777b538SAndroid Build Coastguard Worker #include "net/base/url_util.h"
28*6777b538SAndroid Build Coastguard Worker #include "net/cert/asn1_util.h"
29*6777b538SAndroid Build Coastguard Worker #include "net/cert/time_conversions.h"
30*6777b538SAndroid Build Coastguard Worker #include "net/cert/x509_util.h"
31*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/include/openssl/evp.h"
32*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/include/openssl/pool.h"
33*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/include/openssl/sha.h"
34*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/cert_errors.h"
35*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/name_constraints.h"
36*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/parsed_certificate.h"
37*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/parser.h"
38*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/pem.h"
39*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/signature_algorithm.h"
40*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/verify_certificate_chain.h"
41*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/verify_name_match.h"
42*6777b538SAndroid Build Coastguard Worker #include "third_party/boringssl/src/pki/verify_signed_data.h"
43*6777b538SAndroid Build Coastguard Worker #include "url/url_canon.h"
44*6777b538SAndroid Build Coastguard Worker 
45*6777b538SAndroid Build Coastguard Worker namespace net {
46*6777b538SAndroid Build Coastguard Worker 
47*6777b538SAndroid Build Coastguard Worker namespace {
48*6777b538SAndroid Build Coastguard Worker 
49*6777b538SAndroid Build Coastguard Worker // Indicates the order to use when trying to decode binary data, which is
50*6777b538SAndroid Build Coastguard Worker // based on (speculation) as to what will be most common -> least common
51*6777b538SAndroid Build Coastguard Worker const X509Certificate::Format kFormatDecodePriority[] = {
52*6777b538SAndroid Build Coastguard Worker   X509Certificate::FORMAT_SINGLE_CERTIFICATE,
53*6777b538SAndroid Build Coastguard Worker   X509Certificate::FORMAT_PKCS7
54*6777b538SAndroid Build Coastguard Worker };
55*6777b538SAndroid Build Coastguard Worker 
56*6777b538SAndroid Build Coastguard Worker // The PEM block header used for DER certificates
57*6777b538SAndroid Build Coastguard Worker const char kCertificateHeader[] = "CERTIFICATE";
58*6777b538SAndroid Build Coastguard Worker // The PEM block header used for PKCS#7 data
59*6777b538SAndroid Build Coastguard Worker const char kPKCS7Header[] = "PKCS7";
60*6777b538SAndroid Build Coastguard Worker 
61*6777b538SAndroid Build Coastguard Worker // Utility to split |src| on the first occurrence of |c|, if any. |right| will
62*6777b538SAndroid Build Coastguard Worker // either be empty if |c| was not found, or will contain the remainder of the
63*6777b538SAndroid Build Coastguard Worker // string including the split character itself.
SplitOnChar(std::string_view src,char c,std::string_view * left,std::string_view * right)64*6777b538SAndroid Build Coastguard Worker void SplitOnChar(std::string_view src,
65*6777b538SAndroid Build Coastguard Worker                  char c,
66*6777b538SAndroid Build Coastguard Worker                  std::string_view* left,
67*6777b538SAndroid Build Coastguard Worker                  std::string_view* right) {
68*6777b538SAndroid Build Coastguard Worker   size_t pos = src.find(c);
69*6777b538SAndroid Build Coastguard Worker   if (pos == std::string_view::npos) {
70*6777b538SAndroid Build Coastguard Worker     *left = src;
71*6777b538SAndroid Build Coastguard Worker     *right = std::string_view();
72*6777b538SAndroid Build Coastguard Worker   } else {
73*6777b538SAndroid Build Coastguard Worker     *left = src.substr(0, pos);
74*6777b538SAndroid Build Coastguard Worker     *right = src.substr(pos);
75*6777b538SAndroid Build Coastguard Worker   }
76*6777b538SAndroid Build Coastguard Worker }
77*6777b538SAndroid Build Coastguard Worker 
78*6777b538SAndroid Build Coastguard Worker // Sets |value| to the Value from a DER Sequence Tag-Length-Value and return
79*6777b538SAndroid Build Coastguard Worker // true, or return false if the TLV was not a valid DER Sequence.
ParseSequenceValue(const bssl::der::Input & tlv,bssl::der::Input * value)80*6777b538SAndroid Build Coastguard Worker [[nodiscard]] bool ParseSequenceValue(const bssl::der::Input& tlv,
81*6777b538SAndroid Build Coastguard Worker                                       bssl::der::Input* value) {
82*6777b538SAndroid Build Coastguard Worker   bssl::der::Parser parser(tlv);
83*6777b538SAndroid Build Coastguard Worker   return parser.ReadTag(CBS_ASN1_SEQUENCE, value) && !parser.HasMore();
84*6777b538SAndroid Build Coastguard Worker }
85*6777b538SAndroid Build Coastguard Worker 
86*6777b538SAndroid Build Coastguard Worker // Normalize |cert|'s Issuer and store it in |out_normalized_issuer|, returning
87*6777b538SAndroid Build Coastguard Worker // true on success or false if there was a parsing error.
GetNormalizedCertIssuer(CRYPTO_BUFFER * cert,std::string * out_normalized_issuer)88*6777b538SAndroid Build Coastguard Worker bool GetNormalizedCertIssuer(CRYPTO_BUFFER* cert,
89*6777b538SAndroid Build Coastguard Worker                              std::string* out_normalized_issuer) {
90*6777b538SAndroid Build Coastguard Worker   bssl::der::Input tbs_certificate_tlv;
91*6777b538SAndroid Build Coastguard Worker   bssl::der::Input signature_algorithm_tlv;
92*6777b538SAndroid Build Coastguard Worker   bssl::der::BitString signature_value;
93*6777b538SAndroid Build Coastguard Worker   if (!bssl::ParseCertificate(
94*6777b538SAndroid Build Coastguard Worker           bssl::der::Input(CRYPTO_BUFFER_data(cert), CRYPTO_BUFFER_len(cert)),
95*6777b538SAndroid Build Coastguard Worker           &tbs_certificate_tlv, &signature_algorithm_tlv, &signature_value,
96*6777b538SAndroid Build Coastguard Worker           nullptr)) {
97*6777b538SAndroid Build Coastguard Worker     return false;
98*6777b538SAndroid Build Coastguard Worker   }
99*6777b538SAndroid Build Coastguard Worker   bssl::ParsedTbsCertificate tbs;
100*6777b538SAndroid Build Coastguard Worker   if (!ParseTbsCertificate(tbs_certificate_tlv,
101*6777b538SAndroid Build Coastguard Worker                            x509_util::DefaultParseCertificateOptions(), &tbs,
102*6777b538SAndroid Build Coastguard Worker                            nullptr))
103*6777b538SAndroid Build Coastguard Worker     return false;
104*6777b538SAndroid Build Coastguard Worker 
105*6777b538SAndroid Build Coastguard Worker   bssl::der::Input issuer_value;
106*6777b538SAndroid Build Coastguard Worker   if (!ParseSequenceValue(tbs.issuer_tlv, &issuer_value))
107*6777b538SAndroid Build Coastguard Worker     return false;
108*6777b538SAndroid Build Coastguard Worker 
109*6777b538SAndroid Build Coastguard Worker   bssl::CertErrors errors;
110*6777b538SAndroid Build Coastguard Worker   return NormalizeName(issuer_value, out_normalized_issuer, &errors);
111*6777b538SAndroid Build Coastguard Worker }
112*6777b538SAndroid Build Coastguard Worker 
CreateCertBufferFromBytesWithSanityCheck(base::span<const uint8_t> data)113*6777b538SAndroid Build Coastguard Worker bssl::UniquePtr<CRYPTO_BUFFER> CreateCertBufferFromBytesWithSanityCheck(
114*6777b538SAndroid Build Coastguard Worker     base::span<const uint8_t> data) {
115*6777b538SAndroid Build Coastguard Worker   bssl::der::Input tbs_certificate_tlv;
116*6777b538SAndroid Build Coastguard Worker   bssl::der::Input signature_algorithm_tlv;
117*6777b538SAndroid Build Coastguard Worker   bssl::der::BitString signature_value;
118*6777b538SAndroid Build Coastguard Worker   // Do a bare minimum of DER parsing here to see if the input looks
119*6777b538SAndroid Build Coastguard Worker   // certificate-ish.
120*6777b538SAndroid Build Coastguard Worker   if (!bssl::ParseCertificate(bssl::der::Input(data), &tbs_certificate_tlv,
121*6777b538SAndroid Build Coastguard Worker                               &signature_algorithm_tlv, &signature_value,
122*6777b538SAndroid Build Coastguard Worker                               nullptr)) {
123*6777b538SAndroid Build Coastguard Worker     return nullptr;
124*6777b538SAndroid Build Coastguard Worker   }
125*6777b538SAndroid Build Coastguard Worker   return x509_util::CreateCryptoBuffer(data);
126*6777b538SAndroid Build Coastguard Worker }
127*6777b538SAndroid Build Coastguard Worker 
128*6777b538SAndroid Build Coastguard Worker }  // namespace
129*6777b538SAndroid Build Coastguard Worker 
130*6777b538SAndroid Build Coastguard Worker // static
CreateFromBuffer(bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)131*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromBuffer(
132*6777b538SAndroid Build Coastguard Worker     bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,
133*6777b538SAndroid Build Coastguard Worker     std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates) {
134*6777b538SAndroid Build Coastguard Worker   return CreateFromBufferUnsafeOptions(std::move(cert_buffer),
135*6777b538SAndroid Build Coastguard Worker                                        std::move(intermediates), {});
136*6777b538SAndroid Build Coastguard Worker }
137*6777b538SAndroid Build Coastguard Worker 
138*6777b538SAndroid Build Coastguard Worker // static
CreateFromBufferUnsafeOptions(bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates,UnsafeCreateOptions options)139*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromBufferUnsafeOptions(
140*6777b538SAndroid Build Coastguard Worker     bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,
141*6777b538SAndroid Build Coastguard Worker     std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates,
142*6777b538SAndroid Build Coastguard Worker     UnsafeCreateOptions options) {
143*6777b538SAndroid Build Coastguard Worker   DCHECK(cert_buffer);
144*6777b538SAndroid Build Coastguard Worker   ParsedFields parsed;
145*6777b538SAndroid Build Coastguard Worker   if (!parsed.Initialize(cert_buffer.get(), options)) {
146*6777b538SAndroid Build Coastguard Worker     return nullptr;
147*6777b538SAndroid Build Coastguard Worker   }
148*6777b538SAndroid Build Coastguard Worker   return base::WrapRefCounted(new X509Certificate(
149*6777b538SAndroid Build Coastguard Worker       std::move(parsed), std::move(cert_buffer), std::move(intermediates)));
150*6777b538SAndroid Build Coastguard Worker }
151*6777b538SAndroid Build Coastguard Worker 
152*6777b538SAndroid Build Coastguard Worker // static
CreateFromDERCertChain(const std::vector<std::string_view> & der_certs)153*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromDERCertChain(
154*6777b538SAndroid Build Coastguard Worker     const std::vector<std::string_view>& der_certs) {
155*6777b538SAndroid Build Coastguard Worker   return CreateFromDERCertChainUnsafeOptions(der_certs, {});
156*6777b538SAndroid Build Coastguard Worker }
157*6777b538SAndroid Build Coastguard Worker 
158*6777b538SAndroid Build Coastguard Worker // static
159*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate>
CreateFromDERCertChainUnsafeOptions(const std::vector<std::string_view> & der_certs,UnsafeCreateOptions options)160*6777b538SAndroid Build Coastguard Worker X509Certificate::CreateFromDERCertChainUnsafeOptions(
161*6777b538SAndroid Build Coastguard Worker     const std::vector<std::string_view>& der_certs,
162*6777b538SAndroid Build Coastguard Worker     UnsafeCreateOptions options) {
163*6777b538SAndroid Build Coastguard Worker   TRACE_EVENT0("io", "X509Certificate::CreateFromDERCertChain");
164*6777b538SAndroid Build Coastguard Worker   if (der_certs.empty())
165*6777b538SAndroid Build Coastguard Worker     return nullptr;
166*6777b538SAndroid Build Coastguard Worker 
167*6777b538SAndroid Build Coastguard Worker   std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediate_ca_certs;
168*6777b538SAndroid Build Coastguard Worker   intermediate_ca_certs.reserve(der_certs.size() - 1);
169*6777b538SAndroid Build Coastguard Worker   for (size_t i = 1; i < der_certs.size(); i++) {
170*6777b538SAndroid Build Coastguard Worker     intermediate_ca_certs.push_back(
171*6777b538SAndroid Build Coastguard Worker         x509_util::CreateCryptoBuffer(der_certs[i]));
172*6777b538SAndroid Build Coastguard Worker   }
173*6777b538SAndroid Build Coastguard Worker 
174*6777b538SAndroid Build Coastguard Worker   return CreateFromBufferUnsafeOptions(
175*6777b538SAndroid Build Coastguard Worker       x509_util::CreateCryptoBuffer(der_certs[0]),
176*6777b538SAndroid Build Coastguard Worker       std::move(intermediate_ca_certs), options);
177*6777b538SAndroid Build Coastguard Worker }
178*6777b538SAndroid Build Coastguard Worker 
179*6777b538SAndroid Build Coastguard Worker // static
CreateFromBytes(base::span<const uint8_t> data)180*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromBytes(
181*6777b538SAndroid Build Coastguard Worker     base::span<const uint8_t> data) {
182*6777b538SAndroid Build Coastguard Worker   return CreateFromBytesUnsafeOptions(data, {});
183*6777b538SAndroid Build Coastguard Worker }
184*6777b538SAndroid Build Coastguard Worker 
185*6777b538SAndroid Build Coastguard Worker // static
CreateFromBytesUnsafeOptions(base::span<const uint8_t> data,UnsafeCreateOptions options)186*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromBytesUnsafeOptions(
187*6777b538SAndroid Build Coastguard Worker     base::span<const uint8_t> data,
188*6777b538SAndroid Build Coastguard Worker     UnsafeCreateOptions options) {
189*6777b538SAndroid Build Coastguard Worker   scoped_refptr<X509Certificate> cert = CreateFromBufferUnsafeOptions(
190*6777b538SAndroid Build Coastguard Worker       x509_util::CreateCryptoBuffer(data), {}, options);
191*6777b538SAndroid Build Coastguard Worker   return cert;
192*6777b538SAndroid Build Coastguard Worker }
193*6777b538SAndroid Build Coastguard Worker 
194*6777b538SAndroid Build Coastguard Worker // static
CreateFromPickle(base::PickleIterator * pickle_iter)195*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromPickle(
196*6777b538SAndroid Build Coastguard Worker     base::PickleIterator* pickle_iter) {
197*6777b538SAndroid Build Coastguard Worker   return CreateFromPickleUnsafeOptions(pickle_iter, {});
198*6777b538SAndroid Build Coastguard Worker }
199*6777b538SAndroid Build Coastguard Worker 
200*6777b538SAndroid Build Coastguard Worker // static
CreateFromPickleUnsafeOptions(base::PickleIterator * pickle_iter,UnsafeCreateOptions options)201*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CreateFromPickleUnsafeOptions(
202*6777b538SAndroid Build Coastguard Worker     base::PickleIterator* pickle_iter,
203*6777b538SAndroid Build Coastguard Worker     UnsafeCreateOptions options) {
204*6777b538SAndroid Build Coastguard Worker   size_t chain_length = 0;
205*6777b538SAndroid Build Coastguard Worker   if (!pickle_iter->ReadLength(&chain_length))
206*6777b538SAndroid Build Coastguard Worker     return nullptr;
207*6777b538SAndroid Build Coastguard Worker 
208*6777b538SAndroid Build Coastguard Worker   std::vector<std::string_view> cert_chain;
209*6777b538SAndroid Build Coastguard Worker   const char* data = nullptr;
210*6777b538SAndroid Build Coastguard Worker   size_t data_length = 0;
211*6777b538SAndroid Build Coastguard Worker   for (size_t i = 0; i < chain_length; ++i) {
212*6777b538SAndroid Build Coastguard Worker     if (!pickle_iter->ReadData(&data, &data_length))
213*6777b538SAndroid Build Coastguard Worker       return nullptr;
214*6777b538SAndroid Build Coastguard Worker     cert_chain.emplace_back(data, data_length);
215*6777b538SAndroid Build Coastguard Worker   }
216*6777b538SAndroid Build Coastguard Worker   return CreateFromDERCertChainUnsafeOptions(cert_chain, options);
217*6777b538SAndroid Build Coastguard Worker }
218*6777b538SAndroid Build Coastguard Worker 
219*6777b538SAndroid Build Coastguard Worker // static
CreateCertificateListFromBytes(base::span<const uint8_t> data,int format)220*6777b538SAndroid Build Coastguard Worker CertificateList X509Certificate::CreateCertificateListFromBytes(
221*6777b538SAndroid Build Coastguard Worker     base::span<const uint8_t> data,
222*6777b538SAndroid Build Coastguard Worker     int format) {
223*6777b538SAndroid Build Coastguard Worker   std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> certificates;
224*6777b538SAndroid Build Coastguard Worker 
225*6777b538SAndroid Build Coastguard Worker   // Check to see if it is in a PEM-encoded form. This check is performed
226*6777b538SAndroid Build Coastguard Worker   // first, as both OS X and NSS will both try to convert if they detect
227*6777b538SAndroid Build Coastguard Worker   // PEM encoding, except they don't do it consistently between the two.
228*6777b538SAndroid Build Coastguard Worker   std::string_view data_string(reinterpret_cast<const char*>(data.data()),
229*6777b538SAndroid Build Coastguard Worker                                data.size());
230*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> pem_headers;
231*6777b538SAndroid Build Coastguard Worker 
232*6777b538SAndroid Build Coastguard Worker   // To maintain compatibility with NSS/Firefox, CERTIFICATE is a universally
233*6777b538SAndroid Build Coastguard Worker   // valid PEM block header for any format.
234*6777b538SAndroid Build Coastguard Worker   pem_headers.push_back(kCertificateHeader);
235*6777b538SAndroid Build Coastguard Worker   if (format & FORMAT_PKCS7)
236*6777b538SAndroid Build Coastguard Worker     pem_headers.push_back(kPKCS7Header);
237*6777b538SAndroid Build Coastguard Worker 
238*6777b538SAndroid Build Coastguard Worker   bssl::PEMTokenizer pem_tokenizer(data_string, pem_headers);
239*6777b538SAndroid Build Coastguard Worker   while (pem_tokenizer.GetNext()) {
240*6777b538SAndroid Build Coastguard Worker     std::string decoded(pem_tokenizer.data());
241*6777b538SAndroid Build Coastguard Worker 
242*6777b538SAndroid Build Coastguard Worker     bssl::UniquePtr<CRYPTO_BUFFER> handle;
243*6777b538SAndroid Build Coastguard Worker     if (format & FORMAT_PEM_CERT_SEQUENCE) {
244*6777b538SAndroid Build Coastguard Worker       handle =
245*6777b538SAndroid Build Coastguard Worker           CreateCertBufferFromBytesWithSanityCheck(base::as_byte_span(decoded));
246*6777b538SAndroid Build Coastguard Worker     }
247*6777b538SAndroid Build Coastguard Worker     if (handle) {
248*6777b538SAndroid Build Coastguard Worker       // Parsed a DER encoded certificate. All PEM blocks that follow must
249*6777b538SAndroid Build Coastguard Worker       // also be DER encoded certificates wrapped inside of PEM blocks.
250*6777b538SAndroid Build Coastguard Worker       format = FORMAT_PEM_CERT_SEQUENCE;
251*6777b538SAndroid Build Coastguard Worker       certificates.push_back(std::move(handle));
252*6777b538SAndroid Build Coastguard Worker       continue;
253*6777b538SAndroid Build Coastguard Worker     }
254*6777b538SAndroid Build Coastguard Worker 
255*6777b538SAndroid Build Coastguard Worker     // If the first block failed to parse as a DER certificate, and
256*6777b538SAndroid Build Coastguard Worker     // formats other than PEM are acceptable, check to see if the decoded
257*6777b538SAndroid Build Coastguard Worker     // data is one of the accepted formats.
258*6777b538SAndroid Build Coastguard Worker     if (format & ~FORMAT_PEM_CERT_SEQUENCE) {
259*6777b538SAndroid Build Coastguard Worker       for (size_t i = 0;
260*6777b538SAndroid Build Coastguard Worker            certificates.empty() && i < std::size(kFormatDecodePriority); ++i) {
261*6777b538SAndroid Build Coastguard Worker         if (format & kFormatDecodePriority[i]) {
262*6777b538SAndroid Build Coastguard Worker           certificates = CreateCertBuffersFromBytes(base::as_byte_span(decoded),
263*6777b538SAndroid Build Coastguard Worker                                                     kFormatDecodePriority[i]);
264*6777b538SAndroid Build Coastguard Worker         }
265*6777b538SAndroid Build Coastguard Worker       }
266*6777b538SAndroid Build Coastguard Worker     }
267*6777b538SAndroid Build Coastguard Worker 
268*6777b538SAndroid Build Coastguard Worker     // Stop parsing after the first block for any format but a sequence of
269*6777b538SAndroid Build Coastguard Worker     // PEM-encoded DER certificates. The case of FORMAT_PEM_CERT_SEQUENCE
270*6777b538SAndroid Build Coastguard Worker     // is handled above, and continues processing until a certificate fails
271*6777b538SAndroid Build Coastguard Worker     // to parse.
272*6777b538SAndroid Build Coastguard Worker     break;
273*6777b538SAndroid Build Coastguard Worker   }
274*6777b538SAndroid Build Coastguard Worker 
275*6777b538SAndroid Build Coastguard Worker   // Try each of the formats, in order of parse preference, to see if |data|
276*6777b538SAndroid Build Coastguard Worker   // contains the binary representation of a Format, if it failed to parse
277*6777b538SAndroid Build Coastguard Worker   // as a PEM certificate/chain.
278*6777b538SAndroid Build Coastguard Worker   for (size_t i = 0;
279*6777b538SAndroid Build Coastguard Worker        certificates.empty() && i < std::size(kFormatDecodePriority); ++i) {
280*6777b538SAndroid Build Coastguard Worker     if (format & kFormatDecodePriority[i])
281*6777b538SAndroid Build Coastguard Worker       certificates = CreateCertBuffersFromBytes(data, kFormatDecodePriority[i]);
282*6777b538SAndroid Build Coastguard Worker   }
283*6777b538SAndroid Build Coastguard Worker 
284*6777b538SAndroid Build Coastguard Worker   CertificateList results;
285*6777b538SAndroid Build Coastguard Worker   // No certificates parsed.
286*6777b538SAndroid Build Coastguard Worker   if (certificates.empty())
287*6777b538SAndroid Build Coastguard Worker     return results;
288*6777b538SAndroid Build Coastguard Worker 
289*6777b538SAndroid Build Coastguard Worker   for (auto& it : certificates) {
290*6777b538SAndroid Build Coastguard Worker     scoped_refptr<X509Certificate> cert = CreateFromBuffer(std::move(it), {});
291*6777b538SAndroid Build Coastguard Worker     if (cert)
292*6777b538SAndroid Build Coastguard Worker       results.push_back(std::move(cert));
293*6777b538SAndroid Build Coastguard Worker   }
294*6777b538SAndroid Build Coastguard Worker 
295*6777b538SAndroid Build Coastguard Worker   return results;
296*6777b538SAndroid Build Coastguard Worker }
297*6777b538SAndroid Build Coastguard Worker 
CloneWithDifferentIntermediates(std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)298*6777b538SAndroid Build Coastguard Worker scoped_refptr<X509Certificate> X509Certificate::CloneWithDifferentIntermediates(
299*6777b538SAndroid Build Coastguard Worker     std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates) {
300*6777b538SAndroid Build Coastguard Worker   // If intermediates are the same, return another reference to the same
301*6777b538SAndroid Build Coastguard Worker   // object. Note that this only does a pointer equality comparison on the
302*6777b538SAndroid Build Coastguard Worker   // CRYPTO_BUFFERs, which is generally sufficient, but in some edge cases
303*6777b538SAndroid Build Coastguard Worker   // buffers have equal contents but with different addresses. This is
304*6777b538SAndroid Build Coastguard Worker   // acceptable as this is just an optimization.
305*6777b538SAndroid Build Coastguard Worker   if (intermediates == intermediate_ca_certs_) {
306*6777b538SAndroid Build Coastguard Worker     return this;
307*6777b538SAndroid Build Coastguard Worker   }
308*6777b538SAndroid Build Coastguard Worker 
309*6777b538SAndroid Build Coastguard Worker   return base::WrapRefCounted(
310*6777b538SAndroid Build Coastguard Worker       new X509Certificate(*this, std::move(intermediates)));
311*6777b538SAndroid Build Coastguard Worker }
312*6777b538SAndroid Build Coastguard Worker 
Persist(base::Pickle * pickle) const313*6777b538SAndroid Build Coastguard Worker void X509Certificate::Persist(base::Pickle* pickle) const {
314*6777b538SAndroid Build Coastguard Worker   DCHECK(cert_buffer_);
315*6777b538SAndroid Build Coastguard Worker   // This would be an absolutely insane number of intermediates.
316*6777b538SAndroid Build Coastguard Worker   if (intermediate_ca_certs_.size() > static_cast<size_t>(INT_MAX) - 1) {
317*6777b538SAndroid Build Coastguard Worker     NOTREACHED();
318*6777b538SAndroid Build Coastguard Worker     return;
319*6777b538SAndroid Build Coastguard Worker   }
320*6777b538SAndroid Build Coastguard Worker   pickle->WriteInt(static_cast<int>(intermediate_ca_certs_.size() + 1));
321*6777b538SAndroid Build Coastguard Worker   pickle->WriteString(x509_util::CryptoBufferAsStringPiece(cert_buffer_.get()));
322*6777b538SAndroid Build Coastguard Worker   for (const auto& intermediate : intermediate_ca_certs_) {
323*6777b538SAndroid Build Coastguard Worker     pickle->WriteString(
324*6777b538SAndroid Build Coastguard Worker         x509_util::CryptoBufferAsStringPiece(intermediate.get()));
325*6777b538SAndroid Build Coastguard Worker   }
326*6777b538SAndroid Build Coastguard Worker }
327*6777b538SAndroid Build Coastguard Worker 
GetSubjectAltName(std::vector<std::string> * dns_names,std::vector<std::string> * ip_addrs) const328*6777b538SAndroid Build Coastguard Worker bool X509Certificate::GetSubjectAltName(
329*6777b538SAndroid Build Coastguard Worker     std::vector<std::string>* dns_names,
330*6777b538SAndroid Build Coastguard Worker     std::vector<std::string>* ip_addrs) const {
331*6777b538SAndroid Build Coastguard Worker   if (dns_names)
332*6777b538SAndroid Build Coastguard Worker     dns_names->clear();
333*6777b538SAndroid Build Coastguard Worker   if (ip_addrs)
334*6777b538SAndroid Build Coastguard Worker     ip_addrs->clear();
335*6777b538SAndroid Build Coastguard Worker 
336*6777b538SAndroid Build Coastguard Worker   bssl::der::Input tbs_certificate_tlv;
337*6777b538SAndroid Build Coastguard Worker   bssl::der::Input signature_algorithm_tlv;
338*6777b538SAndroid Build Coastguard Worker   bssl::der::BitString signature_value;
339*6777b538SAndroid Build Coastguard Worker   if (!bssl::ParseCertificate(
340*6777b538SAndroid Build Coastguard Worker           bssl::der::Input(CRYPTO_BUFFER_data(cert_buffer_.get()),
341*6777b538SAndroid Build Coastguard Worker                            CRYPTO_BUFFER_len(cert_buffer_.get())),
342*6777b538SAndroid Build Coastguard Worker           &tbs_certificate_tlv, &signature_algorithm_tlv, &signature_value,
343*6777b538SAndroid Build Coastguard Worker           nullptr)) {
344*6777b538SAndroid Build Coastguard Worker     return false;
345*6777b538SAndroid Build Coastguard Worker   }
346*6777b538SAndroid Build Coastguard Worker 
347*6777b538SAndroid Build Coastguard Worker   bssl::ParsedTbsCertificate tbs;
348*6777b538SAndroid Build Coastguard Worker   if (!ParseTbsCertificate(tbs_certificate_tlv,
349*6777b538SAndroid Build Coastguard Worker                            x509_util::DefaultParseCertificateOptions(), &tbs,
350*6777b538SAndroid Build Coastguard Worker                            nullptr))
351*6777b538SAndroid Build Coastguard Worker     return false;
352*6777b538SAndroid Build Coastguard Worker   if (!tbs.extensions_tlv)
353*6777b538SAndroid Build Coastguard Worker     return false;
354*6777b538SAndroid Build Coastguard Worker 
355*6777b538SAndroid Build Coastguard Worker   std::map<bssl::der::Input, bssl::ParsedExtension> extensions;
356*6777b538SAndroid Build Coastguard Worker   if (!ParseExtensions(tbs.extensions_tlv.value(), &extensions))
357*6777b538SAndroid Build Coastguard Worker     return false;
358*6777b538SAndroid Build Coastguard Worker 
359*6777b538SAndroid Build Coastguard Worker   bssl::ParsedExtension subject_alt_names_extension;
360*6777b538SAndroid Build Coastguard Worker   if (!ConsumeExtension(bssl::der::Input(bssl::kSubjectAltNameOid), &extensions,
361*6777b538SAndroid Build Coastguard Worker                         &subject_alt_names_extension)) {
362*6777b538SAndroid Build Coastguard Worker     return false;
363*6777b538SAndroid Build Coastguard Worker   }
364*6777b538SAndroid Build Coastguard Worker 
365*6777b538SAndroid Build Coastguard Worker   bssl::CertErrors errors;
366*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<bssl::GeneralNames> subject_alt_names =
367*6777b538SAndroid Build Coastguard Worker       bssl::GeneralNames::Create(subject_alt_names_extension.value, &errors);
368*6777b538SAndroid Build Coastguard Worker   if (!subject_alt_names)
369*6777b538SAndroid Build Coastguard Worker     return false;
370*6777b538SAndroid Build Coastguard Worker 
371*6777b538SAndroid Build Coastguard Worker   if (dns_names) {
372*6777b538SAndroid Build Coastguard Worker     for (const auto& dns_name : subject_alt_names->dns_names)
373*6777b538SAndroid Build Coastguard Worker       dns_names->push_back(std::string(dns_name));
374*6777b538SAndroid Build Coastguard Worker   }
375*6777b538SAndroid Build Coastguard Worker   if (ip_addrs) {
376*6777b538SAndroid Build Coastguard Worker     for (const auto& addr : subject_alt_names->ip_addresses) {
377*6777b538SAndroid Build Coastguard Worker       ip_addrs->push_back(std::string(addr.AsStringView()));
378*6777b538SAndroid Build Coastguard Worker     }
379*6777b538SAndroid Build Coastguard Worker   }
380*6777b538SAndroid Build Coastguard Worker 
381*6777b538SAndroid Build Coastguard Worker   return !subject_alt_names->dns_names.empty() ||
382*6777b538SAndroid Build Coastguard Worker          !subject_alt_names->ip_addresses.empty();
383*6777b538SAndroid Build Coastguard Worker }
384*6777b538SAndroid Build Coastguard Worker 
HasExpired() const385*6777b538SAndroid Build Coastguard Worker bool X509Certificate::HasExpired() const {
386*6777b538SAndroid Build Coastguard Worker   return base::Time::Now() > valid_expiry();
387*6777b538SAndroid Build Coastguard Worker }
388*6777b538SAndroid Build Coastguard Worker 
EqualsExcludingChain(const X509Certificate * other) const389*6777b538SAndroid Build Coastguard Worker bool X509Certificate::EqualsExcludingChain(const X509Certificate* other) const {
390*6777b538SAndroid Build Coastguard Worker   return x509_util::CryptoBufferEqual(cert_buffer_.get(),
391*6777b538SAndroid Build Coastguard Worker                                       other->cert_buffer_.get());
392*6777b538SAndroid Build Coastguard Worker }
393*6777b538SAndroid Build Coastguard Worker 
EqualsIncludingChain(const X509Certificate * other) const394*6777b538SAndroid Build Coastguard Worker bool X509Certificate::EqualsIncludingChain(const X509Certificate* other) const {
395*6777b538SAndroid Build Coastguard Worker   if (intermediate_ca_certs_.size() != other->intermediate_ca_certs_.size() ||
396*6777b538SAndroid Build Coastguard Worker       !EqualsExcludingChain(other)) {
397*6777b538SAndroid Build Coastguard Worker     return false;
398*6777b538SAndroid Build Coastguard Worker   }
399*6777b538SAndroid Build Coastguard Worker   for (size_t i = 0; i < intermediate_ca_certs_.size(); ++i) {
400*6777b538SAndroid Build Coastguard Worker     if (!x509_util::CryptoBufferEqual(intermediate_ca_certs_[i].get(),
401*6777b538SAndroid Build Coastguard Worker                                       other->intermediate_ca_certs_[i].get())) {
402*6777b538SAndroid Build Coastguard Worker       return false;
403*6777b538SAndroid Build Coastguard Worker     }
404*6777b538SAndroid Build Coastguard Worker   }
405*6777b538SAndroid Build Coastguard Worker   return true;
406*6777b538SAndroid Build Coastguard Worker }
407*6777b538SAndroid Build Coastguard Worker 
IsIssuedByEncoded(const std::vector<std::string> & valid_issuers) const408*6777b538SAndroid Build Coastguard Worker bool X509Certificate::IsIssuedByEncoded(
409*6777b538SAndroid Build Coastguard Worker     const std::vector<std::string>& valid_issuers) const {
410*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> normalized_issuers;
411*6777b538SAndroid Build Coastguard Worker   bssl::CertErrors errors;
412*6777b538SAndroid Build Coastguard Worker   for (const auto& raw_issuer : valid_issuers) {
413*6777b538SAndroid Build Coastguard Worker     bssl::der::Input issuer_value;
414*6777b538SAndroid Build Coastguard Worker     std::string normalized_issuer;
415*6777b538SAndroid Build Coastguard Worker     if (!ParseSequenceValue(bssl::der::Input(raw_issuer), &issuer_value) ||
416*6777b538SAndroid Build Coastguard Worker         !NormalizeName(issuer_value, &normalized_issuer, &errors)) {
417*6777b538SAndroid Build Coastguard Worker       continue;
418*6777b538SAndroid Build Coastguard Worker     }
419*6777b538SAndroid Build Coastguard Worker     normalized_issuers.push_back(std::move(normalized_issuer));
420*6777b538SAndroid Build Coastguard Worker   }
421*6777b538SAndroid Build Coastguard Worker 
422*6777b538SAndroid Build Coastguard Worker   std::string normalized_cert_issuer;
423*6777b538SAndroid Build Coastguard Worker   if (!GetNormalizedCertIssuer(cert_buffer_.get(), &normalized_cert_issuer))
424*6777b538SAndroid Build Coastguard Worker     return false;
425*6777b538SAndroid Build Coastguard Worker   if (base::Contains(normalized_issuers, normalized_cert_issuer))
426*6777b538SAndroid Build Coastguard Worker     return true;
427*6777b538SAndroid Build Coastguard Worker 
428*6777b538SAndroid Build Coastguard Worker   for (const auto& intermediate : intermediate_ca_certs_) {
429*6777b538SAndroid Build Coastguard Worker     if (!GetNormalizedCertIssuer(intermediate.get(), &normalized_cert_issuer))
430*6777b538SAndroid Build Coastguard Worker       return false;
431*6777b538SAndroid Build Coastguard Worker     if (base::Contains(normalized_issuers, normalized_cert_issuer))
432*6777b538SAndroid Build Coastguard Worker       return true;
433*6777b538SAndroid Build Coastguard Worker   }
434*6777b538SAndroid Build Coastguard Worker   return false;
435*6777b538SAndroid Build Coastguard Worker }
436*6777b538SAndroid Build Coastguard Worker 
437*6777b538SAndroid Build Coastguard Worker // static
VerifyHostname(std::string_view hostname,const std::vector<std::string> & cert_san_dns_names,const std::vector<std::string> & cert_san_ip_addrs)438*6777b538SAndroid Build Coastguard Worker bool X509Certificate::VerifyHostname(
439*6777b538SAndroid Build Coastguard Worker     std::string_view hostname,
440*6777b538SAndroid Build Coastguard Worker     const std::vector<std::string>& cert_san_dns_names,
441*6777b538SAndroid Build Coastguard Worker     const std::vector<std::string>& cert_san_ip_addrs) {
442*6777b538SAndroid Build Coastguard Worker   DCHECK(!hostname.empty());
443*6777b538SAndroid Build Coastguard Worker 
444*6777b538SAndroid Build Coastguard Worker   if (cert_san_dns_names.empty() && cert_san_ip_addrs.empty()) {
445*6777b538SAndroid Build Coastguard Worker     // Either a dNSName or iPAddress subjectAltName MUST be present in order
446*6777b538SAndroid Build Coastguard Worker     // to match, so fail quickly if not.
447*6777b538SAndroid Build Coastguard Worker     return false;
448*6777b538SAndroid Build Coastguard Worker   }
449*6777b538SAndroid Build Coastguard Worker 
450*6777b538SAndroid Build Coastguard Worker   // Perform name verification following http://tools.ietf.org/html/rfc6125.
451*6777b538SAndroid Build Coastguard Worker   // The terminology used in this method is as per that RFC:-
452*6777b538SAndroid Build Coastguard Worker   // Reference identifier == the host the local user/agent is intending to
453*6777b538SAndroid Build Coastguard Worker   //                         access, i.e. the thing displayed in the URL bar.
454*6777b538SAndroid Build Coastguard Worker   // Presented identifier(s) == name(s) the server knows itself as, in its cert.
455*6777b538SAndroid Build Coastguard Worker 
456*6777b538SAndroid Build Coastguard Worker   // CanonicalizeHost requires surrounding brackets to parse an IPv6 address.
457*6777b538SAndroid Build Coastguard Worker   const std::string host_or_ip = hostname.find(':') != std::string::npos
458*6777b538SAndroid Build Coastguard Worker                                      ? base::StrCat({"[", hostname, "]"})
459*6777b538SAndroid Build Coastguard Worker                                      : std::string(hostname);
460*6777b538SAndroid Build Coastguard Worker   url::CanonHostInfo host_info;
461*6777b538SAndroid Build Coastguard Worker   std::string reference_name = CanonicalizeHost(host_or_ip, &host_info);
462*6777b538SAndroid Build Coastguard Worker 
463*6777b538SAndroid Build Coastguard Worker   // If the host cannot be canonicalized, fail fast.
464*6777b538SAndroid Build Coastguard Worker   if (reference_name.empty())
465*6777b538SAndroid Build Coastguard Worker     return false;
466*6777b538SAndroid Build Coastguard Worker 
467*6777b538SAndroid Build Coastguard Worker   // Fully handle all cases where |hostname| contains an IP address.
468*6777b538SAndroid Build Coastguard Worker   if (host_info.IsIPAddress()) {
469*6777b538SAndroid Build Coastguard Worker     std::string_view ip_addr_string(
470*6777b538SAndroid Build Coastguard Worker         reinterpret_cast<const char*>(host_info.address),
471*6777b538SAndroid Build Coastguard Worker         host_info.AddressLength());
472*6777b538SAndroid Build Coastguard Worker     return base::Contains(cert_san_ip_addrs, ip_addr_string);
473*6777b538SAndroid Build Coastguard Worker   }
474*6777b538SAndroid Build Coastguard Worker 
475*6777b538SAndroid Build Coastguard Worker   // The host portion of a URL may support a variety of name resolution formats
476*6777b538SAndroid Build Coastguard Worker   // and services. However, the only supported name types in this code are IP
477*6777b538SAndroid Build Coastguard Worker   // addresses, which have been handled above via iPAddress subjectAltNames,
478*6777b538SAndroid Build Coastguard Worker   // and DNS names, via dNSName subjectAltNames.
479*6777b538SAndroid Build Coastguard Worker   // Validate that the host conforms to the DNS preferred name syntax, in
480*6777b538SAndroid Build Coastguard Worker   // either relative or absolute form, and exclude the "root" label for DNS.
481*6777b538SAndroid Build Coastguard Worker   if (reference_name == "." || !IsCanonicalizedHostCompliant(reference_name))
482*6777b538SAndroid Build Coastguard Worker     return false;
483*6777b538SAndroid Build Coastguard Worker 
484*6777b538SAndroid Build Coastguard Worker   // CanonicalizeHost does not normalize absolute vs relative DNS names. If
485*6777b538SAndroid Build Coastguard Worker   // the input name was absolute (included trailing .), normalize it as if it
486*6777b538SAndroid Build Coastguard Worker   // was relative.
487*6777b538SAndroid Build Coastguard Worker   if (reference_name.back() == '.')
488*6777b538SAndroid Build Coastguard Worker     reference_name.pop_back();
489*6777b538SAndroid Build Coastguard Worker 
490*6777b538SAndroid Build Coastguard Worker   // |reference_domain| is the remainder of |host| after the leading host
491*6777b538SAndroid Build Coastguard Worker   // component is stripped off, but includes the leading dot e.g.
492*6777b538SAndroid Build Coastguard Worker   // "www.f.com" -> ".f.com".
493*6777b538SAndroid Build Coastguard Worker   // If there is no meaningful domain part to |host| (e.g. it contains no dots)
494*6777b538SAndroid Build Coastguard Worker   // then |reference_domain| will be empty.
495*6777b538SAndroid Build Coastguard Worker   std::string_view reference_host, reference_domain;
496*6777b538SAndroid Build Coastguard Worker   SplitOnChar(reference_name, '.', &reference_host, &reference_domain);
497*6777b538SAndroid Build Coastguard Worker   bool allow_wildcards = false;
498*6777b538SAndroid Build Coastguard Worker   if (!reference_domain.empty()) {
499*6777b538SAndroid Build Coastguard Worker     DCHECK(reference_domain.starts_with("."));
500*6777b538SAndroid Build Coastguard Worker 
501*6777b538SAndroid Build Coastguard Worker     // Do not allow wildcards for public/ICANN registry controlled domains -
502*6777b538SAndroid Build Coastguard Worker     // that is, prevent *.com or *.co.uk as valid presented names, but do not
503*6777b538SAndroid Build Coastguard Worker     // prevent *.appspot.com (a private registry controlled domain).
504*6777b538SAndroid Build Coastguard Worker     // In addition, unknown top-level domains (such as 'intranet' domains or
505*6777b538SAndroid Build Coastguard Worker     // new TLDs/gTLDs not yet added to the registry controlled domain dataset)
506*6777b538SAndroid Build Coastguard Worker     // are also implicitly prevented.
507*6777b538SAndroid Build Coastguard Worker     // Because |reference_domain| must contain at least one name component that
508*6777b538SAndroid Build Coastguard Worker     // is not registry controlled, this ensures that all reference domains
509*6777b538SAndroid Build Coastguard Worker     // contain at least three domain components when using wildcards.
510*6777b538SAndroid Build Coastguard Worker     size_t registry_length =
511*6777b538SAndroid Build Coastguard Worker         registry_controlled_domains::GetCanonicalHostRegistryLength(
512*6777b538SAndroid Build Coastguard Worker             reference_name,
513*6777b538SAndroid Build Coastguard Worker             registry_controlled_domains::INCLUDE_UNKNOWN_REGISTRIES,
514*6777b538SAndroid Build Coastguard Worker             registry_controlled_domains::EXCLUDE_PRIVATE_REGISTRIES);
515*6777b538SAndroid Build Coastguard Worker 
516*6777b538SAndroid Build Coastguard Worker     // Because |reference_name| was already canonicalized, the following
517*6777b538SAndroid Build Coastguard Worker     // should never happen.
518*6777b538SAndroid Build Coastguard Worker     CHECK_NE(std::string::npos, registry_length);
519*6777b538SAndroid Build Coastguard Worker 
520*6777b538SAndroid Build Coastguard Worker     // Account for the leading dot in |reference_domain|.
521*6777b538SAndroid Build Coastguard Worker     bool is_registry_controlled =
522*6777b538SAndroid Build Coastguard Worker         registry_length != 0 &&
523*6777b538SAndroid Build Coastguard Worker         registry_length == (reference_domain.size() - 1);
524*6777b538SAndroid Build Coastguard Worker 
525*6777b538SAndroid Build Coastguard Worker     // Additionally, do not attempt wildcard matching for purely numeric
526*6777b538SAndroid Build Coastguard Worker     // hostnames.
527*6777b538SAndroid Build Coastguard Worker     allow_wildcards =
528*6777b538SAndroid Build Coastguard Worker         !is_registry_controlled &&
529*6777b538SAndroid Build Coastguard Worker         reference_name.find_first_not_of("0123456789.") != std::string::npos;
530*6777b538SAndroid Build Coastguard Worker   }
531*6777b538SAndroid Build Coastguard Worker 
532*6777b538SAndroid Build Coastguard Worker   // Now step through the DNS names doing wild card comparison (if necessary)
533*6777b538SAndroid Build Coastguard Worker   // on each against the reference name.
534*6777b538SAndroid Build Coastguard Worker   for (const auto& cert_san_dns_name : cert_san_dns_names) {
535*6777b538SAndroid Build Coastguard Worker     // Catch badly corrupt cert names up front.
536*6777b538SAndroid Build Coastguard Worker     if (cert_san_dns_name.empty() ||
537*6777b538SAndroid Build Coastguard Worker         cert_san_dns_name.find('\0') != std::string::npos) {
538*6777b538SAndroid Build Coastguard Worker       continue;
539*6777b538SAndroid Build Coastguard Worker     }
540*6777b538SAndroid Build Coastguard Worker     std::string presented_name(base::ToLowerASCII(cert_san_dns_name));
541*6777b538SAndroid Build Coastguard Worker 
542*6777b538SAndroid Build Coastguard Worker     // Remove trailing dot, if any.
543*6777b538SAndroid Build Coastguard Worker     if (*presented_name.rbegin() == '.')
544*6777b538SAndroid Build Coastguard Worker       presented_name.resize(presented_name.length() - 1);
545*6777b538SAndroid Build Coastguard Worker 
546*6777b538SAndroid Build Coastguard Worker     // The hostname must be at least as long as the cert name it is matching,
547*6777b538SAndroid Build Coastguard Worker     // as we require the wildcard (if present) to match at least one character.
548*6777b538SAndroid Build Coastguard Worker     if (presented_name.length() > reference_name.length())
549*6777b538SAndroid Build Coastguard Worker       continue;
550*6777b538SAndroid Build Coastguard Worker 
551*6777b538SAndroid Build Coastguard Worker     std::string_view presented_host, presented_domain;
552*6777b538SAndroid Build Coastguard Worker     SplitOnChar(presented_name, '.', &presented_host, &presented_domain);
553*6777b538SAndroid Build Coastguard Worker 
554*6777b538SAndroid Build Coastguard Worker     if (presented_domain != reference_domain)
555*6777b538SAndroid Build Coastguard Worker       continue;
556*6777b538SAndroid Build Coastguard Worker 
557*6777b538SAndroid Build Coastguard Worker     if (presented_host != "*") {
558*6777b538SAndroid Build Coastguard Worker       if (presented_host == reference_host)
559*6777b538SAndroid Build Coastguard Worker         return true;
560*6777b538SAndroid Build Coastguard Worker       continue;
561*6777b538SAndroid Build Coastguard Worker     }
562*6777b538SAndroid Build Coastguard Worker 
563*6777b538SAndroid Build Coastguard Worker     if (!allow_wildcards)
564*6777b538SAndroid Build Coastguard Worker       continue;
565*6777b538SAndroid Build Coastguard Worker 
566*6777b538SAndroid Build Coastguard Worker     return true;
567*6777b538SAndroid Build Coastguard Worker   }
568*6777b538SAndroid Build Coastguard Worker   return false;
569*6777b538SAndroid Build Coastguard Worker }
570*6777b538SAndroid Build Coastguard Worker 
VerifyNameMatch(std::string_view hostname) const571*6777b538SAndroid Build Coastguard Worker bool X509Certificate::VerifyNameMatch(std::string_view hostname) const {
572*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> dns_names, ip_addrs;
573*6777b538SAndroid Build Coastguard Worker   GetSubjectAltName(&dns_names, &ip_addrs);
574*6777b538SAndroid Build Coastguard Worker   return VerifyHostname(hostname, dns_names, ip_addrs);
575*6777b538SAndroid Build Coastguard Worker }
576*6777b538SAndroid Build Coastguard Worker 
577*6777b538SAndroid Build Coastguard Worker // static
GetPEMEncodedFromDER(std::string_view der_encoded,std::string * pem_encoded)578*6777b538SAndroid Build Coastguard Worker bool X509Certificate::GetPEMEncodedFromDER(std::string_view der_encoded,
579*6777b538SAndroid Build Coastguard Worker                                            std::string* pem_encoded) {
580*6777b538SAndroid Build Coastguard Worker   if (der_encoded.empty())
581*6777b538SAndroid Build Coastguard Worker     return false;
582*6777b538SAndroid Build Coastguard Worker 
583*6777b538SAndroid Build Coastguard Worker   *pem_encoded = bssl::PEMEncode(der_encoded, "CERTIFICATE");
584*6777b538SAndroid Build Coastguard Worker   return true;
585*6777b538SAndroid Build Coastguard Worker }
586*6777b538SAndroid Build Coastguard Worker 
587*6777b538SAndroid Build Coastguard Worker // static
GetPEMEncoded(const CRYPTO_BUFFER * cert_buffer,std::string * pem_encoded)588*6777b538SAndroid Build Coastguard Worker bool X509Certificate::GetPEMEncoded(const CRYPTO_BUFFER* cert_buffer,
589*6777b538SAndroid Build Coastguard Worker                                     std::string* pem_encoded) {
590*6777b538SAndroid Build Coastguard Worker   return GetPEMEncodedFromDER(x509_util::CryptoBufferAsStringPiece(cert_buffer),
591*6777b538SAndroid Build Coastguard Worker                               pem_encoded);
592*6777b538SAndroid Build Coastguard Worker }
593*6777b538SAndroid Build Coastguard Worker 
GetPEMEncodedChain(std::vector<std::string> * pem_encoded) const594*6777b538SAndroid Build Coastguard Worker bool X509Certificate::GetPEMEncodedChain(
595*6777b538SAndroid Build Coastguard Worker     std::vector<std::string>* pem_encoded) const {
596*6777b538SAndroid Build Coastguard Worker   std::vector<std::string> encoded_chain;
597*6777b538SAndroid Build Coastguard Worker   std::string pem_data;
598*6777b538SAndroid Build Coastguard Worker   if (!GetPEMEncoded(cert_buffer(), &pem_data))
599*6777b538SAndroid Build Coastguard Worker     return false;
600*6777b538SAndroid Build Coastguard Worker   encoded_chain.push_back(pem_data);
601*6777b538SAndroid Build Coastguard Worker   for (const auto& intermediate_ca_cert : intermediate_ca_certs_) {
602*6777b538SAndroid Build Coastguard Worker     if (!GetPEMEncoded(intermediate_ca_cert.get(), &pem_data))
603*6777b538SAndroid Build Coastguard Worker       return false;
604*6777b538SAndroid Build Coastguard Worker     encoded_chain.push_back(pem_data);
605*6777b538SAndroid Build Coastguard Worker   }
606*6777b538SAndroid Build Coastguard Worker   pem_encoded->swap(encoded_chain);
607*6777b538SAndroid Build Coastguard Worker   return true;
608*6777b538SAndroid Build Coastguard Worker }
609*6777b538SAndroid Build Coastguard Worker 
610*6777b538SAndroid Build Coastguard Worker // static
GetPublicKeyInfo(const CRYPTO_BUFFER * cert_buffer,size_t * size_bits,PublicKeyType * type)611*6777b538SAndroid Build Coastguard Worker void X509Certificate::GetPublicKeyInfo(const CRYPTO_BUFFER* cert_buffer,
612*6777b538SAndroid Build Coastguard Worker                                        size_t* size_bits,
613*6777b538SAndroid Build Coastguard Worker                                        PublicKeyType* type) {
614*6777b538SAndroid Build Coastguard Worker   *type = kPublicKeyTypeUnknown;
615*6777b538SAndroid Build Coastguard Worker   *size_bits = 0;
616*6777b538SAndroid Build Coastguard Worker 
617*6777b538SAndroid Build Coastguard Worker   std::string_view spki;
618*6777b538SAndroid Build Coastguard Worker   if (!asn1::ExtractSPKIFromDERCert(
619*6777b538SAndroid Build Coastguard Worker           std::string_view(
620*6777b538SAndroid Build Coastguard Worker               reinterpret_cast<const char*>(CRYPTO_BUFFER_data(cert_buffer)),
621*6777b538SAndroid Build Coastguard Worker               CRYPTO_BUFFER_len(cert_buffer)),
622*6777b538SAndroid Build Coastguard Worker           &spki)) {
623*6777b538SAndroid Build Coastguard Worker     return;
624*6777b538SAndroid Build Coastguard Worker   }
625*6777b538SAndroid Build Coastguard Worker 
626*6777b538SAndroid Build Coastguard Worker   bssl::UniquePtr<EVP_PKEY> pkey;
627*6777b538SAndroid Build Coastguard Worker   crypto::OpenSSLErrStackTracer err_tracer(FROM_HERE);
628*6777b538SAndroid Build Coastguard Worker   CBS cbs;
629*6777b538SAndroid Build Coastguard Worker   CBS_init(&cbs, reinterpret_cast<const uint8_t*>(spki.data()), spki.size());
630*6777b538SAndroid Build Coastguard Worker   pkey.reset(EVP_parse_public_key(&cbs));
631*6777b538SAndroid Build Coastguard Worker   if (!pkey)
632*6777b538SAndroid Build Coastguard Worker     return;
633*6777b538SAndroid Build Coastguard Worker 
634*6777b538SAndroid Build Coastguard Worker   switch (EVP_PKEY_id(pkey.get())) {
635*6777b538SAndroid Build Coastguard Worker     case EVP_PKEY_RSA:
636*6777b538SAndroid Build Coastguard Worker       *type = kPublicKeyTypeRSA;
637*6777b538SAndroid Build Coastguard Worker       break;
638*6777b538SAndroid Build Coastguard Worker     case EVP_PKEY_DSA:
639*6777b538SAndroid Build Coastguard Worker       *type = kPublicKeyTypeDSA;
640*6777b538SAndroid Build Coastguard Worker       break;
641*6777b538SAndroid Build Coastguard Worker     case EVP_PKEY_EC:
642*6777b538SAndroid Build Coastguard Worker       *type = kPublicKeyTypeECDSA;
643*6777b538SAndroid Build Coastguard Worker       break;
644*6777b538SAndroid Build Coastguard Worker     case EVP_PKEY_DH:
645*6777b538SAndroid Build Coastguard Worker       *type = kPublicKeyTypeDH;
646*6777b538SAndroid Build Coastguard Worker       break;
647*6777b538SAndroid Build Coastguard Worker   }
648*6777b538SAndroid Build Coastguard Worker   *size_bits = base::saturated_cast<size_t>(EVP_PKEY_bits(pkey.get()));
649*6777b538SAndroid Build Coastguard Worker }
650*6777b538SAndroid Build Coastguard Worker 
651*6777b538SAndroid Build Coastguard Worker // static
652*6777b538SAndroid Build Coastguard Worker std::vector<bssl::UniquePtr<CRYPTO_BUFFER>>
CreateCertBuffersFromBytes(base::span<const uint8_t> data,Format format)653*6777b538SAndroid Build Coastguard Worker X509Certificate::CreateCertBuffersFromBytes(base::span<const uint8_t> data,
654*6777b538SAndroid Build Coastguard Worker                                             Format format) {
655*6777b538SAndroid Build Coastguard Worker   std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> results;
656*6777b538SAndroid Build Coastguard Worker 
657*6777b538SAndroid Build Coastguard Worker   switch (format) {
658*6777b538SAndroid Build Coastguard Worker     case FORMAT_SINGLE_CERTIFICATE: {
659*6777b538SAndroid Build Coastguard Worker       bssl::UniquePtr<CRYPTO_BUFFER> handle =
660*6777b538SAndroid Build Coastguard Worker           CreateCertBufferFromBytesWithSanityCheck(data);
661*6777b538SAndroid Build Coastguard Worker       if (handle)
662*6777b538SAndroid Build Coastguard Worker         results.push_back(std::move(handle));
663*6777b538SAndroid Build Coastguard Worker       break;
664*6777b538SAndroid Build Coastguard Worker     }
665*6777b538SAndroid Build Coastguard Worker     case FORMAT_PKCS7: {
666*6777b538SAndroid Build Coastguard Worker       x509_util::CreateCertBuffersFromPKCS7Bytes(data, &results);
667*6777b538SAndroid Build Coastguard Worker       break;
668*6777b538SAndroid Build Coastguard Worker     }
669*6777b538SAndroid Build Coastguard Worker     default: {
670*6777b538SAndroid Build Coastguard Worker       NOTREACHED() << "Certificate format " << format << " unimplemented";
671*6777b538SAndroid Build Coastguard Worker       break;
672*6777b538SAndroid Build Coastguard Worker     }
673*6777b538SAndroid Build Coastguard Worker   }
674*6777b538SAndroid Build Coastguard Worker 
675*6777b538SAndroid Build Coastguard Worker   return results;
676*6777b538SAndroid Build Coastguard Worker }
677*6777b538SAndroid Build Coastguard Worker 
678*6777b538SAndroid Build Coastguard Worker // static
CalculateFingerprint256(const CRYPTO_BUFFER * cert)679*6777b538SAndroid Build Coastguard Worker SHA256HashValue X509Certificate::CalculateFingerprint256(
680*6777b538SAndroid Build Coastguard Worker     const CRYPTO_BUFFER* cert) {
681*6777b538SAndroid Build Coastguard Worker   SHA256HashValue sha256;
682*6777b538SAndroid Build Coastguard Worker 
683*6777b538SAndroid Build Coastguard Worker   SHA256(CRYPTO_BUFFER_data(cert), CRYPTO_BUFFER_len(cert), sha256.data);
684*6777b538SAndroid Build Coastguard Worker   return sha256;
685*6777b538SAndroid Build Coastguard Worker }
686*6777b538SAndroid Build Coastguard Worker 
CalculateChainFingerprint256() const687*6777b538SAndroid Build Coastguard Worker SHA256HashValue X509Certificate::CalculateChainFingerprint256() const {
688*6777b538SAndroid Build Coastguard Worker   SHA256HashValue sha256;
689*6777b538SAndroid Build Coastguard Worker   memset(sha256.data, 0, sizeof(sha256.data));
690*6777b538SAndroid Build Coastguard Worker 
691*6777b538SAndroid Build Coastguard Worker   SHA256_CTX sha256_ctx;
692*6777b538SAndroid Build Coastguard Worker   SHA256_Init(&sha256_ctx);
693*6777b538SAndroid Build Coastguard Worker   SHA256_Update(&sha256_ctx, CRYPTO_BUFFER_data(cert_buffer_.get()),
694*6777b538SAndroid Build Coastguard Worker                 CRYPTO_BUFFER_len(cert_buffer_.get()));
695*6777b538SAndroid Build Coastguard Worker   for (const auto& cert : intermediate_ca_certs_) {
696*6777b538SAndroid Build Coastguard Worker     SHA256_Update(&sha256_ctx, CRYPTO_BUFFER_data(cert.get()),
697*6777b538SAndroid Build Coastguard Worker                   CRYPTO_BUFFER_len(cert.get()));
698*6777b538SAndroid Build Coastguard Worker   }
699*6777b538SAndroid Build Coastguard Worker   SHA256_Final(sha256.data, &sha256_ctx);
700*6777b538SAndroid Build Coastguard Worker 
701*6777b538SAndroid Build Coastguard Worker   return sha256;
702*6777b538SAndroid Build Coastguard Worker }
703*6777b538SAndroid Build Coastguard Worker 
704*6777b538SAndroid Build Coastguard Worker // static
IsSelfSigned(CRYPTO_BUFFER * cert_buffer)705*6777b538SAndroid Build Coastguard Worker bool X509Certificate::IsSelfSigned(CRYPTO_BUFFER* cert_buffer) {
706*6777b538SAndroid Build Coastguard Worker   std::shared_ptr<const bssl::ParsedCertificate> parsed_cert =
707*6777b538SAndroid Build Coastguard Worker       bssl::ParsedCertificate::Create(
708*6777b538SAndroid Build Coastguard Worker           bssl::UpRef(cert_buffer), x509_util::DefaultParseCertificateOptions(),
709*6777b538SAndroid Build Coastguard Worker           /*errors=*/nullptr);
710*6777b538SAndroid Build Coastguard Worker   if (!parsed_cert) {
711*6777b538SAndroid Build Coastguard Worker     return false;
712*6777b538SAndroid Build Coastguard Worker   }
713*6777b538SAndroid Build Coastguard Worker   return VerifyCertificateIsSelfSigned(*parsed_cert, /*cache=*/nullptr,
714*6777b538SAndroid Build Coastguard Worker                                        /*errors=*/nullptr);
715*6777b538SAndroid Build Coastguard Worker }
716*6777b538SAndroid Build Coastguard Worker 
X509Certificate(ParsedFields parsed,bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)717*6777b538SAndroid Build Coastguard Worker X509Certificate::X509Certificate(
718*6777b538SAndroid Build Coastguard Worker     ParsedFields parsed,
719*6777b538SAndroid Build Coastguard Worker     bssl::UniquePtr<CRYPTO_BUFFER> cert_buffer,
720*6777b538SAndroid Build Coastguard Worker     std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)
721*6777b538SAndroid Build Coastguard Worker     : parsed_(std::move(parsed)),
722*6777b538SAndroid Build Coastguard Worker       cert_buffer_(std::move(cert_buffer)),
723*6777b538SAndroid Build Coastguard Worker       intermediate_ca_certs_(std::move(intermediates)) {}
724*6777b538SAndroid Build Coastguard Worker 
X509Certificate(const X509Certificate & other,std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)725*6777b538SAndroid Build Coastguard Worker X509Certificate::X509Certificate(
726*6777b538SAndroid Build Coastguard Worker     const X509Certificate& other,
727*6777b538SAndroid Build Coastguard Worker     std::vector<bssl::UniquePtr<CRYPTO_BUFFER>> intermediates)
728*6777b538SAndroid Build Coastguard Worker     : parsed_(other.parsed_),
729*6777b538SAndroid Build Coastguard Worker       cert_buffer_(bssl::UpRef(other.cert_buffer_)),
730*6777b538SAndroid Build Coastguard Worker       intermediate_ca_certs_(std::move(intermediates)) {}
731*6777b538SAndroid Build Coastguard Worker 
732*6777b538SAndroid Build Coastguard Worker X509Certificate::~X509Certificate() = default;
733*6777b538SAndroid Build Coastguard Worker 
734*6777b538SAndroid Build Coastguard Worker X509Certificate::ParsedFields::ParsedFields() = default;
735*6777b538SAndroid Build Coastguard Worker X509Certificate::ParsedFields::ParsedFields(const ParsedFields&) = default;
736*6777b538SAndroid Build Coastguard Worker X509Certificate::ParsedFields::ParsedFields(ParsedFields&&) = default;
737*6777b538SAndroid Build Coastguard Worker X509Certificate::ParsedFields::~ParsedFields() = default;
738*6777b538SAndroid Build Coastguard Worker 
Initialize(const CRYPTO_BUFFER * cert_buffer,X509Certificate::UnsafeCreateOptions options)739*6777b538SAndroid Build Coastguard Worker bool X509Certificate::ParsedFields::Initialize(
740*6777b538SAndroid Build Coastguard Worker     const CRYPTO_BUFFER* cert_buffer,
741*6777b538SAndroid Build Coastguard Worker     X509Certificate::UnsafeCreateOptions options) {
742*6777b538SAndroid Build Coastguard Worker   bssl::der::Input tbs_certificate_tlv;
743*6777b538SAndroid Build Coastguard Worker   bssl::der::Input signature_algorithm_tlv;
744*6777b538SAndroid Build Coastguard Worker   bssl::der::BitString signature_value;
745*6777b538SAndroid Build Coastguard Worker 
746*6777b538SAndroid Build Coastguard Worker   if (!bssl::ParseCertificate(bssl::der::Input(CRYPTO_BUFFER_data(cert_buffer),
747*6777b538SAndroid Build Coastguard Worker                                                CRYPTO_BUFFER_len(cert_buffer)),
748*6777b538SAndroid Build Coastguard Worker                               &tbs_certificate_tlv, &signature_algorithm_tlv,
749*6777b538SAndroid Build Coastguard Worker                               &signature_value, nullptr)) {
750*6777b538SAndroid Build Coastguard Worker     return false;
751*6777b538SAndroid Build Coastguard Worker   }
752*6777b538SAndroid Build Coastguard Worker 
753*6777b538SAndroid Build Coastguard Worker   bssl::ParsedTbsCertificate tbs;
754*6777b538SAndroid Build Coastguard Worker   if (!ParseTbsCertificate(tbs_certificate_tlv,
755*6777b538SAndroid Build Coastguard Worker                            x509_util::DefaultParseCertificateOptions(), &tbs,
756*6777b538SAndroid Build Coastguard Worker                            nullptr))
757*6777b538SAndroid Build Coastguard Worker     return false;
758*6777b538SAndroid Build Coastguard Worker 
759*6777b538SAndroid Build Coastguard Worker   CertPrincipal::PrintableStringHandling printable_string_handling =
760*6777b538SAndroid Build Coastguard Worker       options.printable_string_is_utf8
761*6777b538SAndroid Build Coastguard Worker           ? CertPrincipal::PrintableStringHandling::kAsUTF8Hack
762*6777b538SAndroid Build Coastguard Worker           : CertPrincipal::PrintableStringHandling::kDefault;
763*6777b538SAndroid Build Coastguard Worker   if (!subject_.ParseDistinguishedName(tbs.subject_tlv,
764*6777b538SAndroid Build Coastguard Worker                                        printable_string_handling) ||
765*6777b538SAndroid Build Coastguard Worker       !issuer_.ParseDistinguishedName(tbs.issuer_tlv,
766*6777b538SAndroid Build Coastguard Worker                                       printable_string_handling)) {
767*6777b538SAndroid Build Coastguard Worker     return false;
768*6777b538SAndroid Build Coastguard Worker   }
769*6777b538SAndroid Build Coastguard Worker 
770*6777b538SAndroid Build Coastguard Worker   if (!GeneralizedTimeToTime(tbs.validity_not_before, &valid_start_) ||
771*6777b538SAndroid Build Coastguard Worker       !GeneralizedTimeToTime(tbs.validity_not_after, &valid_expiry_)) {
772*6777b538SAndroid Build Coastguard Worker     return false;
773*6777b538SAndroid Build Coastguard Worker   }
774*6777b538SAndroid Build Coastguard Worker   serial_number_ = tbs.serial_number.AsString();
775*6777b538SAndroid Build Coastguard Worker   return true;
776*6777b538SAndroid Build Coastguard Worker }
777*6777b538SAndroid Build Coastguard Worker 
778*6777b538SAndroid Build Coastguard Worker }  // namespace net
779