// 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 <pk11pub.h>
#include <secerr.h>
#include <sechash.h>
#include <stdlib.h>
#include "base/logging.h"
#include "crypto/nss_util.h"
#include "crypto/third_party/nss/chromium-nss.h"
namespace crypto {
namespace {
HASH_HashType ToNSSHashType(SignatureVerifier::HashAlgorithm hash_alg) {
switch (hash_alg) {
case SignatureVerifier::SHA1:
return HASH_AlgSHA1;
case SignatureVerifier::SHA256:
return HASH_AlgSHA256;
}
return HASH_AlgNULL;
}
SECStatus VerifyRSAPSS_End(SECKEYPublicKey* public_key,
HASHContext* hash_context,
HASH_HashType mask_hash_alg,
unsigned int salt_len,
const unsigned char* signature,
unsigned int signature_len) {
unsigned int hash_len = HASH_ResultLenContext(hash_context);
std::vector<unsigned char> hash(hash_len);
HASH_End(hash_context, &hash[0], &hash_len, hash.size());
unsigned int modulus_len = SECKEY_PublicKeyStrength(public_key);
if (signature_len != modulus_len) {
PORT_SetError(SEC_ERROR_BAD_SIGNATURE);
return SECFailure;
}
std::vector<unsigned char> enc(signature_len);
SECStatus rv = PK11_PubEncryptRaw(public_key, &enc[0],
const_cast<unsigned char*>(signature),
signature_len, NULL);
if (rv != SECSuccess) {
LOG(WARNING) << "PK11_PubEncryptRaw failed";
return rv;
}
return emsa_pss_verify(&hash[0], &enc[0], enc.size(),
HASH_GetType(hash_context), mask_hash_alg,
salt_len);
}
} // namespace
SignatureVerifier::SignatureVerifier()
: vfy_context_(NULL),
hash_alg_(SHA1),
mask_hash_alg_(SHA1),
salt_len_(0),
public_key_(NULL),
hash_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) {
if (vfy_context_ || hash_context_)
return false;
signature_.assign(signature, signature + signature_len);
SECKEYPublicKey* public_key = DecodePublicKeyInfo(public_key_info,
public_key_info_len);
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,
SEC_ASN1_GET(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;
}
bool SignatureVerifier::VerifyInitRSAPSS(HashAlgorithm hash_alg,
HashAlgorithm mask_hash_alg,
int salt_len,
const uint8* signature,
int signature_len,
const uint8* public_key_info,
int public_key_info_len) {
if (vfy_context_ || hash_context_)
return false;
signature_.assign(signature, signature + signature_len);
SECKEYPublicKey* public_key = DecodePublicKeyInfo(public_key_info,
public_key_info_len);
if (!public_key)
return false;
public_key_ = public_key;
hash_alg_ = hash_alg;
mask_hash_alg_ = mask_hash_alg;
salt_len_ = salt_len;
hash_context_ = HASH_Create(ToNSSHashType(hash_alg_));
if (!hash_context_)
return false;
HASH_Begin(hash_context_);
return true;
}
void SignatureVerifier::VerifyUpdate(const uint8* data_part,
int data_part_len) {
if (vfy_context_) {
SECStatus rv = VFY_Update(vfy_context_, data_part, data_part_len);
DCHECK_EQ(SECSuccess, rv);
} else {
HASH_Update(hash_context_, data_part, data_part_len);
}
}
bool SignatureVerifier::VerifyFinal() {
SECStatus rv;
if (vfy_context_) {
rv = VFY_End(vfy_context_);
} else {
rv = VerifyRSAPSS_End(public_key_, hash_context_,
ToNSSHashType(mask_hash_alg_), salt_len_,
signature_.data(),
signature_.size());
}
Reset();
// If signature verification fails, the error code is
// SEC_ERROR_BAD_SIGNATURE (-8182).
return (rv == SECSuccess);
}
// static
SECKEYPublicKey* SignatureVerifier::DecodePublicKeyInfo(
const uint8* public_key_info,
int public_key_info_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 NULL;
SECKEYPublicKey* public_key = SECKEY_ExtractPublicKey(spki);
SECKEY_DestroySubjectPublicKeyInfo(spki); // Done with spki.
return public_key;
}
void SignatureVerifier::Reset() {
if (vfy_context_) {
VFY_DestroyContext(vfy_context_, PR_TRUE);
vfy_context_ = NULL;
}
if (hash_context_) {
HASH_Destroy(hash_context_);
hash_context_ = NULL;
}
if (public_key_) {
SECKEY_DestroyPublicKey(public_key_);
public_key_ = NULL;
}
signature_.clear();
}
} // namespace crypto