1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "net/base/x509_certificate.h"
6
7 #include "base/lazy_instance.h"
8 #include "base/logging.h"
9 #include "base/pickle.h"
10 #include "base/sha1.h"
11 #include "base/string_tokenizer.h"
12 #include "base/string_util.h"
13 #include "base/utf_string_conversions.h"
14 #include "crypto/rsa_private_key.h"
15 #include "crypto/scoped_capi_types.h"
16 #include "net/base/asn1_util.h"
17 #include "net/base/cert_status_flags.h"
18 #include "net/base/cert_verify_result.h"
19 #include "net/base/ev_root_ca_metadata.h"
20 #include "net/base/net_errors.h"
21 #include "net/base/scoped_cert_chain_context.h"
22 #include "net/base/test_root_certs.h"
23 #include "net/base/x509_certificate_known_roots_win.h"
24
25 #pragma comment(lib, "crypt32.lib")
26
27 using base::Time;
28
29 namespace net {
30
31 namespace {
32
33 typedef crypto::ScopedCAPIHandle<
34 HCERTSTORE,
35 crypto::CAPIDestroyerWithFlags<HCERTSTORE,
36 CertCloseStore, 0> > ScopedHCERTSTORE;
37
38 struct FreeChainEngineFunctor {
operator ()net::__anon43786bc30111::FreeChainEngineFunctor39 void operator()(HCERTCHAINENGINE engine) const {
40 if (engine)
41 CertFreeCertificateChainEngine(engine);
42 }
43 };
44
45 typedef crypto::ScopedCAPIHandle<HCERTCHAINENGINE, FreeChainEngineFunctor>
46 ScopedHCERTCHAINENGINE;
47
48 //-----------------------------------------------------------------------------
49
50 // TODO(wtc): This is a copy of the MapSecurityError function in
51 // ssl_client_socket_win.cc. Another function that maps Windows error codes
52 // to our network error codes is WinInetUtil::OSErrorToNetError. We should
53 // eliminate the code duplication.
MapSecurityError(SECURITY_STATUS err)54 int MapSecurityError(SECURITY_STATUS err) {
55 // There are numerous security error codes, but these are the ones we thus
56 // far find interesting.
57 switch (err) {
58 case SEC_E_WRONG_PRINCIPAL: // Schannel
59 case CERT_E_CN_NO_MATCH: // CryptoAPI
60 return ERR_CERT_COMMON_NAME_INVALID;
61 case SEC_E_UNTRUSTED_ROOT: // Schannel
62 case CERT_E_UNTRUSTEDROOT: // CryptoAPI
63 return ERR_CERT_AUTHORITY_INVALID;
64 case SEC_E_CERT_EXPIRED: // Schannel
65 case CERT_E_EXPIRED: // CryptoAPI
66 return ERR_CERT_DATE_INVALID;
67 case CRYPT_E_NO_REVOCATION_CHECK:
68 return ERR_CERT_NO_REVOCATION_MECHANISM;
69 case CRYPT_E_REVOCATION_OFFLINE:
70 return ERR_CERT_UNABLE_TO_CHECK_REVOCATION;
71 case CRYPT_E_REVOKED: // Schannel and CryptoAPI
72 return ERR_CERT_REVOKED;
73 case SEC_E_CERT_UNKNOWN:
74 case CERT_E_ROLE:
75 return ERR_CERT_INVALID;
76 case CERT_E_WRONG_USAGE:
77 // TODO(wtc): Should we add ERR_CERT_WRONG_USAGE?
78 return ERR_CERT_INVALID;
79 // We received an unexpected_message or illegal_parameter alert message
80 // from the server.
81 case SEC_E_ILLEGAL_MESSAGE:
82 return ERR_SSL_PROTOCOL_ERROR;
83 case SEC_E_ALGORITHM_MISMATCH:
84 return ERR_SSL_VERSION_OR_CIPHER_MISMATCH;
85 case SEC_E_INVALID_HANDLE:
86 return ERR_UNEXPECTED;
87 case SEC_E_OK:
88 return OK;
89 default:
90 LOG(WARNING) << "Unknown error " << err << " mapped to net::ERR_FAILED";
91 return ERR_FAILED;
92 }
93 }
94
95 // Map the errors in the chain_context->TrustStatus.dwErrorStatus returned by
96 // CertGetCertificateChain to our certificate status flags.
MapCertChainErrorStatusToCertStatus(DWORD error_status)97 int MapCertChainErrorStatusToCertStatus(DWORD error_status) {
98 int cert_status = 0;
99
100 // We don't include CERT_TRUST_IS_NOT_TIME_NESTED because it's obsolete and
101 // we wouldn't consider it an error anyway
102 const DWORD kDateInvalidErrors = CERT_TRUST_IS_NOT_TIME_VALID |
103 CERT_TRUST_CTL_IS_NOT_TIME_VALID;
104 if (error_status & kDateInvalidErrors)
105 cert_status |= CERT_STATUS_DATE_INVALID;
106
107 const DWORD kAuthorityInvalidErrors = CERT_TRUST_IS_UNTRUSTED_ROOT |
108 CERT_TRUST_IS_EXPLICIT_DISTRUST |
109 CERT_TRUST_IS_PARTIAL_CHAIN;
110 if (error_status & kAuthorityInvalidErrors)
111 cert_status |= CERT_STATUS_AUTHORITY_INVALID;
112
113 if ((error_status & CERT_TRUST_REVOCATION_STATUS_UNKNOWN) &&
114 !(error_status & CERT_TRUST_IS_OFFLINE_REVOCATION))
115 cert_status |= CERT_STATUS_NO_REVOCATION_MECHANISM;
116
117 if (error_status & CERT_TRUST_IS_OFFLINE_REVOCATION)
118 cert_status |= CERT_STATUS_UNABLE_TO_CHECK_REVOCATION;
119
120 if (error_status & CERT_TRUST_IS_REVOKED)
121 cert_status |= CERT_STATUS_REVOKED;
122
123 const DWORD kWrongUsageErrors = CERT_TRUST_IS_NOT_VALID_FOR_USAGE |
124 CERT_TRUST_CTL_IS_NOT_VALID_FOR_USAGE;
125 if (error_status & kWrongUsageErrors) {
126 // TODO(wtc): Should we add CERT_STATUS_WRONG_USAGE?
127 cert_status |= CERT_STATUS_INVALID;
128 }
129
130 // The rest of the errors.
131 const DWORD kCertInvalidErrors =
132 CERT_TRUST_IS_NOT_SIGNATURE_VALID |
133 CERT_TRUST_IS_CYCLIC |
134 CERT_TRUST_INVALID_EXTENSION |
135 CERT_TRUST_INVALID_POLICY_CONSTRAINTS |
136 CERT_TRUST_INVALID_BASIC_CONSTRAINTS |
137 CERT_TRUST_INVALID_NAME_CONSTRAINTS |
138 CERT_TRUST_CTL_IS_NOT_SIGNATURE_VALID |
139 CERT_TRUST_HAS_NOT_SUPPORTED_NAME_CONSTRAINT |
140 CERT_TRUST_HAS_NOT_DEFINED_NAME_CONSTRAINT |
141 CERT_TRUST_HAS_NOT_PERMITTED_NAME_CONSTRAINT |
142 CERT_TRUST_HAS_EXCLUDED_NAME_CONSTRAINT |
143 CERT_TRUST_NO_ISSUANCE_CHAIN_POLICY |
144 CERT_TRUST_HAS_NOT_SUPPORTED_CRITICAL_EXT;
145 if (error_status & kCertInvalidErrors)
146 cert_status |= CERT_STATUS_INVALID;
147
148 return cert_status;
149 }
150
ExplodedTimeToSystemTime(const base::Time::Exploded & exploded,SYSTEMTIME * system_time)151 void ExplodedTimeToSystemTime(const base::Time::Exploded& exploded,
152 SYSTEMTIME* system_time) {
153 system_time->wYear = exploded.year;
154 system_time->wMonth = exploded.month;
155 system_time->wDayOfWeek = exploded.day_of_week;
156 system_time->wDay = exploded.day_of_month;
157 system_time->wHour = exploded.hour;
158 system_time->wMinute = exploded.minute;
159 system_time->wSecond = exploded.second;
160 system_time->wMilliseconds = exploded.millisecond;
161 }
162
163 //-----------------------------------------------------------------------------
164
165 // Wrappers of malloc and free for CRYPT_DECODE_PARA, which requires the
166 // WINAPI calling convention.
MyCryptAlloc(size_t size)167 void* WINAPI MyCryptAlloc(size_t size) {
168 return malloc(size);
169 }
170
MyCryptFree(void * p)171 void WINAPI MyCryptFree(void* p) {
172 free(p);
173 }
174
175 // Decodes the cert's subjectAltName extension into a CERT_ALT_NAME_INFO
176 // structure and stores it in *output.
GetCertSubjectAltName(PCCERT_CONTEXT cert,scoped_ptr_malloc<CERT_ALT_NAME_INFO> * output)177 void GetCertSubjectAltName(PCCERT_CONTEXT cert,
178 scoped_ptr_malloc<CERT_ALT_NAME_INFO>* output) {
179 PCERT_EXTENSION extension = CertFindExtension(szOID_SUBJECT_ALT_NAME2,
180 cert->pCertInfo->cExtension,
181 cert->pCertInfo->rgExtension);
182 if (!extension)
183 return;
184
185 CRYPT_DECODE_PARA decode_para;
186 decode_para.cbSize = sizeof(decode_para);
187 decode_para.pfnAlloc = MyCryptAlloc;
188 decode_para.pfnFree = MyCryptFree;
189 CERT_ALT_NAME_INFO* alt_name_info = NULL;
190 DWORD alt_name_info_size = 0;
191 BOOL rv;
192 rv = CryptDecodeObjectEx(X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
193 szOID_SUBJECT_ALT_NAME2,
194 extension->Value.pbData,
195 extension->Value.cbData,
196 CRYPT_DECODE_ALLOC_FLAG | CRYPT_DECODE_NOCOPY_FLAG,
197 &decode_para,
198 &alt_name_info,
199 &alt_name_info_size);
200 if (rv)
201 output->reset(alt_name_info);
202 }
203
204 // Returns true if any common name in the certificate's Subject field contains
205 // a NULL character.
CertSubjectCommonNameHasNull(PCCERT_CONTEXT cert)206 bool CertSubjectCommonNameHasNull(PCCERT_CONTEXT cert) {
207 CRYPT_DECODE_PARA decode_para;
208 decode_para.cbSize = sizeof(decode_para);
209 decode_para.pfnAlloc = MyCryptAlloc;
210 decode_para.pfnFree = MyCryptFree;
211 CERT_NAME_INFO* name_info = NULL;
212 DWORD name_info_size = 0;
213 BOOL rv;
214 rv = CryptDecodeObjectEx(X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
215 X509_NAME,
216 cert->pCertInfo->Subject.pbData,
217 cert->pCertInfo->Subject.cbData,
218 CRYPT_DECODE_ALLOC_FLAG | CRYPT_DECODE_NOCOPY_FLAG,
219 &decode_para,
220 &name_info,
221 &name_info_size);
222 if (rv) {
223 scoped_ptr_malloc<CERT_NAME_INFO> scoped_name_info(name_info);
224
225 // The Subject field may have multiple common names. According to the
226 // "PKI Layer Cake" paper, CryptoAPI uses every common name in the
227 // Subject field, so we inspect every common name.
228 //
229 // From RFC 5280:
230 // X520CommonName ::= CHOICE {
231 // teletexString TeletexString (SIZE (1..ub-common-name)),
232 // printableString PrintableString (SIZE (1..ub-common-name)),
233 // universalString UniversalString (SIZE (1..ub-common-name)),
234 // utf8String UTF8String (SIZE (1..ub-common-name)),
235 // bmpString BMPString (SIZE (1..ub-common-name)) }
236 //
237 // We also check IA5String and VisibleString.
238 for (DWORD i = 0; i < name_info->cRDN; ++i) {
239 PCERT_RDN rdn = &name_info->rgRDN[i];
240 for (DWORD j = 0; j < rdn->cRDNAttr; ++j) {
241 PCERT_RDN_ATTR rdn_attr = &rdn->rgRDNAttr[j];
242 if (strcmp(rdn_attr->pszObjId, szOID_COMMON_NAME) == 0) {
243 switch (rdn_attr->dwValueType) {
244 // After the CryptoAPI ASN.1 security vulnerabilities described in
245 // http://www.microsoft.com/technet/security/Bulletin/MS09-056.mspx
246 // were patched, we get CERT_RDN_ENCODED_BLOB for a common name
247 // that contains a NULL character.
248 case CERT_RDN_ENCODED_BLOB:
249 break;
250 // Array of 8-bit characters.
251 case CERT_RDN_PRINTABLE_STRING:
252 case CERT_RDN_TELETEX_STRING:
253 case CERT_RDN_IA5_STRING:
254 case CERT_RDN_VISIBLE_STRING:
255 for (DWORD k = 0; k < rdn_attr->Value.cbData; ++k) {
256 if (rdn_attr->Value.pbData[k] == '\0')
257 return true;
258 }
259 break;
260 // Array of 16-bit characters.
261 case CERT_RDN_BMP_STRING:
262 case CERT_RDN_UTF8_STRING: {
263 DWORD num_wchars = rdn_attr->Value.cbData / 2;
264 wchar_t* common_name =
265 reinterpret_cast<wchar_t*>(rdn_attr->Value.pbData);
266 for (DWORD k = 0; k < num_wchars; ++k) {
267 if (common_name[k] == L'\0')
268 return true;
269 }
270 break;
271 }
272 // Array of ints (32-bit).
273 case CERT_RDN_UNIVERSAL_STRING: {
274 DWORD num_ints = rdn_attr->Value.cbData / 4;
275 int* common_name =
276 reinterpret_cast<int*>(rdn_attr->Value.pbData);
277 for (DWORD k = 0; k < num_ints; ++k) {
278 if (common_name[k] == 0)
279 return true;
280 }
281 break;
282 }
283 default:
284 NOTREACHED();
285 break;
286 }
287 }
288 }
289 }
290 }
291 return false;
292 }
293
294 // Saves some information about the certificate chain chain_context in
295 // *verify_result. The caller MUST initialize *verify_result before calling
296 // this function.
GetCertChainInfo(PCCERT_CHAIN_CONTEXT chain_context,CertVerifyResult * verify_result)297 void GetCertChainInfo(PCCERT_CHAIN_CONTEXT chain_context,
298 CertVerifyResult* verify_result) {
299 PCERT_SIMPLE_CHAIN first_chain = chain_context->rgpChain[0];
300 int num_elements = first_chain->cElement;
301 PCERT_CHAIN_ELEMENT* element = first_chain->rgpElement;
302
303 // Each chain starts with the end entity certificate (i = 0) and ends with
304 // the root CA certificate (i = num_elements - 1). Do not inspect the
305 // signature algorithm of the root CA certificate because the signature on
306 // the trust anchor is not important.
307 for (int i = 0; i < num_elements - 1; ++i) {
308 PCCERT_CONTEXT cert = element[i]->pCertContext;
309 const char* algorithm = cert->pCertInfo->SignatureAlgorithm.pszObjId;
310 if (strcmp(algorithm, szOID_RSA_MD5RSA) == 0) {
311 // md5WithRSAEncryption: 1.2.840.113549.1.1.4
312 verify_result->has_md5 = true;
313 if (i != 0)
314 verify_result->has_md5_ca = true;
315 } else if (strcmp(algorithm, szOID_RSA_MD2RSA) == 0) {
316 // md2WithRSAEncryption: 1.2.840.113549.1.1.2
317 verify_result->has_md2 = true;
318 if (i != 0)
319 verify_result->has_md2_ca = true;
320 } else if (strcmp(algorithm, szOID_RSA_MD4RSA) == 0) {
321 // md4WithRSAEncryption: 1.2.840.113549.1.1.3
322 verify_result->has_md4 = true;
323 }
324 }
325 }
326
327 // Decodes the cert's certificatePolicies extension into a CERT_POLICIES_INFO
328 // structure and stores it in *output.
GetCertPoliciesInfo(PCCERT_CONTEXT cert,scoped_ptr_malloc<CERT_POLICIES_INFO> * output)329 void GetCertPoliciesInfo(PCCERT_CONTEXT cert,
330 scoped_ptr_malloc<CERT_POLICIES_INFO>* output) {
331 PCERT_EXTENSION extension = CertFindExtension(szOID_CERT_POLICIES,
332 cert->pCertInfo->cExtension,
333 cert->pCertInfo->rgExtension);
334 if (!extension)
335 return;
336
337 CRYPT_DECODE_PARA decode_para;
338 decode_para.cbSize = sizeof(decode_para);
339 decode_para.pfnAlloc = MyCryptAlloc;
340 decode_para.pfnFree = MyCryptFree;
341 CERT_POLICIES_INFO* policies_info = NULL;
342 DWORD policies_info_size = 0;
343 BOOL rv;
344 rv = CryptDecodeObjectEx(X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
345 szOID_CERT_POLICIES,
346 extension->Value.pbData,
347 extension->Value.cbData,
348 CRYPT_DECODE_ALLOC_FLAG | CRYPT_DECODE_NOCOPY_FLAG,
349 &decode_para,
350 &policies_info,
351 &policies_info_size);
352 if (rv)
353 output->reset(policies_info);
354 }
355
356 // Helper function to parse a principal from a WinInet description of that
357 // principal.
ParsePrincipal(const std::string & description,CertPrincipal * principal)358 void ParsePrincipal(const std::string& description,
359 CertPrincipal* principal) {
360 // The description of the principal is a string with each LDAP value on
361 // a separate line.
362 const std::string kDelimiters("\r\n");
363
364 std::vector<std::string> common_names, locality_names, state_names,
365 country_names;
366
367 // TODO(jcampan): add business_category and serial_number.
368 const std::string kPrefixes[] = { std::string("CN="),
369 std::string("L="),
370 std::string("S="),
371 std::string("C="),
372 std::string("STREET="),
373 std::string("O="),
374 std::string("OU="),
375 std::string("DC=") };
376
377 std::vector<std::string>* values[] = {
378 &common_names, &locality_names,
379 &state_names, &country_names,
380 &(principal->street_addresses),
381 &(principal->organization_names),
382 &(principal->organization_unit_names),
383 &(principal->domain_components) };
384 DCHECK(arraysize(kPrefixes) == arraysize(values));
385
386 StringTokenizer str_tok(description, kDelimiters);
387 while (str_tok.GetNext()) {
388 std::string entry = str_tok.token();
389 for (int i = 0; i < arraysize(kPrefixes); i++) {
390 if (!entry.compare(0, kPrefixes[i].length(), kPrefixes[i])) {
391 std::string value = entry.substr(kPrefixes[i].length());
392 // Remove enclosing double-quotes if any.
393 if (value.size() >= 2 &&
394 value[0] == '"' && value[value.size() - 1] == '"')
395 value = value.substr(1, value.size() - 2);
396 values[i]->push_back(value);
397 break;
398 }
399 }
400 }
401
402 // We don't expect to have more than one CN, L, S, and C. If there is more
403 // than one entry for CN, L, S, and C, we will use the first entry. Although
404 // RFC 2818 Section 3.1 says the "most specific" CN should be used, that term
405 // has been removed in draft-saintandre-tls-server-id-check, which requires
406 // that the Subject field contains only one CN. So it is fine for us to just
407 // use the first CN.
408 std::vector<std::string>* single_value_lists[4] = {
409 &common_names, &locality_names, &state_names, &country_names };
410 std::string* single_values[4] = {
411 &principal->common_name, &principal->locality_name,
412 &principal->state_or_province_name, &principal->country_name };
413 for (int i = 0; i < arraysize(single_value_lists); ++i) {
414 int length = static_cast<int>(single_value_lists[i]->size());
415 if (!single_value_lists[i]->empty())
416 *(single_values[i]) = (*(single_value_lists[i]))[0];
417 }
418 }
419
AddCertsFromStore(HCERTSTORE store,X509Certificate::OSCertHandles * results)420 void AddCertsFromStore(HCERTSTORE store,
421 X509Certificate::OSCertHandles* results) {
422 PCCERT_CONTEXT cert = NULL;
423
424 while ((cert = CertEnumCertificatesInStore(store, cert)) != NULL) {
425 PCCERT_CONTEXT to_add = NULL;
426 if (CertAddCertificateContextToStore(
427 NULL, // The cert won't be persisted in any cert store. This breaks
428 // any association the context currently has to |store|, which
429 // allows us, the caller, to safely close |store| without
430 // releasing the cert handles.
431 cert,
432 CERT_STORE_ADD_USE_EXISTING,
433 &to_add) && to_add != NULL) {
434 // When processing stores generated from PKCS#7/PKCS#12 files, it
435 // appears that the order returned is the inverse of the order that it
436 // appeared in the file.
437 // TODO(rsleevi): Ensure this order is consistent across all Win
438 // versions
439 results->insert(results->begin(), to_add);
440 }
441 }
442 }
443
ParsePKCS7(const char * data,size_t length)444 X509Certificate::OSCertHandles ParsePKCS7(const char* data, size_t length) {
445 X509Certificate::OSCertHandles results;
446 CERT_BLOB data_blob;
447 data_blob.cbData = length;
448 data_blob.pbData = reinterpret_cast<BYTE*>(const_cast<char*>(data));
449
450 HCERTSTORE out_store = NULL;
451
452 DWORD expected_types = CERT_QUERY_CONTENT_FLAG_PKCS7_SIGNED |
453 CERT_QUERY_CONTENT_FLAG_PKCS7_SIGNED_EMBED |
454 CERT_QUERY_CONTENT_FLAG_PKCS7_UNSIGNED;
455
456 if (!CryptQueryObject(CERT_QUERY_OBJECT_BLOB, &data_blob, expected_types,
457 CERT_QUERY_FORMAT_FLAG_BINARY, 0, NULL, NULL, NULL,
458 &out_store, NULL, NULL) || out_store == NULL) {
459 return results;
460 }
461
462 AddCertsFromStore(out_store, &results);
463 CertCloseStore(out_store, CERT_CLOSE_STORE_CHECK_FLAG);
464
465 return results;
466 }
467
AppendPublicKeyHashes(PCCERT_CHAIN_CONTEXT chain,std::vector<SHA1Fingerprint> * hashes)468 void AppendPublicKeyHashes(PCCERT_CHAIN_CONTEXT chain,
469 std::vector<SHA1Fingerprint>* hashes) {
470 if (chain->cChain == 0)
471 return;
472
473 PCERT_SIMPLE_CHAIN first_chain = chain->rgpChain[0];
474 PCERT_CHAIN_ELEMENT* const element = first_chain->rgpElement;
475
476 const DWORD num_elements = first_chain->cElement;
477 for (DWORD i = 0; i < num_elements; i++) {
478 PCCERT_CONTEXT cert = element[i]->pCertContext;
479
480 base::StringPiece der_bytes(
481 reinterpret_cast<const char*>(cert->pbCertEncoded),
482 cert->cbCertEncoded);
483 base::StringPiece spki_bytes;
484 if (!asn1::ExtractSPKIFromDERCert(der_bytes, &spki_bytes))
485 continue;
486
487 SHA1Fingerprint hash;
488 base::SHA1HashBytes(reinterpret_cast<const uint8*>(spki_bytes.data()),
489 spki_bytes.size(), hash.data);
490 hashes->push_back(hash);
491 }
492 }
493
494 } // namespace
495
Initialize()496 void X509Certificate::Initialize() {
497 std::wstring subject_info;
498 std::wstring issuer_info;
499 DWORD name_size;
500 DCHECK(cert_handle_);
501 name_size = CertNameToStr(cert_handle_->dwCertEncodingType,
502 &cert_handle_->pCertInfo->Subject,
503 CERT_X500_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
504 NULL, 0);
505 name_size = CertNameToStr(cert_handle_->dwCertEncodingType,
506 &cert_handle_->pCertInfo->Subject,
507 CERT_X500_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
508 WriteInto(&subject_info, name_size), name_size);
509 name_size = CertNameToStr(cert_handle_->dwCertEncodingType,
510 &cert_handle_->pCertInfo->Issuer,
511 CERT_X500_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
512 NULL, 0);
513 name_size = CertNameToStr(cert_handle_->dwCertEncodingType,
514 &cert_handle_->pCertInfo->Issuer,
515 CERT_X500_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
516 WriteInto(&issuer_info, name_size), name_size);
517 ParsePrincipal(WideToUTF8(subject_info), &subject_);
518 ParsePrincipal(WideToUTF8(issuer_info), &issuer_);
519
520 valid_start_ = Time::FromFileTime(cert_handle_->pCertInfo->NotBefore);
521 valid_expiry_ = Time::FromFileTime(cert_handle_->pCertInfo->NotAfter);
522
523 fingerprint_ = CalculateFingerprint(cert_handle_);
524
525 const CRYPT_INTEGER_BLOB* serial = &cert_handle_->pCertInfo->SerialNumber;
526 scoped_array<uint8> serial_bytes(new uint8[serial->cbData]);
527 for (unsigned i = 0; i < serial->cbData; i++)
528 serial_bytes[i] = serial->pbData[serial->cbData - i - 1];
529 serial_number_ = std::string(
530 reinterpret_cast<char*>(serial_bytes.get()), serial->cbData);
531 // Remove leading zeros.
532 while (serial_number_.size() > 1 && serial_number_[0] == 0)
533 serial_number_ = serial_number_.substr(1, serial_number_.size() - 1);
534 }
535
536 // IsIssuedByKnownRoot returns true if the given chain is rooted at a root CA
537 // which we recognise as a standard root.
538 // static
IsIssuedByKnownRoot(PCCERT_CHAIN_CONTEXT chain_context)539 bool X509Certificate::IsIssuedByKnownRoot(PCCERT_CHAIN_CONTEXT chain_context) {
540 PCERT_SIMPLE_CHAIN first_chain = chain_context->rgpChain[0];
541 int num_elements = first_chain->cElement;
542 if (num_elements < 1)
543 return false;
544 PCERT_CHAIN_ELEMENT* element = first_chain->rgpElement;
545 PCCERT_CONTEXT cert = element[num_elements - 1]->pCertContext;
546
547 SHA1Fingerprint hash = CalculateFingerprint(cert);
548 return IsSHA1HashInSortedArray(
549 hash, &kKnownRootCertSHA1Hashes[0][0], sizeof(kKnownRootCertSHA1Hashes));
550 }
551
552 // static
CreateSelfSigned(crypto::RSAPrivateKey * key,const std::string & subject,uint32 serial_number,base::TimeDelta valid_duration)553 X509Certificate* X509Certificate::CreateSelfSigned(
554 crypto::RSAPrivateKey* key,
555 const std::string& subject,
556 uint32 serial_number,
557 base::TimeDelta valid_duration) {
558 // Get the ASN.1 encoding of the certificate subject.
559 std::wstring w_subject = ASCIIToWide(subject);
560 DWORD encoded_subject_length = 0;
561 if (!CertStrToName(
562 X509_ASN_ENCODING,
563 w_subject.c_str(),
564 CERT_X500_NAME_STR, NULL, NULL, &encoded_subject_length, NULL)) {
565 return NULL;
566 }
567
568 scoped_array<BYTE> encoded_subject(new BYTE[encoded_subject_length]);
569 if (!CertStrToName(
570 X509_ASN_ENCODING,
571 w_subject.c_str(),
572 CERT_X500_NAME_STR, NULL,
573 encoded_subject.get(),
574 &encoded_subject_length, NULL)) {
575 return NULL;
576 }
577
578 CERT_NAME_BLOB subject_name;
579 memset(&subject_name, 0, sizeof(subject_name));
580 subject_name.cbData = encoded_subject_length;
581 subject_name.pbData = encoded_subject.get();
582
583 CRYPT_ALGORITHM_IDENTIFIER sign_algo;
584 memset(&sign_algo, 0, sizeof(sign_algo));
585 sign_algo.pszObjId = szOID_RSA_SHA1RSA;
586
587 base::Time not_before = base::Time::Now();
588 base::Time not_after = not_before + valid_duration;
589 base::Time::Exploded exploded;
590
591 // Create the system time structs representing our exploded times.
592 not_before.UTCExplode(&exploded);
593 SYSTEMTIME start_time;
594 ExplodedTimeToSystemTime(exploded, &start_time);
595 not_after.UTCExplode(&exploded);
596 SYSTEMTIME end_time;
597 ExplodedTimeToSystemTime(exploded, &end_time);
598
599 PCCERT_CONTEXT cert_handle =
600 CertCreateSelfSignCertificate(key->provider(), &subject_name,
601 CERT_CREATE_SELFSIGN_NO_KEY_INFO, NULL,
602 &sign_algo, &start_time, &end_time, NULL);
603 DCHECK(cert_handle) << "Failed to create self-signed certificate: "
604 << GetLastError();
605 if (!cert_handle)
606 return NULL;
607
608 X509Certificate* cert = CreateFromHandle(cert_handle,
609 SOURCE_LONE_CERT_IMPORT,
610 OSCertHandles());
611 FreeOSCertHandle(cert_handle);
612 return cert;
613 }
614
GetDNSNames(std::vector<std::string> * dns_names) const615 void X509Certificate::GetDNSNames(std::vector<std::string>* dns_names) const {
616 dns_names->clear();
617 if (cert_handle_) {
618 scoped_ptr_malloc<CERT_ALT_NAME_INFO> alt_name_info;
619 GetCertSubjectAltName(cert_handle_, &alt_name_info);
620 CERT_ALT_NAME_INFO* alt_name = alt_name_info.get();
621 if (alt_name) {
622 int num_entries = alt_name->cAltEntry;
623 for (int i = 0; i < num_entries; i++) {
624 // dNSName is an ASN.1 IA5String representing a string of ASCII
625 // characters, so we can use WideToASCII here.
626 if (alt_name->rgAltEntry[i].dwAltNameChoice == CERT_ALT_NAME_DNS_NAME)
627 dns_names->push_back(
628 WideToASCII(alt_name->rgAltEntry[i].pwszDNSName));
629 }
630 }
631 }
632 if (dns_names->empty())
633 dns_names->push_back(subject_.common_name);
634 }
635
636 class GlobalCertStore {
637 public:
cert_store()638 HCERTSTORE cert_store() {
639 return cert_store_;
640 }
641
642 private:
643 friend struct base::DefaultLazyInstanceTraits<GlobalCertStore>;
644
GlobalCertStore()645 GlobalCertStore()
646 : cert_store_(CertOpenStore(CERT_STORE_PROV_MEMORY, 0, NULL, 0, NULL)) {
647 }
648
~GlobalCertStore()649 ~GlobalCertStore() {
650 CertCloseStore(cert_store_, 0 /* flags */);
651 }
652
653 const HCERTSTORE cert_store_;
654
655 DISALLOW_COPY_AND_ASSIGN(GlobalCertStore);
656 };
657
658 static base::LazyInstance<GlobalCertStore> g_cert_store(
659 base::LINKER_INITIALIZED);
660
661 // static
cert_store()662 HCERTSTORE X509Certificate::cert_store() {
663 return g_cert_store.Get().cert_store();
664 }
665
Verify(const std::string & hostname,int flags,CertVerifyResult * verify_result) const666 int X509Certificate::Verify(const std::string& hostname,
667 int flags,
668 CertVerifyResult* verify_result) const {
669 verify_result->Reset();
670 if (!cert_handle_)
671 return ERR_UNEXPECTED;
672
673 if (IsBlacklisted()) {
674 verify_result->cert_status |= CERT_STATUS_REVOKED;
675 return ERR_CERT_REVOKED;
676 }
677
678 // Build and validate certificate chain.
679
680 CERT_CHAIN_PARA chain_para;
681 memset(&chain_para, 0, sizeof(chain_para));
682 chain_para.cbSize = sizeof(chain_para);
683 // ExtendedKeyUsage.
684 // We still need to request szOID_SERVER_GATED_CRYPTO and szOID_SGC_NETSCAPE
685 // today because some certificate chains need them. IE also requests these
686 // two usages.
687 static const LPSTR usage[] = {
688 szOID_PKIX_KP_SERVER_AUTH,
689 szOID_SERVER_GATED_CRYPTO,
690 szOID_SGC_NETSCAPE
691 };
692 chain_para.RequestedUsage.dwType = USAGE_MATCH_TYPE_OR;
693 chain_para.RequestedUsage.Usage.cUsageIdentifier = arraysize(usage);
694 chain_para.RequestedUsage.Usage.rgpszUsageIdentifier =
695 const_cast<LPSTR*>(usage);
696 // We can set CERT_CHAIN_RETURN_LOWER_QUALITY_CONTEXTS to get more chains.
697 DWORD chain_flags = CERT_CHAIN_CACHE_END_CERT;
698 if (flags & VERIFY_REV_CHECKING_ENABLED) {
699 verify_result->cert_status |= CERT_STATUS_REV_CHECKING_ENABLED;
700 chain_flags |= CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT;
701 } else {
702 chain_flags |= CERT_CHAIN_REVOCATION_CHECK_CACHE_ONLY;
703 // EV requires revocation checking.
704 flags &= ~VERIFY_EV_CERT;
705 }
706
707 // Get the certificatePolicies extension of the certificate.
708 scoped_ptr_malloc<CERT_POLICIES_INFO> policies_info;
709 LPSTR ev_policy_oid = NULL;
710 if (flags & VERIFY_EV_CERT) {
711 GetCertPoliciesInfo(cert_handle_, &policies_info);
712 if (policies_info.get()) {
713 EVRootCAMetadata* metadata = EVRootCAMetadata::GetInstance();
714 for (DWORD i = 0; i < policies_info->cPolicyInfo; ++i) {
715 LPSTR policy_oid = policies_info->rgPolicyInfo[i].pszPolicyIdentifier;
716 if (metadata->IsEVPolicyOID(policy_oid)) {
717 ev_policy_oid = policy_oid;
718 chain_para.RequestedIssuancePolicy.dwType = USAGE_MATCH_TYPE_AND;
719 chain_para.RequestedIssuancePolicy.Usage.cUsageIdentifier = 1;
720 chain_para.RequestedIssuancePolicy.Usage.rgpszUsageIdentifier =
721 &ev_policy_oid;
722 break;
723 }
724 }
725 }
726 }
727
728 // For non-test scenarios, use the default HCERTCHAINENGINE, NULL, which
729 // corresponds to HCCE_CURRENT_USER and is is initialized as needed by
730 // crypt32. However, when testing, it is necessary to create a new
731 // HCERTCHAINENGINE and use that instead. This is because each
732 // HCERTCHAINENGINE maintains a cache of information about certificates
733 // encountered, and each test run may modify the trust status of a
734 // certificate.
735 ScopedHCERTCHAINENGINE chain_engine(NULL);
736 if (TestRootCerts::HasInstance())
737 chain_engine.reset(TestRootCerts::GetInstance()->GetChainEngine());
738
739 PCCERT_CHAIN_CONTEXT chain_context;
740 // IE passes a non-NULL pTime argument that specifies the current system
741 // time. IE passes CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT as the
742 // chain_flags argument.
743 if (!CertGetCertificateChain(
744 chain_engine,
745 cert_handle_,
746 NULL, // current system time
747 cert_handle_->hCertStore,
748 &chain_para,
749 chain_flags,
750 NULL, // reserved
751 &chain_context)) {
752 return MapSecurityError(GetLastError());
753 }
754 if (chain_context->TrustStatus.dwErrorStatus &
755 CERT_TRUST_IS_NOT_VALID_FOR_USAGE) {
756 ev_policy_oid = NULL;
757 chain_para.RequestedIssuancePolicy.Usage.cUsageIdentifier = 0;
758 chain_para.RequestedIssuancePolicy.Usage.rgpszUsageIdentifier = NULL;
759 CertFreeCertificateChain(chain_context);
760 if (!CertGetCertificateChain(
761 chain_engine,
762 cert_handle_,
763 NULL, // current system time
764 cert_handle_->hCertStore,
765 &chain_para,
766 chain_flags,
767 NULL, // reserved
768 &chain_context)) {
769 return MapSecurityError(GetLastError());
770 }
771 }
772 ScopedCertChainContext scoped_chain_context(chain_context);
773
774 GetCertChainInfo(chain_context, verify_result);
775 verify_result->cert_status |= MapCertChainErrorStatusToCertStatus(
776 chain_context->TrustStatus.dwErrorStatus);
777
778 // Treat certificates signed using broken signature algorithms as invalid.
779 if (verify_result->has_md4)
780 verify_result->cert_status |= CERT_STATUS_INVALID;
781
782 // Flag certificates signed using weak signature algorithms.
783 if (verify_result->has_md2)
784 verify_result->cert_status |= CERT_STATUS_WEAK_SIGNATURE_ALGORITHM;
785
786 // Flag certificates that have a Subject common name with a NULL character.
787 if (CertSubjectCommonNameHasNull(cert_handle_))
788 verify_result->cert_status |= CERT_STATUS_INVALID;
789
790 std::wstring wstr_hostname = ASCIIToWide(hostname);
791
792 SSL_EXTRA_CERT_CHAIN_POLICY_PARA extra_policy_para;
793 memset(&extra_policy_para, 0, sizeof(extra_policy_para));
794 extra_policy_para.cbSize = sizeof(extra_policy_para);
795 extra_policy_para.dwAuthType = AUTHTYPE_SERVER;
796 extra_policy_para.fdwChecks = 0;
797 extra_policy_para.pwszServerName =
798 const_cast<wchar_t*>(wstr_hostname.c_str());
799
800 CERT_CHAIN_POLICY_PARA policy_para;
801 memset(&policy_para, 0, sizeof(policy_para));
802 policy_para.cbSize = sizeof(policy_para);
803 policy_para.dwFlags = 0;
804 policy_para.pvExtraPolicyPara = &extra_policy_para;
805
806 CERT_CHAIN_POLICY_STATUS policy_status;
807 memset(&policy_status, 0, sizeof(policy_status));
808 policy_status.cbSize = sizeof(policy_status);
809
810 if (!CertVerifyCertificateChainPolicy(
811 CERT_CHAIN_POLICY_SSL,
812 chain_context,
813 &policy_para,
814 &policy_status)) {
815 return MapSecurityError(GetLastError());
816 }
817
818 if (policy_status.dwError) {
819 verify_result->cert_status |= MapNetErrorToCertStatus(
820 MapSecurityError(policy_status.dwError));
821
822 // CertVerifyCertificateChainPolicy reports only one error (in
823 // policy_status.dwError) if the certificate has multiple errors.
824 // CertGetCertificateChain doesn't report certificate name mismatch, so
825 // CertVerifyCertificateChainPolicy is the only function that can report
826 // certificate name mismatch.
827 //
828 // To prevent a potential certificate name mismatch from being hidden by
829 // some other certificate error, if we get any other certificate error,
830 // we call CertVerifyCertificateChainPolicy again, ignoring all other
831 // certificate errors. Both extra_policy_para.fdwChecks and
832 // policy_para.dwFlags allow us to ignore certificate errors, so we set
833 // them both.
834 if (policy_status.dwError != CERT_E_CN_NO_MATCH) {
835 const DWORD extra_ignore_flags =
836 0x00000080 | // SECURITY_FLAG_IGNORE_REVOCATION
837 0x00000100 | // SECURITY_FLAG_IGNORE_UNKNOWN_CA
838 0x00002000 | // SECURITY_FLAG_IGNORE_CERT_DATE_INVALID
839 0x00000200; // SECURITY_FLAG_IGNORE_WRONG_USAGE
840 extra_policy_para.fdwChecks = extra_ignore_flags;
841 const DWORD ignore_flags =
842 CERT_CHAIN_POLICY_IGNORE_ALL_NOT_TIME_VALID_FLAGS |
843 CERT_CHAIN_POLICY_IGNORE_INVALID_BASIC_CONSTRAINTS_FLAG |
844 CERT_CHAIN_POLICY_ALLOW_UNKNOWN_CA_FLAG |
845 CERT_CHAIN_POLICY_IGNORE_WRONG_USAGE_FLAG |
846 CERT_CHAIN_POLICY_IGNORE_INVALID_NAME_FLAG |
847 CERT_CHAIN_POLICY_IGNORE_INVALID_POLICY_FLAG |
848 CERT_CHAIN_POLICY_IGNORE_ALL_REV_UNKNOWN_FLAGS |
849 CERT_CHAIN_POLICY_ALLOW_TESTROOT_FLAG |
850 CERT_CHAIN_POLICY_TRUST_TESTROOT_FLAG |
851 CERT_CHAIN_POLICY_IGNORE_NOT_SUPPORTED_CRITICAL_EXT_FLAG |
852 CERT_CHAIN_POLICY_IGNORE_PEER_TRUST_FLAG;
853 policy_para.dwFlags = ignore_flags;
854 if (!CertVerifyCertificateChainPolicy(
855 CERT_CHAIN_POLICY_SSL,
856 chain_context,
857 &policy_para,
858 &policy_status)) {
859 return MapSecurityError(GetLastError());
860 }
861 if (policy_status.dwError) {
862 verify_result->cert_status |= MapNetErrorToCertStatus(
863 MapSecurityError(policy_status.dwError));
864 }
865 }
866 }
867
868 // TODO(wtc): Suppress CERT_STATUS_NO_REVOCATION_MECHANISM for now to be
869 // compatible with WinHTTP, which doesn't report this error (bug 3004).
870 verify_result->cert_status &= ~CERT_STATUS_NO_REVOCATION_MECHANISM;
871
872 if (IsCertStatusError(verify_result->cert_status))
873 return MapCertStatusToNetError(verify_result->cert_status);
874
875 AppendPublicKeyHashes(chain_context, &verify_result->public_key_hashes);
876 verify_result->is_issued_by_known_root = IsIssuedByKnownRoot(chain_context);
877
878 if (ev_policy_oid && CheckEV(chain_context, ev_policy_oid))
879 verify_result->cert_status |= CERT_STATUS_IS_EV;
880
881 if (IsPublicKeyBlacklisted(verify_result->public_key_hashes)) {
882 verify_result->cert_status |= CERT_STATUS_AUTHORITY_INVALID;
883 return MapCertStatusToNetError(verify_result->cert_status);
884 }
885
886 return OK;
887 }
888
GetDEREncoded(std::string * encoded)889 bool X509Certificate::GetDEREncoded(std::string* encoded) {
890 if (!cert_handle_->pbCertEncoded || !cert_handle_->cbCertEncoded)
891 return false;
892 encoded->clear();
893 encoded->append(reinterpret_cast<char*>(cert_handle_->pbCertEncoded),
894 cert_handle_->cbCertEncoded);
895 return true;
896 }
897
898 // Returns true if the certificate is an extended-validation certificate.
899 //
900 // This function checks the certificatePolicies extensions of the
901 // certificates in the certificate chain according to Section 7 (pp. 11-12)
902 // of the EV Certificate Guidelines Version 1.0 at
903 // http://cabforum.org/EV_Certificate_Guidelines.pdf.
CheckEV(PCCERT_CHAIN_CONTEXT chain_context,const char * policy_oid) const904 bool X509Certificate::CheckEV(PCCERT_CHAIN_CONTEXT chain_context,
905 const char* policy_oid) const {
906 DCHECK(chain_context->cChain != 0);
907 // If the cert doesn't match any of the policies, the
908 // CERT_TRUST_IS_NOT_VALID_FOR_USAGE bit (0x10) in
909 // chain_context->TrustStatus.dwErrorStatus is set.
910 DWORD error_status = chain_context->TrustStatus.dwErrorStatus;
911 DWORD info_status = chain_context->TrustStatus.dwInfoStatus;
912 if (!chain_context->cChain || error_status != CERT_TRUST_NO_ERROR)
913 return false;
914
915 // Check the end certificate simple chain (chain_context->rgpChain[0]).
916 // If the end certificate's certificatePolicies extension contains the
917 // EV policy OID of the root CA, return true.
918 PCERT_CHAIN_ELEMENT* element = chain_context->rgpChain[0]->rgpElement;
919 int num_elements = chain_context->rgpChain[0]->cElement;
920 if (num_elements < 2)
921 return false;
922
923 // Look up the EV policy OID of the root CA.
924 PCCERT_CONTEXT root_cert = element[num_elements - 1]->pCertContext;
925 SHA1Fingerprint fingerprint = CalculateFingerprint(root_cert);
926 EVRootCAMetadata* metadata = EVRootCAMetadata::GetInstance();
927 return metadata->HasEVPolicyOID(fingerprint, policy_oid);
928 }
929
VerifyEV() const930 bool X509Certificate::VerifyEV() const {
931 // We don't call this private method, but we do need to implement it because
932 // it's defined in x509_certificate.h. We perform EV checking in the
933 // Verify() above.
934 NOTREACHED();
935 return false;
936 }
937
938 // static
IsSameOSCert(X509Certificate::OSCertHandle a,X509Certificate::OSCertHandle b)939 bool X509Certificate::IsSameOSCert(X509Certificate::OSCertHandle a,
940 X509Certificate::OSCertHandle b) {
941 DCHECK(a && b);
942 if (a == b)
943 return true;
944 return a->cbCertEncoded == b->cbCertEncoded &&
945 memcmp(a->pbCertEncoded, b->pbCertEncoded, a->cbCertEncoded) == 0;
946 }
947
948 // static
CreateOSCertHandleFromBytes(const char * data,int length)949 X509Certificate::OSCertHandle X509Certificate::CreateOSCertHandleFromBytes(
950 const char* data, int length) {
951 OSCertHandle cert_handle = NULL;
952 if (!CertAddEncodedCertificateToStore(
953 NULL, // the cert won't be persisted in any cert store
954 X509_ASN_ENCODING,
955 reinterpret_cast<const BYTE*>(data), length,
956 CERT_STORE_ADD_USE_EXISTING,
957 &cert_handle))
958 return NULL;
959
960 return cert_handle;
961 }
962
CreateOSCertHandlesFromBytes(const char * data,int length,Format format)963 X509Certificate::OSCertHandles X509Certificate::CreateOSCertHandlesFromBytes(
964 const char* data, int length, Format format) {
965 OSCertHandles results;
966 switch (format) {
967 case FORMAT_SINGLE_CERTIFICATE: {
968 OSCertHandle handle = CreateOSCertHandleFromBytes(data, length);
969 if (handle != NULL)
970 results.push_back(handle);
971 break;
972 }
973 case FORMAT_PKCS7:
974 results = ParsePKCS7(data, length);
975 break;
976 default:
977 NOTREACHED() << "Certificate format " << format << " unimplemented";
978 break;
979 }
980
981 return results;
982 }
983
984
985 // static
DupOSCertHandle(OSCertHandle cert_handle)986 X509Certificate::OSCertHandle X509Certificate::DupOSCertHandle(
987 OSCertHandle cert_handle) {
988 return CertDuplicateCertificateContext(cert_handle);
989 }
990
991 // static
FreeOSCertHandle(OSCertHandle cert_handle)992 void X509Certificate::FreeOSCertHandle(OSCertHandle cert_handle) {
993 CertFreeCertificateContext(cert_handle);
994 }
995
996 // static
CalculateFingerprint(OSCertHandle cert)997 SHA1Fingerprint X509Certificate::CalculateFingerprint(
998 OSCertHandle cert) {
999 DCHECK(NULL != cert->pbCertEncoded);
1000 DCHECK(0 != cert->cbCertEncoded);
1001
1002 BOOL rv;
1003 SHA1Fingerprint sha1;
1004 DWORD sha1_size = sizeof(sha1.data);
1005 rv = CryptHashCertificate(NULL, CALG_SHA1, 0, cert->pbCertEncoded,
1006 cert->cbCertEncoded, sha1.data, &sha1_size);
1007 DCHECK(rv && sha1_size == sizeof(sha1.data));
1008 if (!rv)
1009 memset(sha1.data, 0, sizeof(sha1.data));
1010 return sha1;
1011 }
1012
1013 // static
1014 X509Certificate::OSCertHandle
ReadCertHandleFromPickle(const Pickle & pickle,void ** pickle_iter)1015 X509Certificate::ReadCertHandleFromPickle(const Pickle& pickle,
1016 void** pickle_iter) {
1017 const char* data;
1018 int length;
1019 if (!pickle.ReadData(pickle_iter, &data, &length))
1020 return NULL;
1021
1022 OSCertHandle cert_handle = NULL;
1023 if (!CertAddSerializedElementToStore(
1024 NULL, // the cert won't be persisted in any cert store
1025 reinterpret_cast<const BYTE*>(data), length,
1026 CERT_STORE_ADD_USE_EXISTING, 0, CERT_STORE_CERTIFICATE_CONTEXT_FLAG,
1027 NULL, reinterpret_cast<const void **>(&cert_handle))) {
1028 return NULL;
1029 }
1030
1031 return cert_handle;
1032 }
1033
1034 // static
WriteCertHandleToPickle(OSCertHandle cert_handle,Pickle * pickle)1035 bool X509Certificate::WriteCertHandleToPickle(OSCertHandle cert_handle,
1036 Pickle* pickle) {
1037 DWORD length = 0;
1038 if (!CertSerializeCertificateStoreElement(cert_handle, 0, NULL, &length))
1039 return false;
1040
1041 std::vector<BYTE> buffer(length);
1042 // Serialize |cert_handle| in a way that will preserve any extended
1043 // attributes set on the handle, such as the location to the certificate's
1044 // private key.
1045 if (!CertSerializeCertificateStoreElement(cert_handle, 0, &buffer[0],
1046 &length)) {
1047 return false;
1048 }
1049
1050 return pickle->WriteData(reinterpret_cast<const char*>(&buffer[0]),
1051 length);
1052 }
1053
1054 } // namespace net
1055