Skip to content

Commit 6cd698d

Browse files
hinmanrsc
authored andcommitted
crypto/x509: add Admin & User Keychains to FetchPEMRoots on Darwin
in root_cgo_darwin.go only certificates from the System Domain were being used in FetchPEMRoots. This patch adds support for getting certificates from all three domains (System, Admin, User). Also it will only read trusted certificates from those Keychains. Because it is possible to trust a non Root certificate, this patch also adds a checks to see if the Subject and Issuer name are the same. Fixes #14514 Change-Id: Ia03936d7a61d1e24e99f31c92f9927ae48b2b494 Reviewed-on: https://go-review.googlesource.com/20351 Reviewed-by: Russ Cox <rsc@golang.org>
1 parent b30fcbc commit 6cd698d

File tree

1 file changed

+52
-24
lines changed

1 file changed

+52
-24
lines changed

src/crypto/x509/root_cgo_darwin.go

+52-24
Original file line numberDiff line numberDiff line change
@@ -21,41 +21,69 @@ package x509
2121
// Note: The CFDataRef returned in pemRoots must be released (using CFRelease) after
2222
// we've consumed its content.
2323
int FetchPEMRoots(CFDataRef *pemRoots) {
24-
if (pemRoots == NULL) {
25-
return -1;
26-
}
24+
// Get certificates from all domains, not just System, this lets
25+
// the user add CAs to their "login" keychain, and Admins to add
26+
// to the "System" keychain
27+
SecTrustSettingsDomain domains[] = { kSecTrustSettingsDomainSystem,
28+
kSecTrustSettingsDomainAdmin,
29+
kSecTrustSettingsDomainUser };
2730
28-
CFArrayRef certs = NULL;
29-
OSStatus err = SecTrustCopyAnchorCertificates(&certs);
30-
if (err != noErr) {
31+
int numDomains = sizeof(domains)/sizeof(SecTrustSettingsDomain);
32+
if (pemRoots == NULL) {
3133
return -1;
3234
}
3335
3436
CFMutableDataRef combinedData = CFDataCreateMutable(kCFAllocatorDefault, 0);
35-
int i, ncerts = CFArrayGetCount(certs);
36-
for (i = 0; i < ncerts; i++) {
37-
CFDataRef data = NULL;
38-
SecCertificateRef cert = (SecCertificateRef)CFArrayGetValueAtIndex(certs, i);
39-
if (cert == NULL) {
40-
continue;
41-
}
42-
43-
// Note: SecKeychainItemExport is deprecated as of 10.7 in favor of SecItemExport.
44-
// Once we support weak imports via cgo we should prefer that, and fall back to this
45-
// for older systems.
46-
err = SecKeychainItemExport(cert, kSecFormatX509Cert, kSecItemPemArmour, NULL, &data);
37+
for (int i = 0; i < numDomains; i++) {
38+
CFArrayRef certs = NULL;
39+
// Only get certificates from domain that are trusted
40+
OSStatus err = SecTrustSettingsCopyCertificates(domains[i], &certs);
4741
if (err != noErr) {
4842
continue;
4943
}
5044
51-
if (data != NULL) {
52-
CFDataAppendBytes(combinedData, CFDataGetBytePtr(data), CFDataGetLength(data));
53-
CFRelease(data);
54-
}
55-
}
45+
int numCerts = CFArrayGetCount(certs);
46+
for (int j = 0; j < numCerts; j++) {
47+
CFDataRef data = NULL;
48+
CFErrorRef errRef = NULL;
49+
SecCertificateRef cert = (SecCertificateRef)CFArrayGetValueAtIndex(certs, j);
50+
if (cert == NULL) {
51+
continue;
52+
}
53+
// We only want to add Root CAs, so make sure Subject and Issuer Name match
54+
CFDataRef subjectName = SecCertificateCopyNormalizedSubjectContent(cert, &errRef);
55+
if (errRef != NULL) {
56+
CFRelease(errRef);
57+
continue;
58+
}
59+
CFDataRef issuerName = SecCertificateCopyNormalizedIssuerContent(cert, &errRef);
60+
if (errRef != NULL) {
61+
CFRelease(subjectName);
62+
CFRelease(errRef);
63+
continue;
64+
}
65+
Boolean equal = CFEqual(subjectName, issuerName);
66+
CFRelease(subjectName);
67+
CFRelease(issuerName);
68+
if (!equal) {
69+
continue;
70+
}
5671
57-
CFRelease(certs);
72+
// Note: SecKeychainItemExport is deprecated as of 10.7 in favor of SecItemExport.
73+
// Once we support weak imports via cgo we should prefer that, and fall back to this
74+
// for older systems.
75+
err = SecKeychainItemExport(cert, kSecFormatX509Cert, kSecItemPemArmour, NULL, &data);
76+
if (err != noErr) {
77+
continue;
78+
}
5879
80+
if (data != NULL) {
81+
CFDataAppendBytes(combinedData, CFDataGetBytePtr(data), CFDataGetLength(data));
82+
CFRelease(data);
83+
}
84+
}
85+
CFRelease(certs);
86+
}
5987
*pemRoots = combinedData;
6088
return 0;
6189
}

0 commit comments

Comments
 (0)