mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-28 08:16:09 +03:00
514 lines
17 KiB
C++
514 lines
17 KiB
C++
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "net/cert/nss_cert_database.h"
|
|
|
|
#include <cert.h>
|
|
#include <certdb.h>
|
|
#include <keyhi.h>
|
|
#include <pk11pub.h>
|
|
#include <secmod.h>
|
|
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
#include "base/bind.h"
|
|
#include "base/callback.h"
|
|
#include "base/logging.h"
|
|
#include "base/macros.h"
|
|
#include "base/observer_list_threadsafe.h"
|
|
#include "base/task/post_task.h"
|
|
#include "base/threading/scoped_blocking_call.h"
|
|
#include "crypto/scoped_nss_types.h"
|
|
#include "net/base/net_errors.h"
|
|
#include "net/cert/cert_database.h"
|
|
#include "net/cert/x509_certificate.h"
|
|
#include "net/cert/x509_util_nss.h"
|
|
#include "net/third_party/mozilla_security_manager/nsNSSCertificateDB.h"
|
|
#include "net/third_party/mozilla_security_manager/nsPKCS12Blob.h"
|
|
|
|
// PSM = Mozilla's Personal Security Manager.
|
|
namespace psm = mozilla_security_manager;
|
|
|
|
namespace net {
|
|
|
|
namespace {
|
|
|
|
// TODO(pneubeck): Move this class out of NSSCertDatabase and to the caller of
|
|
// the c'tor of NSSCertDatabase, see https://crbug.com/395983 .
|
|
// Helper that observes events from the NSSCertDatabase and forwards them to
|
|
// the given CertDatabase.
|
|
class CertNotificationForwarder : public NSSCertDatabase::Observer {
|
|
public:
|
|
explicit CertNotificationForwarder(CertDatabase* cert_db)
|
|
: cert_db_(cert_db) {}
|
|
|
|
~CertNotificationForwarder() override = default;
|
|
|
|
void OnCertDBChanged() override { cert_db_->NotifyObserversCertDBChanged(); }
|
|
|
|
private:
|
|
CertDatabase* cert_db_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(CertNotificationForwarder);
|
|
};
|
|
|
|
} // namespace
|
|
|
|
NSSCertDatabase::ImportCertFailure::ImportCertFailure(
|
|
ScopedCERTCertificate cert,
|
|
int err)
|
|
: certificate(std::move(cert)), net_error(err) {}
|
|
|
|
NSSCertDatabase::ImportCertFailure::ImportCertFailure(
|
|
ImportCertFailure&& other) = default;
|
|
|
|
NSSCertDatabase::ImportCertFailure::~ImportCertFailure() = default;
|
|
|
|
NSSCertDatabase::NSSCertDatabase(crypto::ScopedPK11Slot public_slot,
|
|
crypto::ScopedPK11Slot private_slot)
|
|
: public_slot_(std::move(public_slot)),
|
|
private_slot_(std::move(private_slot)),
|
|
observer_list_(new base::ObserverListThreadSafe<Observer>),
|
|
weak_factory_(this) {
|
|
CHECK(public_slot_);
|
|
|
|
CertDatabase* cert_db = CertDatabase::GetInstance();
|
|
cert_notification_forwarder_.reset(new CertNotificationForwarder(cert_db));
|
|
AddObserver(cert_notification_forwarder_.get());
|
|
|
|
psm::EnsurePKCS12Init();
|
|
}
|
|
|
|
NSSCertDatabase::~NSSCertDatabase() = default;
|
|
|
|
ScopedCERTCertificateList NSSCertDatabase::ListCertsSync() {
|
|
return ListCertsImpl(crypto::ScopedPK11Slot());
|
|
}
|
|
|
|
void NSSCertDatabase::ListCerts(const ListCertsCallback& callback) {
|
|
base::PostTaskWithTraitsAndReplyWithResult(
|
|
FROM_HERE,
|
|
{base::MayBlock(), base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
|
|
base::Bind(&NSSCertDatabase::ListCertsImpl,
|
|
base::Passed(crypto::ScopedPK11Slot())),
|
|
callback);
|
|
}
|
|
|
|
void NSSCertDatabase::ListCertsInSlot(const ListCertsCallback& callback,
|
|
PK11SlotInfo* slot) {
|
|
DCHECK(slot);
|
|
base::PostTaskWithTraitsAndReplyWithResult(
|
|
FROM_HERE,
|
|
{base::MayBlock(), base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
|
|
base::Bind(
|
|
&NSSCertDatabase::ListCertsImpl,
|
|
base::Passed(crypto::ScopedPK11Slot(PK11_ReferenceSlot(slot)))),
|
|
callback);
|
|
}
|
|
|
|
#if defined(OS_CHROMEOS)
|
|
crypto::ScopedPK11Slot NSSCertDatabase::GetSystemSlot() const {
|
|
return crypto::ScopedPK11Slot();
|
|
}
|
|
#endif
|
|
|
|
crypto::ScopedPK11Slot NSSCertDatabase::GetPublicSlot() const {
|
|
return crypto::ScopedPK11Slot(PK11_ReferenceSlot(public_slot_.get()));
|
|
}
|
|
|
|
crypto::ScopedPK11Slot NSSCertDatabase::GetPrivateSlot() const {
|
|
if (!private_slot_)
|
|
return crypto::ScopedPK11Slot();
|
|
return crypto::ScopedPK11Slot(PK11_ReferenceSlot(private_slot_.get()));
|
|
}
|
|
|
|
void NSSCertDatabase::ListModules(std::vector<crypto::ScopedPK11Slot>* modules,
|
|
bool need_rw) const {
|
|
modules->clear();
|
|
|
|
// The wincx arg is unused since we don't call PK11_SetIsLoggedInFunc.
|
|
crypto::ScopedPK11SlotList slot_list(
|
|
PK11_GetAllTokens(CKM_INVALID_MECHANISM,
|
|
need_rw ? PR_TRUE : PR_FALSE, // needRW
|
|
PR_TRUE, // loadCerts (unused)
|
|
nullptr)); // wincx
|
|
if (!slot_list) {
|
|
LOG(ERROR) << "PK11_GetAllTokens failed: " << PORT_GetError();
|
|
return;
|
|
}
|
|
|
|
PK11SlotListElement* slot_element = PK11_GetFirstSafe(slot_list.get());
|
|
while (slot_element) {
|
|
modules->push_back(
|
|
crypto::ScopedPK11Slot(PK11_ReferenceSlot(slot_element->slot)));
|
|
slot_element = PK11_GetNextSafe(slot_list.get(), slot_element,
|
|
PR_FALSE); // restart
|
|
}
|
|
}
|
|
|
|
int NSSCertDatabase::ImportFromPKCS12(
|
|
PK11SlotInfo* slot_info,
|
|
const std::string& data,
|
|
const base::string16& password,
|
|
bool is_extractable,
|
|
ScopedCERTCertificateList* imported_certs) {
|
|
DVLOG(1) << __func__ << " "
|
|
<< PK11_GetModuleID(slot_info) << ":"
|
|
<< PK11_GetSlotID(slot_info);
|
|
int result = psm::nsPKCS12Blob_Import(slot_info,
|
|
data.data(), data.size(),
|
|
password,
|
|
is_extractable,
|
|
imported_certs);
|
|
if (result == OK)
|
|
NotifyObserversCertDBChanged();
|
|
|
|
return result;
|
|
}
|
|
|
|
int NSSCertDatabase::ExportToPKCS12(const ScopedCERTCertificateList& certs,
|
|
const base::string16& password,
|
|
std::string* output) const {
|
|
return psm::nsPKCS12Blob_Export(output, certs, password);
|
|
}
|
|
|
|
CERTCertificate* NSSCertDatabase::FindRootInList(
|
|
const ScopedCERTCertificateList& certificates) const {
|
|
DCHECK_GT(certificates.size(), 0U);
|
|
|
|
if (certificates.size() == 1)
|
|
return certificates[0].get();
|
|
|
|
CERTCertificate* cert0 = certificates[0].get();
|
|
CERTCertificate* cert1 = certificates[1].get();
|
|
CERTCertificate* certn_2 = certificates[certificates.size() - 2].get();
|
|
CERTCertificate* certn_1 = certificates[certificates.size() - 1].get();
|
|
|
|
// Using CERT_CompareName is an alternative, except that it is broken until
|
|
// NSS 3.32 (see https://bugzilla.mozilla.org/show_bug.cgi?id=1361197 ).
|
|
if (SECITEM_CompareItem(&cert1->derIssuer, &cert0->derSubject) == SECEqual)
|
|
return cert0;
|
|
|
|
if (SECITEM_CompareItem(&certn_2->derIssuer, &certn_1->derSubject) ==
|
|
SECEqual) {
|
|
return certn_1;
|
|
}
|
|
|
|
LOG(WARNING) << "certificate list is not a hierarchy";
|
|
return cert0;
|
|
}
|
|
|
|
int NSSCertDatabase::ImportUserCert(const std::string& data) {
|
|
ScopedCERTCertificateList certificates =
|
|
x509_util::CreateCERTCertificateListFromBytes(
|
|
data.c_str(), data.size(), net::X509Certificate::FORMAT_AUTO);
|
|
if (certificates.empty())
|
|
return ERR_CERT_INVALID;
|
|
|
|
int result = psm::ImportUserCert(certificates[0].get());
|
|
|
|
if (result == OK)
|
|
NotifyObserversCertDBChanged();
|
|
|
|
return result;
|
|
}
|
|
|
|
int NSSCertDatabase::ImportUserCert(CERTCertificate* cert) {
|
|
int result = psm::ImportUserCert(cert);
|
|
|
|
if (result == OK)
|
|
NotifyObserversCertDBChanged();
|
|
|
|
return result;
|
|
}
|
|
|
|
bool NSSCertDatabase::ImportCACerts(
|
|
const ScopedCERTCertificateList& certificates,
|
|
TrustBits trust_bits,
|
|
ImportCertFailureList* not_imported) {
|
|
crypto::ScopedPK11Slot slot(GetPublicSlot());
|
|
CERTCertificate* root = FindRootInList(certificates);
|
|
|
|
bool success = psm::ImportCACerts(slot.get(), certificates, root, trust_bits,
|
|
not_imported);
|
|
if (success)
|
|
NotifyObserversCertDBChanged();
|
|
|
|
return success;
|
|
}
|
|
|
|
bool NSSCertDatabase::ImportServerCert(
|
|
const ScopedCERTCertificateList& certificates,
|
|
TrustBits trust_bits,
|
|
ImportCertFailureList* not_imported) {
|
|
crypto::ScopedPK11Slot slot(GetPublicSlot());
|
|
return psm::ImportServerCert(slot.get(), certificates, trust_bits,
|
|
not_imported);
|
|
}
|
|
|
|
NSSCertDatabase::TrustBits NSSCertDatabase::GetCertTrust(
|
|
const CERTCertificate* cert,
|
|
CertType type) const {
|
|
CERTCertTrust trust;
|
|
SECStatus srv = CERT_GetCertTrust(cert, &trust);
|
|
if (srv != SECSuccess) {
|
|
LOG(ERROR) << "CERT_GetCertTrust failed with error " << PORT_GetError();
|
|
return TRUST_DEFAULT;
|
|
}
|
|
// We define our own more "friendly" TrustBits, which means we aren't able to
|
|
// round-trip all possible NSS trust flag combinations. We try to map them in
|
|
// a sensible way.
|
|
switch (type) {
|
|
case CA_CERT: {
|
|
const unsigned kTrustedCA = CERTDB_TRUSTED_CA | CERTDB_TRUSTED_CLIENT_CA;
|
|
const unsigned kCAFlags = kTrustedCA | CERTDB_TERMINAL_RECORD;
|
|
|
|
TrustBits trust_bits = TRUST_DEFAULT;
|
|
if ((trust.sslFlags & kCAFlags) == CERTDB_TERMINAL_RECORD)
|
|
trust_bits |= DISTRUSTED_SSL;
|
|
else if (trust.sslFlags & kTrustedCA)
|
|
trust_bits |= TRUSTED_SSL;
|
|
|
|
if ((trust.emailFlags & kCAFlags) == CERTDB_TERMINAL_RECORD)
|
|
trust_bits |= DISTRUSTED_EMAIL;
|
|
else if (trust.emailFlags & kTrustedCA)
|
|
trust_bits |= TRUSTED_EMAIL;
|
|
|
|
if ((trust.objectSigningFlags & kCAFlags) == CERTDB_TERMINAL_RECORD)
|
|
trust_bits |= DISTRUSTED_OBJ_SIGN;
|
|
else if (trust.objectSigningFlags & kTrustedCA)
|
|
trust_bits |= TRUSTED_OBJ_SIGN;
|
|
|
|
return trust_bits;
|
|
}
|
|
case SERVER_CERT:
|
|
if (trust.sslFlags & CERTDB_TERMINAL_RECORD) {
|
|
if (trust.sslFlags & CERTDB_TRUSTED)
|
|
return TRUSTED_SSL;
|
|
return DISTRUSTED_SSL;
|
|
}
|
|
return TRUST_DEFAULT;
|
|
default:
|
|
return TRUST_DEFAULT;
|
|
}
|
|
}
|
|
|
|
bool NSSCertDatabase::IsUntrusted(const CERTCertificate* cert) const {
|
|
CERTCertTrust nsstrust;
|
|
SECStatus rv = CERT_GetCertTrust(cert, &nsstrust);
|
|
if (rv != SECSuccess) {
|
|
LOG(ERROR) << "CERT_GetCertTrust failed with error " << PORT_GetError();
|
|
return false;
|
|
}
|
|
|
|
// The CERTCertTrust structure contains three trust records:
|
|
// sslFlags, emailFlags, and objectSigningFlags. The three
|
|
// trust records are independent of each other.
|
|
//
|
|
// If the CERTDB_TERMINAL_RECORD bit in a trust record is set,
|
|
// then that trust record is a terminal record. A terminal
|
|
// record is used for explicit trust and distrust of an
|
|
// end-entity or intermediate CA cert.
|
|
//
|
|
// In a terminal record, if neither CERTDB_TRUSTED_CA nor
|
|
// CERTDB_TRUSTED is set, then the terminal record means
|
|
// explicit distrust. On the other hand, if the terminal
|
|
// record has either CERTDB_TRUSTED_CA or CERTDB_TRUSTED bit
|
|
// set, then the terminal record means explicit trust.
|
|
//
|
|
// For a root CA, the trust record does not have
|
|
// the CERTDB_TERMINAL_RECORD bit set.
|
|
|
|
static const unsigned int kTrusted = CERTDB_TRUSTED_CA | CERTDB_TRUSTED;
|
|
if ((nsstrust.sslFlags & CERTDB_TERMINAL_RECORD) != 0 &&
|
|
(nsstrust.sslFlags & kTrusted) == 0) {
|
|
return true;
|
|
}
|
|
if ((nsstrust.emailFlags & CERTDB_TERMINAL_RECORD) != 0 &&
|
|
(nsstrust.emailFlags & kTrusted) == 0) {
|
|
return true;
|
|
}
|
|
if ((nsstrust.objectSigningFlags & CERTDB_TERMINAL_RECORD) != 0 &&
|
|
(nsstrust.objectSigningFlags & kTrusted) == 0) {
|
|
return true;
|
|
}
|
|
|
|
// Self-signed certificates that don't have any trust bits set are untrusted.
|
|
// Other certificates that don't have any trust bits set may still be trusted
|
|
// if they chain up to a trust anchor.
|
|
if (SECITEM_CompareItem(&cert->derIssuer, &cert->derSubject) == SECEqual) {
|
|
return (nsstrust.sslFlags & kTrusted) == 0 &&
|
|
(nsstrust.emailFlags & kTrusted) == 0 &&
|
|
(nsstrust.objectSigningFlags & kTrusted) == 0;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool NSSCertDatabase::IsWebTrustAnchor(const CERTCertificate* cert) const {
|
|
CERTCertTrust nsstrust;
|
|
SECStatus rv = CERT_GetCertTrust(cert, &nsstrust);
|
|
if (rv != SECSuccess) {
|
|
LOG(ERROR) << "CERT_GetCertTrust failed with error " << PORT_GetError();
|
|
return false;
|
|
}
|
|
|
|
// Note: This should return true iff a net::TrustStoreNSS instantiated with
|
|
// SECTrustType trustSSL would classify |cert| as a trust anchor.
|
|
const unsigned int ssl_trust_flags = nsstrust.sslFlags;
|
|
|
|
// Determine if the certificate is a trust anchor.
|
|
if ((ssl_trust_flags & CERTDB_TRUSTED_CA) == CERTDB_TRUSTED_CA) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool NSSCertDatabase::SetCertTrust(CERTCertificate* cert,
|
|
CertType type,
|
|
TrustBits trust_bits) {
|
|
bool success = psm::SetCertTrust(cert, type, trust_bits);
|
|
if (success)
|
|
NotifyObserversCertDBChanged();
|
|
|
|
return success;
|
|
}
|
|
|
|
bool NSSCertDatabase::DeleteCertAndKey(CERTCertificate* cert) {
|
|
if (!DeleteCertAndKeyImpl(cert))
|
|
return false;
|
|
NotifyObserversCertDBChanged();
|
|
return true;
|
|
}
|
|
|
|
void NSSCertDatabase::DeleteCertAndKeyAsync(
|
|
ScopedCERTCertificate cert,
|
|
const DeleteCertCallback& callback) {
|
|
base::PostTaskWithTraitsAndReplyWithResult(
|
|
FROM_HERE,
|
|
{base::MayBlock(), base::TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN},
|
|
base::BindOnce(&NSSCertDatabase::DeleteCertAndKeyImplScoped,
|
|
std::move(cert)),
|
|
base::BindOnce(&NSSCertDatabase::NotifyCertRemovalAndCallBack,
|
|
weak_factory_.GetWeakPtr(), callback));
|
|
}
|
|
|
|
bool NSSCertDatabase::IsReadOnly(const CERTCertificate* cert) const {
|
|
PK11SlotInfo* slot = cert->slot;
|
|
return slot && PK11_IsReadOnly(slot);
|
|
}
|
|
|
|
bool NSSCertDatabase::IsHardwareBacked(const CERTCertificate* cert) const {
|
|
PK11SlotInfo* slot = cert->slot;
|
|
return slot && PK11_IsHW(slot);
|
|
}
|
|
|
|
void NSSCertDatabase::LogUserCertificates(const std::string& log_reason) const {
|
|
}
|
|
|
|
void NSSCertDatabase::AddObserver(Observer* observer) {
|
|
observer_list_->AddObserver(observer);
|
|
}
|
|
|
|
void NSSCertDatabase::RemoveObserver(Observer* observer) {
|
|
observer_list_->RemoveObserver(observer);
|
|
}
|
|
|
|
// static
|
|
ScopedCERTCertificateList NSSCertDatabase::ListCertsImpl(
|
|
crypto::ScopedPK11Slot slot) {
|
|
// This method may acquire the NSS lock or reenter this code via extension
|
|
// hooks (such as smart card UI). To ensure threads are not starved or
|
|
// deadlocked, the base::ScopedBlockingCall below increments the thread pool
|
|
// capacity if this method takes too much time to run.
|
|
base::ScopedBlockingCall scoped_blocking_call(base::BlockingType::MAY_BLOCK);
|
|
|
|
ScopedCERTCertificateList certs;
|
|
CERTCertList* cert_list = nullptr;
|
|
if (slot)
|
|
cert_list = PK11_ListCertsInSlot(slot.get());
|
|
else
|
|
cert_list = PK11_ListCerts(PK11CertListUnique, nullptr);
|
|
|
|
CERTCertListNode* node;
|
|
for (node = CERT_LIST_HEAD(cert_list); !CERT_LIST_END(node, cert_list);
|
|
node = CERT_LIST_NEXT(node)) {
|
|
certs.push_back(x509_util::DupCERTCertificate(node->cert));
|
|
}
|
|
CERT_DestroyCertList(cert_list);
|
|
return certs;
|
|
}
|
|
|
|
void NSSCertDatabase::NotifyCertRemovalAndCallBack(
|
|
const DeleteCertCallback& callback,
|
|
bool success) {
|
|
if (success)
|
|
NotifyObserversCertDBChanged();
|
|
callback.Run(success);
|
|
}
|
|
|
|
void NSSCertDatabase::NotifyObserversCertDBChanged() {
|
|
LogUserCertificates("DBChanged");
|
|
|
|
observer_list_->Notify(FROM_HERE, &Observer::OnCertDBChanged);
|
|
}
|
|
|
|
// static
|
|
std::string NSSCertDatabase::GetCertIssuerCommonName(
|
|
const CERTCertificate* cert) {
|
|
char* nss_issuer_name = CERT_GetCommonName(&cert->issuer);
|
|
if (!nss_issuer_name)
|
|
return std::string();
|
|
|
|
std::string issuer_name = nss_issuer_name;
|
|
PORT_Free(nss_issuer_name);
|
|
|
|
return issuer_name;
|
|
}
|
|
|
|
// static
|
|
bool NSSCertDatabase::DeleteCertAndKeyImpl(CERTCertificate* cert) {
|
|
// This method may acquire the NSS lock or reenter this code via extension
|
|
// hooks (such as smart card UI). To ensure threads are not starved or
|
|
// deadlocked, the base::ScopedBlockingCall below increments the thread pool
|
|
// capacity if this method takes too much time to run.
|
|
base::ScopedBlockingCall scoped_blocking_call(base::BlockingType::MAY_BLOCK);
|
|
|
|
#if defined(OS_CHROMEOS)
|
|
// TODO(https://crbug.com/844537): Remove this after we've collected logs that
|
|
// show device-wide certificates disappearing.
|
|
std::string issuer_name = GetCertIssuerCommonName(cert);
|
|
VLOG(0) << "UserCertLogging: Deleting a certificate with issuer_name="
|
|
<< issuer_name;
|
|
#endif // defined(OS_CHROMEOS)
|
|
|
|
// For some reason, PK11_DeleteTokenCertAndKey only calls
|
|
// SEC_DeletePermCertificate if the private key is found. So, we check
|
|
// whether a private key exists before deciding which function to call to
|
|
// delete the cert.
|
|
SECKEYPrivateKey* privKey = PK11_FindKeyByAnyCert(cert, nullptr);
|
|
if (privKey) {
|
|
SECKEY_DestroyPrivateKey(privKey);
|
|
if (PK11_DeleteTokenCertAndKey(cert, nullptr)) {
|
|
LOG(ERROR) << "PK11_DeleteTokenCertAndKey failed: " << PORT_GetError();
|
|
return false;
|
|
}
|
|
} else {
|
|
if (SEC_DeletePermCertificate(cert)) {
|
|
LOG(ERROR) << "SEC_DeletePermCertificate failed: " << PORT_GetError();
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// static
|
|
bool NSSCertDatabase::DeleteCertAndKeyImplScoped(ScopedCERTCertificate cert) {
|
|
return NSSCertDatabase::DeleteCertAndKeyImpl(cert.get());
|
|
}
|
|
|
|
} // namespace net
|