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 <CommonCrypto/CommonDigest.h>
8 #include <CoreServices/CoreServices.h>
9 #include <Security/Security.h>
10 #include <time.h>
11
12 #include <vector>
13
14 #include "base/lazy_instance.h"
15 #include "base/logging.h"
16 #include "base/mac/scoped_cftyperef.h"
17 #include "base/memory/singleton.h"
18 #include "base/pickle.h"
19 #include "base/sha1.h"
20 #include "base/sys_string_conversions.h"
21 #include "crypto/cssm_init.h"
22 #include "crypto/nss_util.h"
23 #include "crypto/rsa_private_key.h"
24 #include "net/base/asn1_util.h"
25 #include "net/base/cert_status_flags.h"
26 #include "net/base/cert_verify_result.h"
27 #include "net/base/net_errors.h"
28 #include "net/base/test_root_certs.h"
29 #include "net/base/x509_certificate_known_roots_mac.h"
30 #include "third_party/apple_apsl/cssmapplePriv.h"
31 #include "third_party/nss/mozilla/security/nss/lib/certdb/cert.h"
32
33 using base::mac::ScopedCFTypeRef;
34 using base::Time;
35
36 namespace net {
37
38 namespace {
39
40 typedef OSStatus (*SecTrustCopyExtendedResultFuncPtr)(SecTrustRef,
41 CFDictionaryRef*);
42
NetErrorFromOSStatus(OSStatus status)43 int NetErrorFromOSStatus(OSStatus status) {
44 switch (status) {
45 case noErr:
46 return OK;
47 case errSecNotAvailable:
48 case errSecNoCertificateModule:
49 case errSecNoPolicyModule:
50 return ERR_NOT_IMPLEMENTED;
51 case errSecAuthFailed:
52 return ERR_ACCESS_DENIED;
53 default:
54 LOG(ERROR) << "Unknown error " << status << " mapped to ERR_FAILED";
55 return ERR_FAILED;
56 }
57 }
58
CertStatusFromOSStatus(OSStatus status)59 int CertStatusFromOSStatus(OSStatus status) {
60 switch (status) {
61 case noErr:
62 return 0;
63
64 case CSSMERR_TP_INVALID_ANCHOR_CERT:
65 case CSSMERR_TP_NOT_TRUSTED:
66 case CSSMERR_TP_INVALID_CERT_AUTHORITY:
67 return CERT_STATUS_AUTHORITY_INVALID;
68
69 case CSSMERR_TP_CERT_EXPIRED:
70 case CSSMERR_TP_CERT_NOT_VALID_YET:
71 // "Expired" and "not yet valid" collapse into a single status.
72 return CERT_STATUS_DATE_INVALID;
73
74 case CSSMERR_TP_CERT_REVOKED:
75 case CSSMERR_TP_CERT_SUSPENDED:
76 return CERT_STATUS_REVOKED;
77
78 case CSSMERR_APPLETP_HOSTNAME_MISMATCH:
79 return CERT_STATUS_COMMON_NAME_INVALID;
80
81 case CSSMERR_APPLETP_CRL_NOT_FOUND:
82 case CSSMERR_APPLETP_INCOMPLETE_REVOCATION_CHECK:
83 case CSSMERR_APPLETP_OCSP_UNAVAILABLE:
84 return CERT_STATUS_NO_REVOCATION_MECHANISM;
85
86 case CSSMERR_APPLETP_CRL_NOT_TRUSTED:
87 case CSSMERR_APPLETP_CRL_SERVER_DOWN:
88 case CSSMERR_APPLETP_CRL_NOT_VALID_YET:
89 case CSSMERR_APPLETP_NETWORK_FAILURE:
90 case CSSMERR_APPLETP_OCSP_BAD_RESPONSE:
91 case CSSMERR_APPLETP_OCSP_NO_SIGNER:
92 case CSSMERR_APPLETP_OCSP_RESP_UNAUTHORIZED:
93 case CSSMERR_APPLETP_OCSP_RESP_SIG_REQUIRED:
94 case CSSMERR_APPLETP_OCSP_RESP_MALFORMED_REQ:
95 case CSSMERR_APPLETP_OCSP_RESP_INTERNAL_ERR:
96 case CSSMERR_APPLETP_OCSP_RESP_TRY_LATER:
97 // We asked for a revocation check, but didn't get it.
98 return CERT_STATUS_UNABLE_TO_CHECK_REVOCATION;
99
100 default:
101 // Failure was due to something Chromium doesn't define a
102 // specific status for (such as basic constraints violation, or
103 // unknown critical extension)
104 return CERT_STATUS_INVALID;
105 }
106 }
107
OverrideHostnameMismatch(const std::string & hostname,std::vector<std::string> * dns_names)108 bool OverrideHostnameMismatch(const std::string& hostname,
109 std::vector<std::string>* dns_names) {
110 // SecTrustEvaluate() does not check dotted IP addresses. If
111 // hostname is provided as, say, 127.0.0.1, then the error
112 // CSSMERR_APPLETP_HOSTNAME_MISMATCH will always be returned,
113 // even if the certificate contains 127.0.0.1 as one of its names.
114 // We, however, want to allow that behavior. SecTrustEvaluate()
115 // only checks for digits and dots when considering whether a
116 // hostname is an IP address, so IPv6 and hex addresses go through
117 // its normal comparison.
118 bool is_dotted_ip = true;
119 bool override_hostname_mismatch = false;
120 for (std::string::const_iterator c = hostname.begin();
121 c != hostname.end() && is_dotted_ip; ++c)
122 is_dotted_ip = (*c >= '0' && *c <= '9') || *c == '.';
123 if (is_dotted_ip) {
124 for (std::vector<std::string>::const_iterator name = dns_names->begin();
125 name != dns_names->end() && !override_hostname_mismatch; ++name)
126 override_hostname_mismatch = (*name == hostname);
127 }
128 return override_hostname_mismatch;
129 }
130
131 struct CSSMFields {
CSSMFieldsnet::__anon2b8e09460111::CSSMFields132 CSSMFields() : cl_handle(NULL), num_of_fields(0), fields(NULL) {}
~CSSMFieldsnet::__anon2b8e09460111::CSSMFields133 ~CSSMFields() {
134 if (cl_handle)
135 CSSM_CL_FreeFields(cl_handle, num_of_fields, &fields);
136 }
137
138 CSSM_CL_HANDLE cl_handle;
139 uint32 num_of_fields;
140 CSSM_FIELD_PTR fields;
141 };
142
GetCertFields(X509Certificate::OSCertHandle cert_handle,CSSMFields * fields)143 OSStatus GetCertFields(X509Certificate::OSCertHandle cert_handle,
144 CSSMFields* fields) {
145 DCHECK(cert_handle);
146 DCHECK(fields);
147
148 CSSM_DATA cert_data;
149 OSStatus status = SecCertificateGetData(cert_handle, &cert_data);
150 if (status)
151 return status;
152
153 status = SecCertificateGetCLHandle(cert_handle, &fields->cl_handle);
154 if (status) {
155 DCHECK(!fields->cl_handle);
156 return status;
157 }
158
159 status = CSSM_CL_CertGetAllFields(fields->cl_handle, &cert_data,
160 &fields->num_of_fields, &fields->fields);
161 return status;
162 }
163
GetCertGeneralNamesForOID(X509Certificate::OSCertHandle cert_handle,CSSM_OID oid,CE_GeneralNameType name_type,std::vector<std::string> * result)164 void GetCertGeneralNamesForOID(X509Certificate::OSCertHandle cert_handle,
165 CSSM_OID oid, CE_GeneralNameType name_type,
166 std::vector<std::string>* result) {
167 // For future extension: We only support general names of types
168 // GNT_RFC822Name, GNT_DNSName or GNT_URI.
169 DCHECK(name_type == GNT_RFC822Name ||
170 name_type == GNT_DNSName ||
171 name_type == GNT_URI);
172
173 CSSMFields fields;
174 OSStatus status = GetCertFields(cert_handle, &fields);
175 if (status)
176 return;
177
178 for (size_t field = 0; field < fields.num_of_fields; ++field) {
179 if (CSSMOIDEqual(&fields.fields[field].FieldOid, &oid)) {
180 CSSM_X509_EXTENSION_PTR cssm_ext =
181 reinterpret_cast<CSSM_X509_EXTENSION_PTR>(
182 fields.fields[field].FieldValue.Data);
183 CE_GeneralNames* alt_name =
184 reinterpret_cast<CE_GeneralNames*>(cssm_ext->value.parsedValue);
185
186 for (size_t name = 0; name < alt_name->numNames; ++name) {
187 const CE_GeneralName& name_struct = alt_name->generalName[name];
188 // All of the general name types we support are encoded as
189 // IA5String. In general, we should be switching off
190 // |name_struct.nameType| and doing type-appropriate conversions. See
191 // certextensions.h and the comment immediately preceding
192 // CE_GeneralNameType for more information.
193 if (name_struct.nameType == name_type) {
194 const CSSM_DATA& name_data = name_struct.name;
195 std::string value = std::string(
196 reinterpret_cast<const char*>(name_data.Data),
197 name_data.Length);
198 result->push_back(value);
199 }
200 }
201 }
202 }
203 }
204
GetCertDateForOID(X509Certificate::OSCertHandle cert_handle,CSSM_OID oid,Time * result)205 void GetCertDateForOID(X509Certificate::OSCertHandle cert_handle,
206 CSSM_OID oid, Time* result) {
207 *result = Time::Time();
208
209 CSSMFields fields;
210 OSStatus status = GetCertFields(cert_handle, &fields);
211 if (status)
212 return;
213
214 for (size_t field = 0; field < fields.num_of_fields; ++field) {
215 if (CSSMOIDEqual(&fields.fields[field].FieldOid, &oid)) {
216 CSSM_X509_TIME* x509_time = reinterpret_cast<CSSM_X509_TIME*>(
217 fields.fields[field].FieldValue.Data);
218 if (x509_time->timeType != BER_TAG_UTC_TIME &&
219 x509_time->timeType != BER_TAG_GENERALIZED_TIME) {
220 LOG(ERROR) << "Unsupported date/time format "
221 << x509_time->timeType;
222 return;
223 }
224
225 base::StringPiece time_string(
226 reinterpret_cast<const char*>(x509_time->time.Data),
227 x509_time->time.Length);
228 CertDateFormat format = x509_time->timeType == BER_TAG_UTC_TIME ?
229 CERT_DATE_FORMAT_UTC_TIME : CERT_DATE_FORMAT_GENERALIZED_TIME;
230 if (!ParseCertificateDate(time_string, format, result))
231 LOG(ERROR) << "Invalid certificate date/time " << time_string;
232 return;
233 }
234 }
235 }
236
GetCertSerialNumber(X509Certificate::OSCertHandle cert_handle)237 std::string GetCertSerialNumber(X509Certificate::OSCertHandle cert_handle) {
238 CSSMFields fields;
239 OSStatus status = GetCertFields(cert_handle, &fields);
240 if (status)
241 return "";
242
243 std::string ret;
244 for (size_t field = 0; field < fields.num_of_fields; ++field) {
245 if (!CSSMOIDEqual(&fields.fields[field].FieldOid,
246 &CSSMOID_X509V1SerialNumber)) {
247 continue;
248 }
249 ret.assign(
250 reinterpret_cast<char*>(fields.fields[field].FieldValue.Data),
251 fields.fields[field].FieldValue.Length);
252 break;
253 }
254
255 // Remove leading zeros.
256 while (ret.size() > 1 && ret[0] == 0)
257 ret = ret.substr(1, ret.size() - 1);
258
259 return ret;
260 }
261
262 // Creates a SecPolicyRef for the given OID, with optional value.
CreatePolicy(const CSSM_OID * policy_OID,void * option_data,size_t option_length,SecPolicyRef * policy)263 OSStatus CreatePolicy(const CSSM_OID* policy_OID,
264 void* option_data,
265 size_t option_length,
266 SecPolicyRef* policy) {
267 SecPolicySearchRef search;
268 OSStatus err = SecPolicySearchCreate(CSSM_CERT_X_509v3, policy_OID, NULL,
269 &search);
270 if (err)
271 return err;
272 err = SecPolicySearchCopyNext(search, policy);
273 CFRelease(search);
274 if (err)
275 return err;
276
277 if (option_data) {
278 CSSM_DATA options_data = {
279 option_length,
280 reinterpret_cast<uint8_t*>(option_data)
281 };
282 err = SecPolicySetValue(*policy, &options_data);
283 if (err) {
284 CFRelease(*policy);
285 return err;
286 }
287 }
288 return noErr;
289 }
290
291 // Creates a series of SecPolicyRefs to be added to a SecTrustRef used to
292 // validate a certificate for an SSL peer. |hostname| contains the name of
293 // the SSL peer that the certificate should be verified against. |flags| is
294 // a bitwise-OR of VerifyFlags that can further alter how trust is
295 // validated, such as how revocation is checked. If successful, returns
296 // noErr, and stores the resultant array of SecPolicyRefs in |policies|.
CreateTrustPolicies(const std::string & hostname,int flags,ScopedCFTypeRef<CFArrayRef> * policies)297 OSStatus CreateTrustPolicies(const std::string& hostname, int flags,
298 ScopedCFTypeRef<CFArrayRef>* policies) {
299 // Create an SSL SecPolicyRef, and configure it to perform hostname
300 // validation. The hostname check does 99% of what we want, with the
301 // exception of dotted IPv4 addreses, which we handle ourselves below.
302 CSSM_APPLE_TP_SSL_OPTIONS tp_ssl_options = {
303 CSSM_APPLE_TP_SSL_OPTS_VERSION,
304 hostname.size(),
305 hostname.data(),
306 0
307 };
308 SecPolicyRef ssl_policy;
309 OSStatus status = CreatePolicy(&CSSMOID_APPLE_TP_SSL, &tp_ssl_options,
310 sizeof(tp_ssl_options), &ssl_policy);
311 if (status)
312 return status;
313 ScopedCFTypeRef<SecPolicyRef> scoped_ssl_policy(ssl_policy);
314
315 // Manually add OCSP and CRL policies. If neither an OCSP or CRL policy is
316 // specified, the Apple TP module will add whatever the system settings
317 // are, which is not desirable here.
318 //
319 // Note that this causes any locally configured OCSP responder URL to be
320 // ignored.
321 CSSM_APPLE_TP_OCSP_OPTIONS tp_ocsp_options;
322 memset(&tp_ocsp_options, 0, sizeof(tp_ocsp_options));
323 tp_ocsp_options.Version = CSSM_APPLE_TP_OCSP_OPTS_VERSION;
324
325 CSSM_APPLE_TP_CRL_OPTIONS tp_crl_options;
326 memset(&tp_crl_options, 0, sizeof(tp_crl_options));
327 tp_crl_options.Version = CSSM_APPLE_TP_CRL_OPTS_VERSION;
328
329 if (flags & X509Certificate::VERIFY_REV_CHECKING_ENABLED) {
330 // If an OCSP responder is available, use it, and avoid fetching any
331 // CRLs for that certificate if possible, as they may be much larger.
332 tp_ocsp_options.Flags = CSSM_TP_ACTION_OCSP_SUFFICIENT;
333 // Ensure that CRLs can be fetched if a crlDistributionPoint extension
334 // is found. Otherwise, only the local CRL cache will be consulted.
335 tp_crl_options.CrlFlags |= CSSM_TP_ACTION_FETCH_CRL_FROM_NET;
336 } else {
337 // Disable OCSP network fetching, but still permit cached OCSP responses
338 // to be used. This is equivalent to the Windows code's usage of
339 // CERT_CHAIN_REVOCATION_CHECK_CACHE_ONLY.
340 tp_ocsp_options.Flags = CSSM_TP_ACTION_OCSP_DISABLE_NET;
341 // The default CrlFlags will ensure only cached CRLs are used.
342 }
343
344 SecPolicyRef ocsp_policy;
345 status = CreatePolicy(&CSSMOID_APPLE_TP_REVOCATION_OCSP, &tp_ocsp_options,
346 sizeof(tp_ocsp_options), &ocsp_policy);
347 if (status)
348 return status;
349 ScopedCFTypeRef<SecPolicyRef> scoped_ocsp_policy(ocsp_policy);
350
351 SecPolicyRef crl_policy;
352 status = CreatePolicy(&CSSMOID_APPLE_TP_REVOCATION_CRL, &tp_crl_options,
353 sizeof(tp_crl_options), &crl_policy);
354 if (status)
355 return status;
356 ScopedCFTypeRef<SecPolicyRef> scoped_crl_policy(crl_policy);
357
358 CFTypeRef local_policies[] = { ssl_policy, ocsp_policy, crl_policy };
359 CFArrayRef policy_array = CFArrayCreate(kCFAllocatorDefault, local_policies,
360 arraysize(local_policies),
361 &kCFTypeArrayCallBacks);
362 if (!policy_array)
363 return memFullErr;
364
365 policies->reset(policy_array);
366 return noErr;
367 }
368
369 // Gets the issuer for a given cert, starting with the cert itself and
370 // including the intermediate and finally root certificates (if any).
371 // This function calls SecTrust but doesn't actually pay attention to the trust
372 // result: it shouldn't be used to determine trust, just to traverse the chain.
373 // Caller is responsible for releasing the value stored into *out_cert_chain.
CopyCertChain(SecCertificateRef cert_handle,CFArrayRef * out_cert_chain)374 OSStatus CopyCertChain(SecCertificateRef cert_handle,
375 CFArrayRef* out_cert_chain) {
376 DCHECK(cert_handle);
377 DCHECK(out_cert_chain);
378 // Create an SSL policy ref configured for client cert evaluation.
379 SecPolicyRef ssl_policy;
380 OSStatus result = X509Certificate::CreateSSLClientPolicy(&ssl_policy);
381 if (result)
382 return result;
383 ScopedCFTypeRef<SecPolicyRef> scoped_ssl_policy(ssl_policy);
384
385 // Create a SecTrustRef.
386 ScopedCFTypeRef<CFArrayRef> input_certs(CFArrayCreate(
387 NULL, const_cast<const void**>(reinterpret_cast<void**>(&cert_handle)),
388 1, &kCFTypeArrayCallBacks));
389 SecTrustRef trust_ref = NULL;
390 result = SecTrustCreateWithCertificates(input_certs, ssl_policy, &trust_ref);
391 if (result)
392 return result;
393 ScopedCFTypeRef<SecTrustRef> trust(trust_ref);
394
395 // Evaluate trust, which creates the cert chain.
396 SecTrustResultType status;
397 CSSM_TP_APPLE_EVIDENCE_INFO* status_chain;
398 result = SecTrustEvaluate(trust, &status);
399 if (result)
400 return result;
401 return SecTrustGetResult(trust, &status, out_cert_chain, &status_chain);
402 }
403
404 // Returns true if |purpose| is listed as allowed in |usage|. This
405 // function also considers the "Any" purpose. If the attribute is
406 // present and empty, we return false.
ExtendedKeyUsageAllows(const CE_ExtendedKeyUsage * usage,const CSSM_OID * purpose)407 bool ExtendedKeyUsageAllows(const CE_ExtendedKeyUsage* usage,
408 const CSSM_OID* purpose) {
409 for (unsigned p = 0; p < usage->numPurposes; ++p) {
410 if (CSSMOIDEqual(&usage->purposes[p], purpose))
411 return true;
412 if (CSSMOIDEqual(&usage->purposes[p], &CSSMOID_ExtendedKeyUsageAny))
413 return true;
414 }
415 return false;
416 }
417
418 // Test that a given |cert_handle| is actually a valid X.509 certificate, and
419 // return true if it is.
420 //
421 // On OS X, SecCertificateCreateFromData() does not return any errors if
422 // called with invalid data, as long as data is present. The actual decoding
423 // of the certificate does not happen until an API that requires a CSSM
424 // handle is called. While SecCertificateGetCLHandle is the most likely
425 // candidate, as it performs the parsing, it does not check whether the
426 // parsing was actually successful. Instead, SecCertificateGetSubject is
427 // used (supported since 10.3), as a means to check that the certificate
428 // parsed as a valid X.509 certificate.
IsValidOSCertHandle(SecCertificateRef cert_handle)429 bool IsValidOSCertHandle(SecCertificateRef cert_handle) {
430 const CSSM_X509_NAME* sanity_check = NULL;
431 OSStatus status = SecCertificateGetSubject(cert_handle, &sanity_check);
432 return status == noErr && sanity_check;
433 }
434
435 // Parses |data| of length |length|, attempting to decode it as the specified
436 // |format|. If |data| is in the specified format, any certificates contained
437 // within are stored into |output|.
AddCertificatesFromBytes(const char * data,size_t length,SecExternalFormat format,X509Certificate::OSCertHandles * output)438 void AddCertificatesFromBytes(const char* data, size_t length,
439 SecExternalFormat format,
440 X509Certificate::OSCertHandles* output) {
441 SecExternalFormat input_format = format;
442 ScopedCFTypeRef<CFDataRef> local_data(CFDataCreateWithBytesNoCopy(
443 kCFAllocatorDefault, reinterpret_cast<const UInt8*>(data), length,
444 kCFAllocatorNull));
445
446 CFArrayRef items = NULL;
447 OSStatus status = SecKeychainItemImport(local_data, NULL, &input_format,
448 NULL, 0, NULL, NULL, &items);
449 if (status) {
450 DLOG(WARNING) << status << " Unable to import items from data of length "
451 << length;
452 return;
453 }
454
455 ScopedCFTypeRef<CFArrayRef> scoped_items(items);
456 CFTypeID cert_type_id = SecCertificateGetTypeID();
457
458 for (CFIndex i = 0; i < CFArrayGetCount(items); ++i) {
459 SecKeychainItemRef item = reinterpret_cast<SecKeychainItemRef>(
460 const_cast<void*>(CFArrayGetValueAtIndex(items, i)));
461
462 // While inputFormat implies only certificates will be imported, if/when
463 // other formats (eg: PKCS#12) are supported, this may also include
464 // private keys or other items types, so filter appropriately.
465 if (CFGetTypeID(item) == cert_type_id) {
466 SecCertificateRef cert = reinterpret_cast<SecCertificateRef>(item);
467 // OS X ignores |input_format| if it detects that |local_data| is PEM
468 // encoded, attempting to decode data based on internal rules for PEM
469 // block headers. If a PKCS#7 blob is encoded with a PEM block of
470 // CERTIFICATE, OS X 10.5 will return a single, invalid certificate
471 // based on the decoded data. If this happens, the certificate should
472 // not be included in |output|. Because |output| is empty,
473 // CreateCertificateListfromBytes will use PEMTokenizer to decode the
474 // data. When called again with the decoded data, OS X will honor
475 // |input_format|, causing decode to succeed. On OS X 10.6, the data
476 // is properly decoded as a PKCS#7, whether PEM or not, which avoids
477 // the need to fallback to internal decoding.
478 if (IsValidOSCertHandle(cert)) {
479 CFRetain(cert);
480 output->push_back(cert);
481 }
482 }
483 }
484 }
485
486 struct CSSMOIDString {
487 const CSSM_OID* oid_;
488 std::string string_;
489 };
490
491 typedef std::vector<CSSMOIDString> CSSMOIDStringVector;
492
CERTNameToCSSMOIDVector(CERTName * name,CSSMOIDStringVector * out_values)493 bool CERTNameToCSSMOIDVector(CERTName* name, CSSMOIDStringVector* out_values) {
494 struct OIDCSSMMap {
495 SECOidTag sec_OID_;
496 const CSSM_OID* cssm_OID_;
497 };
498
499 const OIDCSSMMap kOIDs[] = {
500 { SEC_OID_AVA_COMMON_NAME, &CSSMOID_CommonName },
501 { SEC_OID_AVA_COUNTRY_NAME, &CSSMOID_CountryName },
502 { SEC_OID_AVA_LOCALITY, &CSSMOID_LocalityName },
503 { SEC_OID_AVA_STATE_OR_PROVINCE, &CSSMOID_StateProvinceName },
504 { SEC_OID_AVA_STREET_ADDRESS, &CSSMOID_StreetAddress },
505 { SEC_OID_AVA_ORGANIZATION_NAME, &CSSMOID_OrganizationName },
506 { SEC_OID_AVA_ORGANIZATIONAL_UNIT_NAME, &CSSMOID_OrganizationalUnitName },
507 { SEC_OID_AVA_DN_QUALIFIER, &CSSMOID_DNQualifier },
508 { SEC_OID_RFC1274_UID, &CSSMOID_UniqueIdentifier },
509 { SEC_OID_PKCS9_EMAIL_ADDRESS, &CSSMOID_EmailAddress },
510 };
511
512 CERTRDN** rdns = name->rdns;
513 for (size_t rdn = 0; rdns[rdn]; ++rdn) {
514 CERTAVA** avas = rdns[rdn]->avas;
515 for (size_t pair = 0; avas[pair] != 0; ++pair) {
516 SECOidTag tag = CERT_GetAVATag(avas[pair]);
517 if (tag == SEC_OID_UNKNOWN) {
518 return false;
519 }
520 CSSMOIDString oidString;
521 bool found_oid = false;
522 for (size_t oid = 0; oid < ARRAYSIZE_UNSAFE(kOIDs); ++oid) {
523 if (kOIDs[oid].sec_OID_ == tag) {
524 SECItem* decode_item = CERT_DecodeAVAValue(&avas[pair]->value);
525 if (!decode_item)
526 return false;
527
528 // TODO(wtc): Pass decode_item to CERT_RFC1485_EscapeAndQuote.
529 std::string value(reinterpret_cast<char*>(decode_item->data),
530 decode_item->len);
531 oidString.oid_ = kOIDs[oid].cssm_OID_;
532 oidString.string_ = value;
533 out_values->push_back(oidString);
534 SECITEM_FreeItem(decode_item, PR_TRUE);
535 found_oid = true;
536 break;
537 }
538 }
539 if (!found_oid) {
540 DLOG(ERROR) << "Unrecognized OID: " << tag;
541 }
542 }
543 }
544 return true;
545 }
546
547 class ScopedCertName {
548 public:
ScopedCertName(CERTName * name)549 explicit ScopedCertName(CERTName* name) : name_(name) { }
~ScopedCertName()550 ~ScopedCertName() {
551 if (name_) CERT_DestroyName(name_);
552 }
operator CERTName*()553 operator CERTName*() { return name_; }
554
555 private:
556 CERTName* name_;
557 };
558
559 class ScopedEncodedCertResults {
560 public:
ScopedEncodedCertResults(CSSM_TP_RESULT_SET * results)561 explicit ScopedEncodedCertResults(CSSM_TP_RESULT_SET* results)
562 : results_(results) { }
~ScopedEncodedCertResults()563 ~ScopedEncodedCertResults() {
564 if (results_) {
565 CSSM_ENCODED_CERT* encCert =
566 reinterpret_cast<CSSM_ENCODED_CERT*>(results_->Results);
567 for (uint32 i = 0; i < results_->NumberOfResults; i++) {
568 crypto::CSSMFree(encCert[i].CertBlob.Data);
569 }
570 }
571 crypto::CSSMFree(results_->Results);
572 crypto::CSSMFree(results_);
573 }
574
575 private:
576 CSSM_TP_RESULT_SET* results_;
577 };
578
AppendPublicKeyHashes(CFArrayRef chain,std::vector<SHA1Fingerprint> * hashes)579 void AppendPublicKeyHashes(CFArrayRef chain,
580 std::vector<SHA1Fingerprint>* hashes) {
581 const CFIndex n = CFArrayGetCount(chain);
582 for (CFIndex i = 0; i < n; i++) {
583 SecCertificateRef cert = reinterpret_cast<SecCertificateRef>(
584 const_cast<void*>(CFArrayGetValueAtIndex(chain, i)));
585
586 CSSM_DATA cert_data;
587 OSStatus err = SecCertificateGetData(cert, &cert_data);
588 DCHECK_EQ(err, noErr);
589 base::StringPiece der_bytes(reinterpret_cast<const char*>(cert_data.Data),
590 cert_data.Length);
591 base::StringPiece spki_bytes;
592 if (!asn1::ExtractSPKIFromDERCert(der_bytes, &spki_bytes))
593 continue;
594
595 SHA1Fingerprint hash;
596 CC_SHA1(spki_bytes.data(), spki_bytes.size(), hash.data);
597 hashes->push_back(hash);
598 }
599 }
600
601 } // namespace
602
Initialize()603 void X509Certificate::Initialize() {
604 const CSSM_X509_NAME* name;
605 OSStatus status = SecCertificateGetSubject(cert_handle_, &name);
606 if (!status)
607 subject_.Parse(name);
608
609 status = SecCertificateGetIssuer(cert_handle_, &name);
610 if (!status)
611 issuer_.Parse(name);
612
613 GetCertDateForOID(cert_handle_, CSSMOID_X509V1ValidityNotBefore,
614 &valid_start_);
615 GetCertDateForOID(cert_handle_, CSSMOID_X509V1ValidityNotAfter,
616 &valid_expiry_);
617
618 fingerprint_ = CalculateFingerprint(cert_handle_);
619 serial_number_ = GetCertSerialNumber(cert_handle_);
620 }
621
622 // IsIssuedByKnownRoot returns true if the given chain is rooted at a root CA
623 // that we recognise as a standard root.
624 // static
IsIssuedByKnownRoot(CFArrayRef chain)625 bool X509Certificate::IsIssuedByKnownRoot(CFArrayRef chain) {
626 int n = CFArrayGetCount(chain);
627 if (n < 1)
628 return false;
629 SecCertificateRef root_ref = reinterpret_cast<SecCertificateRef>(
630 const_cast<void*>(CFArrayGetValueAtIndex(chain, n - 1)));
631 SHA1Fingerprint hash = X509Certificate::CalculateFingerprint(root_ref);
632 return IsSHA1HashInSortedArray(
633 hash, &kKnownRootCertSHA1Hashes[0][0], sizeof(kKnownRootCertSHA1Hashes));
634 }
635
636 // static
CreateSelfSigned(crypto::RSAPrivateKey * key,const std::string & subject,uint32 serial_number,base::TimeDelta valid_duration)637 X509Certificate* X509Certificate::CreateSelfSigned(
638 crypto::RSAPrivateKey* key,
639 const std::string& subject,
640 uint32 serial_number,
641 base::TimeDelta valid_duration) {
642 DCHECK(key);
643 DCHECK(!subject.empty());
644
645 if (valid_duration.InSeconds() > UINT32_MAX) {
646 LOG(ERROR) << "valid_duration too big" << valid_duration.InSeconds();
647 valid_duration = base::TimeDelta::FromSeconds(UINT32_MAX);
648 }
649
650 // There is a comment in
651 // http://www.opensource.apple.com/source/security_certtool/security_certtool-31828/src/CertTool.cpp
652 // that serial_numbers being passed into CSSM_TP_SubmitCredRequest can't have
653 // their high bit set. We will continue though and mask it out below.
654 if (serial_number & 0x80000000)
655 LOG(ERROR) << "serial_number has high bit set " << serial_number;
656
657 // NSS is used to parse the subject string into a set of
658 // CSSM_OID/string pairs. There doesn't appear to be a system routine for
659 // parsing Distinguished Name strings.
660 crypto::EnsureNSSInit();
661
662 CSSMOIDStringVector subject_name_oids;
663 ScopedCertName subject_name(
664 CERT_AsciiToName(const_cast<char*>(subject.c_str())));
665 if (!CERTNameToCSSMOIDVector(subject_name, &subject_name_oids)) {
666 DLOG(ERROR) << "Unable to generate CSSMOIDMap from " << subject;
667 return NULL;
668 }
669
670 // Convert the map of oid/string pairs into an array of
671 // CSSM_APPLE_TP_NAME_OIDs.
672 std::vector<CSSM_APPLE_TP_NAME_OID> cssm_subject_names;
673 for(CSSMOIDStringVector::iterator iter = subject_name_oids.begin();
674 iter != subject_name_oids.end(); ++iter) {
675 CSSM_APPLE_TP_NAME_OID cssm_subject_name;
676 cssm_subject_name.oid = iter->oid_;
677 cssm_subject_name.string = iter->string_.c_str();
678 cssm_subject_names.push_back(cssm_subject_name);
679 }
680
681 if (cssm_subject_names.empty()) {
682 DLOG(ERROR) << "cssm_subject_names.size() == 0. Input: " << subject;
683 return NULL;
684 }
685
686 // Set up a certificate request.
687 CSSM_APPLE_TP_CERT_REQUEST certReq;
688 memset(&certReq, 0, sizeof(certReq));
689 certReq.cspHand = crypto::GetSharedCSPHandle();
690 certReq.clHand = crypto::GetSharedCLHandle();
691 // See comment about serial numbers above.
692 certReq.serialNumber = serial_number & 0x7fffffff;
693 certReq.numSubjectNames = cssm_subject_names.size();
694 certReq.subjectNames = &cssm_subject_names[0];
695 certReq.numIssuerNames = 0; // Root.
696 certReq.issuerNames = NULL;
697 certReq.issuerNameX509 = NULL;
698 certReq.certPublicKey = key->public_key();
699 certReq.issuerPrivateKey = key->key();
700 // These are the Apple defaults.
701 certReq.signatureAlg = CSSM_ALGID_SHA1WithRSA;
702 certReq.signatureOid = CSSMOID_SHA1WithRSA;
703 certReq.notBefore = 0;
704 certReq.notAfter = static_cast<uint32>(valid_duration.InSeconds());
705 certReq.numExtensions = 0;
706 certReq.extensions = NULL;
707 certReq.challengeString = NULL;
708
709 CSSM_TP_REQUEST_SET reqSet;
710 reqSet.NumberOfRequests = 1;
711 reqSet.Requests = &certReq;
712
713 CSSM_FIELD policyId;
714 memset(&policyId, 0, sizeof(policyId));
715 policyId.FieldOid = CSSMOID_APPLE_TP_LOCAL_CERT_GEN;
716
717 CSSM_TP_CALLERAUTH_CONTEXT callerAuthContext;
718 memset(&callerAuthContext, 0, sizeof(callerAuthContext));
719 callerAuthContext.Policy.NumberOfPolicyIds = 1;
720 callerAuthContext.Policy.PolicyIds = &policyId;
721
722 CSSM_TP_HANDLE tp_handle = crypto::GetSharedTPHandle();
723 CSSM_DATA refId;
724 memset(&refId, 0, sizeof(refId));
725 sint32 estTime;
726 CSSM_RETURN crtn = CSSM_TP_SubmitCredRequest(tp_handle, NULL,
727 CSSM_TP_AUTHORITY_REQUEST_CERTISSUE, &reqSet, &callerAuthContext,
728 &estTime, &refId);
729 if(crtn) {
730 DLOG(ERROR) << "CSSM_TP_SubmitCredRequest failed " << crtn;
731 return NULL;
732 }
733
734 CSSM_BOOL confirmRequired;
735 CSSM_TP_RESULT_SET *resultSet = NULL;
736 crtn = CSSM_TP_RetrieveCredResult(tp_handle, &refId, NULL, &estTime,
737 &confirmRequired, &resultSet);
738 ScopedEncodedCertResults scopedResults(resultSet);
739 crypto::CSSMFree(refId.Data);
740 if (crtn) {
741 DLOG(ERROR) << "CSSM_TP_RetrieveCredResult failed " << crtn;
742 return NULL;
743 }
744
745 if (confirmRequired) {
746 // Potential leak here of resultSet. |confirmRequired| should never be
747 // true.
748 DLOG(ERROR) << "CSSM_TP_RetrieveCredResult required confirmation";
749 return NULL;
750 }
751
752 if (resultSet->NumberOfResults != 1) {
753 DLOG(ERROR) << "Unexpected number of results: "
754 << resultSet->NumberOfResults;
755 return NULL;
756 }
757
758 CSSM_ENCODED_CERT* encCert =
759 reinterpret_cast<CSSM_ENCODED_CERT*>(resultSet->Results);
760 base::mac::ScopedCFTypeRef<SecCertificateRef> scoped_cert;
761 SecCertificateRef certificate_ref = NULL;
762 OSStatus os_status =
763 SecCertificateCreateFromData(&encCert->CertBlob, encCert->CertType,
764 encCert->CertEncoding, &certificate_ref);
765 if (os_status != 0) {
766 DLOG(ERROR) << "SecCertificateCreateFromData failed: " << os_status;
767 return NULL;
768 }
769 scoped_cert.reset(certificate_ref);
770
771 return CreateFromHandle(
772 scoped_cert, X509Certificate::SOURCE_LONE_CERT_IMPORT,
773 X509Certificate::OSCertHandles());
774 }
775
GetDNSNames(std::vector<std::string> * dns_names) const776 void X509Certificate::GetDNSNames(std::vector<std::string>* dns_names) const {
777 dns_names->clear();
778
779 GetCertGeneralNamesForOID(cert_handle_, CSSMOID_SubjectAltName, GNT_DNSName,
780 dns_names);
781
782 if (dns_names->empty())
783 dns_names->push_back(subject_.common_name);
784 }
785
Verify(const std::string & hostname,int flags,CertVerifyResult * verify_result) const786 int X509Certificate::Verify(const std::string& hostname, int flags,
787 CertVerifyResult* verify_result) const {
788 verify_result->Reset();
789
790 if (IsBlacklisted()) {
791 verify_result->cert_status |= CERT_STATUS_REVOKED;
792 return ERR_CERT_REVOKED;
793 }
794
795 ScopedCFTypeRef<CFArrayRef> trust_policies;
796 OSStatus status = CreateTrustPolicies(hostname, flags, &trust_policies);
797 if (status)
798 return NetErrorFromOSStatus(status);
799
800 // Create and configure a SecTrustRef, which takes our certificate(s)
801 // and our SSL SecPolicyRef. SecTrustCreateWithCertificates() takes an
802 // array of certificates, the first of which is the certificate we're
803 // verifying, and the subsequent (optional) certificates are used for
804 // chain building.
805 CFMutableArrayRef cert_array = CFArrayCreateMutable(kCFAllocatorDefault, 0,
806 &kCFTypeArrayCallBacks);
807 if (!cert_array)
808 return ERR_OUT_OF_MEMORY;
809 ScopedCFTypeRef<CFArrayRef> scoped_cert_array(cert_array);
810 CFArrayAppendValue(cert_array, cert_handle_);
811 for (size_t i = 0; i < intermediate_ca_certs_.size(); ++i)
812 CFArrayAppendValue(cert_array, intermediate_ca_certs_[i]);
813
814 // From here on, only one thread can be active at a time. We have had a number
815 // of sporadic crashes in the SecTrustEvaluate call below, way down inside
816 // Apple's cert code, which we suspect are caused by a thread-safety issue.
817 // So as a speculative fix allow only one thread to use SecTrust on this cert.
818 base::AutoLock lock(verification_lock_);
819
820 SecTrustRef trust_ref = NULL;
821 status = SecTrustCreateWithCertificates(cert_array, trust_policies,
822 &trust_ref);
823 if (status)
824 return NetErrorFromOSStatus(status);
825 ScopedCFTypeRef<SecTrustRef> scoped_trust_ref(trust_ref);
826
827 if (TestRootCerts::HasInstance()) {
828 status = TestRootCerts::GetInstance()->FixupSecTrustRef(trust_ref);
829 if (status)
830 return NetErrorFromOSStatus(status);
831 }
832
833 CSSM_APPLE_TP_ACTION_DATA tp_action_data;
834 memset(&tp_action_data, 0, sizeof(tp_action_data));
835 tp_action_data.Version = CSSM_APPLE_TP_ACTION_VERSION;
836 // Allow CSSM to download any missing intermediate certificates if an
837 // authorityInfoAccess extension or issuerAltName extension is present.
838 tp_action_data.ActionFlags = CSSM_TP_ACTION_FETCH_CERT_FROM_NET;
839
840 if (flags & VERIFY_REV_CHECKING_ENABLED) {
841 // Require a positive result from an OCSP responder or a CRL (or both)
842 // for every certificate in the chain. The Apple TP automatically
843 // excludes the self-signed root from this requirement. If a certificate
844 // is missing both a crlDistributionPoints extension and an
845 // authorityInfoAccess extension with an OCSP responder URL, then we
846 // will get a kSecTrustResultRecoverableTrustFailure back from
847 // SecTrustEvaluate(), with a
848 // CSSMERR_APPLETP_INCOMPLETE_REVOCATION_CHECK error code. In that case,
849 // we'll set our own result to include
850 // CERT_STATUS_NO_REVOCATION_MECHANISM. If one or both extensions are
851 // present, and a check fails (server unavailable, OCSP retry later,
852 // signature mismatch), then we'll set our own result to include
853 // CERT_STATUS_UNABLE_TO_CHECK_REVOCATION.
854 tp_action_data.ActionFlags |= CSSM_TP_ACTION_REQUIRE_REV_PER_CERT;
855 verify_result->cert_status |= CERT_STATUS_REV_CHECKING_ENABLED;
856 } else {
857 // EV requires revocation checking.
858 // Note, under the hood, SecTrustEvaluate() will modify the OCSP options
859 // so as to attempt OCSP fetching if it believes a certificate may chain
860 // to an EV root. However, because network fetches are disabled in
861 // CreateTrustPolicies() when revocation checking is disabled, these
862 // will only go against the local cache.
863 flags &= ~VERIFY_EV_CERT;
864 }
865
866 CFDataRef action_data_ref =
867 CFDataCreateWithBytesNoCopy(kCFAllocatorDefault,
868 reinterpret_cast<UInt8*>(&tp_action_data),
869 sizeof(tp_action_data), kCFAllocatorNull);
870 if (!action_data_ref)
871 return ERR_OUT_OF_MEMORY;
872 ScopedCFTypeRef<CFDataRef> scoped_action_data_ref(action_data_ref);
873 status = SecTrustSetParameters(trust_ref, CSSM_TP_ACTION_DEFAULT,
874 action_data_ref);
875 if (status)
876 return NetErrorFromOSStatus(status);
877
878 // Verify the certificate. A non-zero result from SecTrustGetResult()
879 // indicates that some fatal error occurred and the chain couldn't be
880 // processed, not that the chain contains no errors. We need to examine the
881 // output of SecTrustGetResult() to determine that.
882 SecTrustResultType trust_result;
883 status = SecTrustEvaluate(trust_ref, &trust_result);
884 if (status)
885 return NetErrorFromOSStatus(status);
886 CFArrayRef completed_chain = NULL;
887 CSSM_TP_APPLE_EVIDENCE_INFO* chain_info;
888 status = SecTrustGetResult(trust_ref, &trust_result, &completed_chain,
889 &chain_info);
890 if (status)
891 return NetErrorFromOSStatus(status);
892 ScopedCFTypeRef<CFArrayRef> scoped_completed_chain(completed_chain);
893
894 // Evaluate the results
895 OSStatus cssm_result;
896 bool got_certificate_error = false;
897 switch (trust_result) {
898 case kSecTrustResultUnspecified:
899 case kSecTrustResultProceed:
900 // Certificate chain is valid and trusted ("unspecified" indicates that
901 // the user has not explicitly set a trust setting)
902 break;
903
904 case kSecTrustResultDeny:
905 case kSecTrustResultConfirm:
906 // Certificate chain is explicitly untrusted. For kSecTrustResultConfirm,
907 // we're following what Secure Transport does and treating it as
908 // "deny".
909 verify_result->cert_status |= CERT_STATUS_AUTHORITY_INVALID;
910 break;
911
912 case kSecTrustResultRecoverableTrustFailure:
913 // Certificate chain has a failure that can be overridden by the user.
914 status = SecTrustGetCssmResultCode(trust_ref, &cssm_result);
915 if (status)
916 return NetErrorFromOSStatus(status);
917 switch (cssm_result) {
918 case CSSMERR_TP_NOT_TRUSTED:
919 case CSSMERR_TP_INVALID_ANCHOR_CERT:
920 verify_result->cert_status |= CERT_STATUS_AUTHORITY_INVALID;
921 break;
922 case CSSMERR_TP_CERT_EXPIRED:
923 case CSSMERR_TP_CERT_NOT_VALID_YET:
924 verify_result->cert_status |= CERT_STATUS_DATE_INVALID;
925 break;
926 case CSSMERR_TP_CERT_REVOKED:
927 case CSSMERR_TP_CERT_SUSPENDED:
928 verify_result->cert_status |= CERT_STATUS_REVOKED;
929 break;
930 default:
931 // Look for specific per-certificate errors below.
932 break;
933 }
934 // Walk the chain of error codes in the CSSM_TP_APPLE_EVIDENCE_INFO
935 // structure which can catch multiple errors from each certificate.
936 for (CFIndex index = 0, chain_count = CFArrayGetCount(completed_chain);
937 index < chain_count; ++index) {
938 if (chain_info[index].StatusBits & CSSM_CERT_STATUS_EXPIRED ||
939 chain_info[index].StatusBits & CSSM_CERT_STATUS_NOT_VALID_YET)
940 verify_result->cert_status |= CERT_STATUS_DATE_INVALID;
941 for (uint32 status_code_index = 0;
942 status_code_index < chain_info[index].NumStatusCodes;
943 ++status_code_index) {
944 got_certificate_error = true;
945 int cert_status = CertStatusFromOSStatus(
946 chain_info[index].StatusCodes[status_code_index]);
947 if (cert_status == CERT_STATUS_COMMON_NAME_INVALID) {
948 std::vector<std::string> names;
949 GetDNSNames(&names);
950 if (OverrideHostnameMismatch(hostname, &names))
951 cert_status = 0;
952 }
953 verify_result->cert_status |= cert_status;
954 }
955 }
956 // Be paranoid and ensure that we recorded at least one certificate
957 // status on receiving kSecTrustResultRecoverableTrustFailure. The
958 // call to SecTrustGetCssmResultCode() should pick up when the chain
959 // is not trusted and the loop through CSSM_TP_APPLE_EVIDENCE_INFO
960 // should pick up everything else, but let's be safe.
961 if (!verify_result->cert_status && !got_certificate_error) {
962 verify_result->cert_status |= CERT_STATUS_INVALID;
963 NOTREACHED();
964 }
965 break;
966
967 default:
968 status = SecTrustGetCssmResultCode(trust_ref, &cssm_result);
969 if (status)
970 return NetErrorFromOSStatus(status);
971 verify_result->cert_status |= CertStatusFromOSStatus(cssm_result);
972 if (!verify_result->cert_status)
973 verify_result->cert_status |= CERT_STATUS_INVALID;
974 break;
975 }
976
977 // TODO(wtc): Suppress CERT_STATUS_NO_REVOCATION_MECHANISM for now to be
978 // compatible with Windows, which in turn implements this behavior to be
979 // compatible with WinHTTP, which doesn't report this error (bug 3004).
980 verify_result->cert_status &= ~CERT_STATUS_NO_REVOCATION_MECHANISM;
981
982 if (IsCertStatusError(verify_result->cert_status))
983 return MapCertStatusToNetError(verify_result->cert_status);
984
985 if (flags & VERIFY_EV_CERT) {
986 // Determine the certificate's EV status using SecTrustCopyExtendedResult(),
987 // which we need to look up because the function wasn't added until
988 // Mac OS X 10.5.7.
989 // Note: "ExtendedResult" means extended validation results.
990 CFBundleRef bundle =
991 CFBundleGetBundleWithIdentifier(CFSTR("com.apple.security"));
992 if (bundle) {
993 SecTrustCopyExtendedResultFuncPtr copy_extended_result =
994 reinterpret_cast<SecTrustCopyExtendedResultFuncPtr>(
995 CFBundleGetFunctionPointerForName(bundle,
996 CFSTR("SecTrustCopyExtendedResult")));
997 if (copy_extended_result) {
998 CFDictionaryRef ev_dict = NULL;
999 status = copy_extended_result(trust_ref, &ev_dict);
1000 if (!status && ev_dict) {
1001 // The returned dictionary contains the EV organization name from the
1002 // server certificate, which we don't need at this point (and we
1003 // have other ways to access, anyway). All we care is that
1004 // SecTrustCopyExtendedResult() returned noErr and a non-NULL
1005 // dictionary.
1006 CFRelease(ev_dict);
1007 verify_result->cert_status |= CERT_STATUS_IS_EV;
1008 }
1009 }
1010 }
1011 }
1012
1013 AppendPublicKeyHashes(completed_chain, &verify_result->public_key_hashes);
1014 verify_result->is_issued_by_known_root = IsIssuedByKnownRoot(completed_chain);
1015
1016 if (IsPublicKeyBlacklisted(verify_result->public_key_hashes)) {
1017 verify_result->cert_status |= CERT_STATUS_AUTHORITY_INVALID;
1018 return MapCertStatusToNetError(verify_result->cert_status);
1019 }
1020
1021 return OK;
1022 }
1023
GetDEREncoded(std::string * encoded)1024 bool X509Certificate::GetDEREncoded(std::string* encoded) {
1025 encoded->clear();
1026 CSSM_DATA der_data;
1027 if(SecCertificateGetData(cert_handle_, &der_data) == noErr) {
1028 encoded->append(reinterpret_cast<char*>(der_data.Data),
1029 der_data.Length);
1030 return true;
1031 }
1032 return false;
1033 }
1034
VerifyEV() const1035 bool X509Certificate::VerifyEV() const {
1036 // We don't call this private method, but we do need to implement it because
1037 // it's defined in x509_certificate.h. We perform EV checking in the
1038 // Verify() above.
1039 NOTREACHED();
1040 return false;
1041 }
1042
1043 // static
IsSameOSCert(X509Certificate::OSCertHandle a,X509Certificate::OSCertHandle b)1044 bool X509Certificate::IsSameOSCert(X509Certificate::OSCertHandle a,
1045 X509Certificate::OSCertHandle b) {
1046 DCHECK(a && b);
1047 if (a == b)
1048 return true;
1049 if (CFEqual(a, b))
1050 return true;
1051 CSSM_DATA a_data, b_data;
1052 return SecCertificateGetData(a, &a_data) == noErr &&
1053 SecCertificateGetData(b, &b_data) == noErr &&
1054 a_data.Length == b_data.Length &&
1055 memcmp(a_data.Data, b_data.Data, a_data.Length) == 0;
1056 }
1057
1058 // static
CreateOSCertHandleFromBytes(const char * data,int length)1059 X509Certificate::OSCertHandle X509Certificate::CreateOSCertHandleFromBytes(
1060 const char* data, int length) {
1061 CSSM_DATA cert_data;
1062 cert_data.Data = const_cast<uint8*>(reinterpret_cast<const uint8*>(data));
1063 cert_data.Length = length;
1064
1065 OSCertHandle cert_handle = NULL;
1066 OSStatus status = SecCertificateCreateFromData(&cert_data,
1067 CSSM_CERT_X_509v3,
1068 CSSM_CERT_ENCODING_DER,
1069 &cert_handle);
1070 if (status != noErr)
1071 return NULL;
1072 if (!IsValidOSCertHandle(cert_handle)) {
1073 CFRelease(cert_handle);
1074 return NULL;
1075 }
1076 return cert_handle;
1077 }
1078
1079 // static
CreateOSCertHandlesFromBytes(const char * data,int length,Format format)1080 X509Certificate::OSCertHandles X509Certificate::CreateOSCertHandlesFromBytes(
1081 const char* data, int length, Format format) {
1082 OSCertHandles results;
1083
1084 switch (format) {
1085 case FORMAT_SINGLE_CERTIFICATE: {
1086 OSCertHandle handle = CreateOSCertHandleFromBytes(data, length);
1087 if (handle)
1088 results.push_back(handle);
1089 break;
1090 }
1091 case FORMAT_PKCS7:
1092 AddCertificatesFromBytes(data, length, kSecFormatPKCS7, &results);
1093 break;
1094 default:
1095 NOTREACHED() << "Certificate format " << format << " unimplemented";
1096 break;
1097 }
1098
1099 return results;
1100 }
1101
1102 // static
DupOSCertHandle(OSCertHandle handle)1103 X509Certificate::OSCertHandle X509Certificate::DupOSCertHandle(
1104 OSCertHandle handle) {
1105 if (!handle)
1106 return NULL;
1107 return reinterpret_cast<OSCertHandle>(const_cast<void*>(CFRetain(handle)));
1108 }
1109
1110 // static
FreeOSCertHandle(OSCertHandle cert_handle)1111 void X509Certificate::FreeOSCertHandle(OSCertHandle cert_handle) {
1112 CFRelease(cert_handle);
1113 }
1114
1115 // static
CalculateFingerprint(OSCertHandle cert)1116 SHA1Fingerprint X509Certificate::CalculateFingerprint(
1117 OSCertHandle cert) {
1118 SHA1Fingerprint sha1;
1119 memset(sha1.data, 0, sizeof(sha1.data));
1120
1121 CSSM_DATA cert_data;
1122 OSStatus status = SecCertificateGetData(cert, &cert_data);
1123 if (status)
1124 return sha1;
1125
1126 DCHECK(cert_data.Data);
1127 DCHECK_NE(cert_data.Length, 0U);
1128
1129 CC_SHA1(cert_data.Data, cert_data.Length, sha1.data);
1130
1131 return sha1;
1132 }
1133
SupportsSSLClientAuth() const1134 bool X509Certificate::SupportsSSLClientAuth() const {
1135 CSSMFields fields;
1136 if (GetCertFields(cert_handle_, &fields) != noErr)
1137 return false;
1138
1139 // Gather the extensions we care about. We do not support
1140 // CSSMOID_NetscapeCertType on OS X.
1141 const CE_ExtendedKeyUsage* ext_key_usage = NULL;
1142 const CE_KeyUsage* key_usage = NULL;
1143 for (unsigned f = 0; f < fields.num_of_fields; ++f) {
1144 const CSSM_FIELD& field = fields.fields[f];
1145 const CSSM_X509_EXTENSION* ext =
1146 reinterpret_cast<const CSSM_X509_EXTENSION*>(field.FieldValue.Data);
1147 if (CSSMOIDEqual(&field.FieldOid, &CSSMOID_KeyUsage)) {
1148 key_usage = reinterpret_cast<const CE_KeyUsage*>(ext->value.parsedValue);
1149 } else if (CSSMOIDEqual(&field.FieldOid, &CSSMOID_ExtendedKeyUsage)) {
1150 ext_key_usage =
1151 reinterpret_cast<const CE_ExtendedKeyUsage*>(ext->value.parsedValue);
1152 }
1153 }
1154
1155 // RFC5280 says to take the intersection of the two extensions.
1156 //
1157 // Our underlying crypto libraries don't expose
1158 // ClientCertificateType, so for now we will not support fixed
1159 // Diffie-Hellman mechanisms. For rsa_sign, we need the
1160 // digitalSignature bit.
1161 //
1162 // In particular, if a key has the nonRepudiation bit and not the
1163 // digitalSignature one, we will not offer it to the user.
1164 if (key_usage && !((*key_usage) & CE_KU_DigitalSignature))
1165 return false;
1166 if (ext_key_usage && !ExtendedKeyUsageAllows(ext_key_usage,
1167 &CSSMOID_ClientAuth))
1168 return false;
1169 return true;
1170 }
1171
IsIssuedBy(const std::vector<CertPrincipal> & valid_issuers)1172 bool X509Certificate::IsIssuedBy(
1173 const std::vector<CertPrincipal>& valid_issuers) {
1174 // Get the cert's issuer chain.
1175 CFArrayRef cert_chain = NULL;
1176 OSStatus result;
1177 result = CopyCertChain(os_cert_handle(), &cert_chain);
1178 if (result)
1179 return false;
1180 ScopedCFTypeRef<CFArrayRef> scoped_cert_chain(cert_chain);
1181
1182 // Check all the certs in the chain for a match.
1183 int n = CFArrayGetCount(cert_chain);
1184 for (int i = 0; i < n; ++i) {
1185 SecCertificateRef cert_handle = reinterpret_cast<SecCertificateRef>(
1186 const_cast<void*>(CFArrayGetValueAtIndex(cert_chain, i)));
1187 scoped_refptr<X509Certificate> cert(X509Certificate::CreateFromHandle(
1188 cert_handle,
1189 X509Certificate::SOURCE_LONE_CERT_IMPORT,
1190 X509Certificate::OSCertHandles()));
1191 for (unsigned j = 0; j < valid_issuers.size(); j++) {
1192 if (cert->issuer().Matches(valid_issuers[j]))
1193 return true;
1194 }
1195 }
1196 return false;
1197 }
1198
1199 // static
CreateSSLClientPolicy(SecPolicyRef * out_policy)1200 OSStatus X509Certificate::CreateSSLClientPolicy(SecPolicyRef* out_policy) {
1201 CSSM_APPLE_TP_SSL_OPTIONS tp_ssl_options = {
1202 CSSM_APPLE_TP_SSL_OPTS_VERSION,
1203 0,
1204 NULL,
1205 CSSM_APPLE_TP_SSL_CLIENT
1206 };
1207 return CreatePolicy(&CSSMOID_APPLE_TP_SSL,
1208 &tp_ssl_options,
1209 sizeof(tp_ssl_options),
1210 out_policy);
1211 }
1212
1213 // static
GetSSLClientCertificates(const std::string & server_domain,const std::vector<CertPrincipal> & valid_issuers,CertificateList * certs)1214 bool X509Certificate::GetSSLClientCertificates(
1215 const std::string& server_domain,
1216 const std::vector<CertPrincipal>& valid_issuers,
1217 CertificateList* certs) {
1218 ScopedCFTypeRef<SecIdentityRef> preferred_identity;
1219 if (!server_domain.empty()) {
1220 // See if there's an identity preference for this domain:
1221 ScopedCFTypeRef<CFStringRef> domain_str(
1222 base::SysUTF8ToCFStringRef("https://" + server_domain));
1223 SecIdentityRef identity = NULL;
1224 // While SecIdentityCopyPreferences appears to take a list of CA issuers
1225 // to restrict the identity search to, within Security.framework the
1226 // argument is ignored and filtering unimplemented. See
1227 // SecIdentity.cpp in libsecurity_keychain, specifically
1228 // _SecIdentityCopyPreferenceMatchingName().
1229 if (SecIdentityCopyPreference(domain_str, 0, NULL, &identity) == noErr)
1230 preferred_identity.reset(identity);
1231 }
1232
1233 // Now enumerate the identities in the available keychains.
1234 SecIdentitySearchRef search = nil;
1235 OSStatus err = SecIdentitySearchCreate(NULL, CSSM_KEYUSE_SIGN, &search);
1236 ScopedCFTypeRef<SecIdentitySearchRef> scoped_search(search);
1237 while (!err) {
1238 SecIdentityRef identity = NULL;
1239 err = SecIdentitySearchCopyNext(search, &identity);
1240 if (err)
1241 break;
1242 ScopedCFTypeRef<SecIdentityRef> scoped_identity(identity);
1243
1244 SecCertificateRef cert_handle;
1245 err = SecIdentityCopyCertificate(identity, &cert_handle);
1246 if (err != noErr)
1247 continue;
1248 ScopedCFTypeRef<SecCertificateRef> scoped_cert_handle(cert_handle);
1249
1250 scoped_refptr<X509Certificate> cert(
1251 CreateFromHandle(cert_handle, SOURCE_LONE_CERT_IMPORT,
1252 OSCertHandles()));
1253 if (cert->HasExpired() || !cert->SupportsSSLClientAuth())
1254 continue;
1255
1256 // Skip duplicates (a cert may be in multiple keychains).
1257 const SHA1Fingerprint& fingerprint = cert->fingerprint();
1258 unsigned i;
1259 for (i = 0; i < certs->size(); ++i) {
1260 if ((*certs)[i]->fingerprint().Equals(fingerprint))
1261 break;
1262 }
1263 if (i < certs->size())
1264 continue;
1265
1266 bool is_preferred = preferred_identity &&
1267 CFEqual(preferred_identity, identity);
1268
1269 // Make sure the issuer matches valid_issuers, if given.
1270 // But an explicit cert preference overrides this.
1271 if (!is_preferred &&
1272 !valid_issuers.empty() &&
1273 !cert->IsIssuedBy(valid_issuers))
1274 continue;
1275
1276 // The cert passes, so add it to the vector.
1277 // If it's the preferred identity, add it at the start (so it'll be
1278 // selected by default in the UI.)
1279 if (is_preferred)
1280 certs->insert(certs->begin(), cert);
1281 else
1282 certs->push_back(cert);
1283 }
1284
1285 if (err != errSecItemNotFound) {
1286 LOG(ERROR) << "SecIdentitySearch error " << err;
1287 return false;
1288 }
1289 return true;
1290 }
1291
CreateClientCertificateChain() const1292 CFArrayRef X509Certificate::CreateClientCertificateChain() const {
1293 // Initialize the result array with just the IdentityRef of the receiver:
1294 OSStatus result;
1295 SecIdentityRef identity;
1296 result = SecIdentityCreateWithCertificate(NULL, cert_handle_, &identity);
1297 if (result) {
1298 LOG(ERROR) << "SecIdentityCreateWithCertificate error " << result;
1299 return NULL;
1300 }
1301 ScopedCFTypeRef<CFMutableArrayRef> chain(
1302 CFArrayCreateMutable(NULL, 0, &kCFTypeArrayCallBacks));
1303 CFArrayAppendValue(chain, identity);
1304
1305 CFArrayRef cert_chain = NULL;
1306 result = CopyCertChain(cert_handle_, &cert_chain);
1307 ScopedCFTypeRef<CFArrayRef> scoped_cert_chain(cert_chain);
1308 if (result) {
1309 LOG(ERROR) << "CreateIdentityCertificateChain error " << result;
1310 return chain.release();
1311 }
1312
1313 // Append the intermediate certs from SecTrust to the result array:
1314 if (cert_chain) {
1315 int chain_count = CFArrayGetCount(cert_chain);
1316 if (chain_count > 1) {
1317 CFArrayAppendArray(chain,
1318 cert_chain,
1319 CFRangeMake(1, chain_count - 1));
1320 }
1321 }
1322
1323 return chain.release();
1324 }
1325
1326 // static
1327 X509Certificate::OSCertHandle
ReadCertHandleFromPickle(const Pickle & pickle,void ** pickle_iter)1328 X509Certificate::ReadCertHandleFromPickle(const Pickle& pickle,
1329 void** pickle_iter) {
1330 const char* data;
1331 int length;
1332 if (!pickle.ReadData(pickle_iter, &data, &length))
1333 return NULL;
1334
1335 return CreateOSCertHandleFromBytes(data, length);
1336 }
1337
1338 // static
WriteCertHandleToPickle(OSCertHandle cert_handle,Pickle * pickle)1339 bool X509Certificate::WriteCertHandleToPickle(OSCertHandle cert_handle,
1340 Pickle* pickle) {
1341 CSSM_DATA cert_data;
1342 OSStatus status = SecCertificateGetData(cert_handle, &cert_data);
1343 if (status)
1344 return false;
1345
1346 return pickle->WriteData(reinterpret_cast<char*>(cert_data.Data),
1347 cert_data.Length);
1348 }
1349
1350 } // namespace net
1351