| Index: crypto/signature_verifier_nss.cc
|
| ===================================================================
|
| --- crypto/signature_verifier_nss.cc (revision 0)
|
| +++ crypto/signature_verifier_nss.cc (revision 0)
|
| @@ -0,0 +1,113 @@
|
| +// Copyright (c) 2011 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 "crypto/signature_verifier.h"
|
| +
|
| +#include <cryptohi.h>
|
| +#include <keyhi.h>
|
| +#include <stdlib.h>
|
| +
|
| +#include "base/logging.h"
|
| +#include "crypto/nss_util.h"
|
| +
|
| +namespace crypto {
|
| +
|
| +SignatureVerifier::SignatureVerifier() : vfy_context_(NULL) {
|
| + EnsureNSSInit();
|
| +}
|
| +
|
| +SignatureVerifier::~SignatureVerifier() {
|
| + Reset();
|
| +}
|
| +
|
| +bool SignatureVerifier::VerifyInit(const uint8* signature_algorithm,
|
| + int signature_algorithm_len,
|
| + const uint8* signature,
|
| + int signature_len,
|
| + const uint8* public_key_info,
|
| + int public_key_info_len) {
|
| + signature_.assign(signature, signature + signature_len);
|
| +
|
| + CERTSubjectPublicKeyInfo* spki = NULL;
|
| + SECItem spki_der;
|
| + spki_der.type = siBuffer;
|
| + spki_der.data = const_cast<uint8*>(public_key_info);
|
| + spki_der.len = public_key_info_len;
|
| + spki = SECKEY_DecodeDERSubjectPublicKeyInfo(&spki_der);
|
| + if (!spki)
|
| + return false;
|
| + SECKEYPublicKey* public_key = SECKEY_ExtractPublicKey(spki);
|
| + SECKEY_DestroySubjectPublicKeyInfo(spki); // Done with spki.
|
| + if (!public_key)
|
| + return false;
|
| +
|
| + PLArenaPool* arena = PORT_NewArena(DER_DEFAULT_CHUNKSIZE);
|
| + if (!arena) {
|
| + SECKEY_DestroyPublicKey(public_key);
|
| + return false;
|
| + }
|
| +
|
| + SECItem sig_alg_der;
|
| + sig_alg_der.type = siBuffer;
|
| + sig_alg_der.data = const_cast<uint8*>(signature_algorithm);
|
| + sig_alg_der.len = signature_algorithm_len;
|
| + SECAlgorithmID sig_alg_id;
|
| + SECStatus rv;
|
| + rv = SEC_QuickDERDecodeItem(arena, &sig_alg_id, SECOID_AlgorithmIDTemplate,
|
| + &sig_alg_der);
|
| + if (rv != SECSuccess) {
|
| + SECKEY_DestroyPublicKey(public_key);
|
| + PORT_FreeArena(arena, PR_TRUE);
|
| + return false;
|
| + }
|
| +
|
| + SECItem sig;
|
| + sig.type = siBuffer;
|
| + sig.data = const_cast<uint8*>(signature);
|
| + sig.len = signature_len;
|
| + SECOidTag hash_alg_tag;
|
| + vfy_context_ = VFY_CreateContextWithAlgorithmID(public_key, &sig,
|
| + &sig_alg_id, &hash_alg_tag,
|
| + NULL);
|
| + SECKEY_DestroyPublicKey(public_key); // Done with public_key.
|
| + PORT_FreeArena(arena, PR_TRUE); // Done with sig_alg_id.
|
| + if (!vfy_context_) {
|
| + // A corrupted RSA signature could be detected without the data, so
|
| + // VFY_CreateContextWithAlgorithmID may fail with SEC_ERROR_BAD_SIGNATURE
|
| + // (-8182).
|
| + return false;
|
| + }
|
| +
|
| + rv = VFY_Begin(vfy_context_);
|
| + if (rv != SECSuccess) {
|
| + NOTREACHED();
|
| + return false;
|
| + }
|
| + return true;
|
| +}
|
| +
|
| +void SignatureVerifier::VerifyUpdate(const uint8* data_part,
|
| + int data_part_len) {
|
| + SECStatus rv = VFY_Update(vfy_context_, data_part, data_part_len);
|
| + DCHECK(rv == SECSuccess);
|
| +}
|
| +
|
| +bool SignatureVerifier::VerifyFinal() {
|
| + SECStatus rv = VFY_End(vfy_context_);
|
| + Reset();
|
| +
|
| + // If signature verification fails, the error code is
|
| + // SEC_ERROR_BAD_SIGNATURE (-8182).
|
| + return (rv == SECSuccess);
|
| +}
|
| +
|
| +void SignatureVerifier::Reset() {
|
| + if (vfy_context_) {
|
| + VFY_DestroyContext(vfy_context_, PR_TRUE);
|
| + vfy_context_ = NULL;
|
| + }
|
| + signature_.clear();
|
| +}
|
| +
|
| +} // namespace crypto
|
|
|
| Property changes on: crypto\signature_verifier_nss.cc
|
| ___________________________________________________________________
|
| Added: svn:eol-style
|
| + LF
|
|
|
|
|