cmssigdata.c

来自「支持SSL v2/v3, TLS, PKCS #5, PKCS #7, PKCS」· C语言 代码 · 共 887 行 · 第 1/2 页

C
887
字号
/* * The contents of this file are subject to the Mozilla Public * License Version 1.1 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.mozilla.org/MPL/ *  * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. *  * The Original Code is the Netscape security libraries. *  * The Initial Developer of the Original Code is Netscape * Communications Corporation.  Portions created by Netscape are  * Copyright (C) 1994-2000 Netscape Communications Corporation.  All * Rights Reserved. *  * Contributor(s): *  * Alternatively, the contents of this file may be used under the * terms of the GNU General Public License Version 2 or later (the * "GPL"), in which case the provisions of the GPL are applicable  * instead of those above.  If you wish to allow use of your  * version of this file only under the terms of the GPL and not to * allow others to use your version of this file under the MPL, * indicate your decision by deleting the provisions above and * replace them with the notice and other provisions required by * the GPL.  If you do not delete the provisions above, a recipient * may use your version of this file under either the MPL or the * GPL. *//* * CMS signedData methods. * * $Id: cmssigdata.c,v 1.10 2000/10/06 23:26:10 nelsonb%netscape.com Exp $ */#include "cmslocal.h"#include "cert.h"#include "cdbhdl.h"#include "secasn1.h"#include "secitem.h"#include "secoid.h"#include "pk11func.h"#include "secerr.h"NSSCMSSignedData *NSS_CMSSignedData_Create(NSSCMSMessage *cmsg){    void *mark;    NSSCMSSignedData *sigd;    PLArenaPool *poolp;    poolp = cmsg->poolp;    mark = PORT_ArenaMark(poolp);    sigd = (NSSCMSSignedData *)PORT_ArenaZAlloc (poolp, sizeof(NSSCMSSignedData));    if (sigd == NULL)	goto loser;    sigd->cmsg = cmsg;    /* signerInfos, certs, certlists, crls are all empty */    /* version is set in NSS_CMSSignedData_Finalize() */    PORT_ArenaUnmark(poolp, mark);    return sigd;loser:    PORT_ArenaRelease(poolp, mark);    return NULL;}voidNSS_CMSSignedData_Destroy(NSSCMSSignedData *sigd){    CERTCertificate **certs, *cert;    CERTCertificateList **certlists, *certlist;    NSSCMSSignerInfo **signerinfos, *si;    if (sigd == NULL)	return;    certs = sigd->certs;    certlists = sigd->certLists;    signerinfos = sigd->signerInfos;    if (certs != NULL) {	while ((cert = *certs++) != NULL)	    CERT_DestroyCertificate (cert);    }    if (certlists != NULL) {	while ((certlist = *certlists++) != NULL)	    CERT_DestroyCertificateList (certlist);    }    if (signerinfos != NULL) {	while ((si = *signerinfos++) != NULL)	    NSS_CMSSignerInfo_Destroy(si);    }    /* everything's in a pool, so don't worry about the storage */}/* * NSS_CMSSignedData_Encode_BeforeStart - do all the necessary things to a SignedData *     before start of encoding. * * In detail: *  - find out about the right value to put into sigd->version *  - come up with a list of digestAlgorithms (which should be the union of the algorithms *         in the signerinfos). *         If we happen to have a pre-set list of algorithms (and digest values!), we *         check if we have all the signerinfos' algorithms. If not, this is an error. */SECStatusNSS_CMSSignedData_Encode_BeforeStart(NSSCMSSignedData *sigd){    NSSCMSSignerInfo *signerinfo;    SECOidTag digestalgtag;    SECItem *dummy;    int version;    SECStatus rv;    PRBool haveDigests = PR_FALSE;    int n, i;    PLArenaPool *poolp;    poolp = sigd->cmsg->poolp;    /* we assume that we have precomputed digests if there is a list of algorithms, and */    /* a chunk of data for each of those algorithms */    if (sigd->digestAlgorithms != NULL && sigd->digests != NULL) {	for (i=0; sigd->digestAlgorithms[i] != NULL; i++) {	    if (sigd->digests[i] == NULL)		break;	}	if (sigd->digestAlgorithms[i] == NULL)	/* reached the end of the array? */	    haveDigests = PR_TRUE;		/* yes: we must have all the digests */    }	        version = NSS_CMS_SIGNED_DATA_VERSION_BASIC;    /* RFC2630 5.1 "version is the syntax version number..." */    if (NSS_CMSContentInfo_GetContentTypeTag(&(sigd->contentInfo)) != SEC_OID_PKCS7_DATA)	version = NSS_CMS_SIGNED_DATA_VERSION_EXT;    /* prepare all the SignerInfos (there may be none) */    for (i=0; i < NSS_CMSSignedData_SignerInfoCount(sigd); i++) {	signerinfo = NSS_CMSSignedData_GetSignerInfo(sigd, i);	/* RFC2630 5.1 "version is the syntax version number..." */	if (NSS_CMSSignerInfo_GetVersion(signerinfo) != NSS_CMS_SIGNER_INFO_VERSION_ISSUERSN)	    version = NSS_CMS_SIGNED_DATA_VERSION_EXT;		/* collect digestAlgorithms from SignerInfos */	/* (we need to know which algorithms we have when the content comes in) */	/* do not overwrite any existing digestAlgorithms (and digest) */	digestalgtag = NSS_CMSSignerInfo_GetDigestAlgTag(signerinfo);	n = NSS_CMSAlgArray_GetIndexByAlgTag(sigd->digestAlgorithms, digestalgtag);	if (n < 0 && haveDigests) {	    /* oops, there is a digestalg we do not have a digest for */	    /* but we were supposed to have all the digests already... */	    goto loser;	} else if (n < 0) {	    /* add the digestAlgorithm & a NULL digest */	    rv = NSS_CMSSignedData_AddDigest(poolp, sigd, digestalgtag, NULL);	    if (rv != SECSuccess)		goto loser;	} else {	    /* found it, nothing to do */	}    }    dummy = SEC_ASN1EncodeInteger(poolp, &(sigd->version), (long)version);    if (dummy == NULL)	return SECFailure;    /* this is a SET OF, so we need to sort them guys */    rv = NSS_CMSArray_SortByDER((void **)sigd->digestAlgorithms, SECOID_AlgorithmIDTemplate,				(void **)sigd->digests);    if (rv != SECSuccess)	return SECFailure;        return SECSuccess;loser:    return SECFailure;}SECStatusNSS_CMSSignedData_Encode_BeforeData(NSSCMSSignedData *sigd){    /* set up the digests */    if (sigd->digestAlgorithms != NULL) {	sigd->contentInfo.digcx = NSS_CMSDigestContext_StartMultiple(sigd->digestAlgorithms);	if (sigd->contentInfo.digcx == NULL)	    return SECFailure;    }    return SECSuccess;}/* * NSS_CMSSignedData_Encode_AfterData - do all the necessary things to a SignedData *     after all the encapsulated data was passed through the encoder. * * In detail: *  - create the signatures in all the SignerInfos * * Please note that nothing is done to the Certificates and CRLs in the message - this * is entirely the responsibility of our callers. */SECStatusNSS_CMSSignedData_Encode_AfterData(NSSCMSSignedData *sigd){    NSSCMSSignerInfo **signerinfos, *signerinfo;    NSSCMSContentInfo *cinfo;    SECOidTag digestalgtag;    SECStatus ret = SECFailure;    SECStatus rv;    SECItem *contentType;    int certcount;    int i, ci, cli, n, rci, si;    PLArenaPool *poolp;    CERTCertificateList *certlist;    extern const SEC_ASN1Template NSSCMSSignerInfoTemplate[];    poolp = sigd->cmsg->poolp;    cinfo = &(sigd->contentInfo);    /* did we have digest calculation going on? */    if (cinfo->digcx) {	rv = NSS_CMSDigestContext_FinishMultiple(cinfo->digcx, poolp, &(sigd->digests));	if (rv != SECSuccess)	    goto loser;		/* error has been set by NSS_CMSDigestContext_FinishMultiple */	cinfo->digcx = NULL;    }    signerinfos = sigd->signerInfos;    certcount = 0;    /* prepare all the SignerInfos (there may be none) */    for (i=0; i < NSS_CMSSignedData_SignerInfoCount(sigd); i++) {	signerinfo = NSS_CMSSignedData_GetSignerInfo(sigd, i);	/* find correct digest for this signerinfo */	digestalgtag = NSS_CMSSignerInfo_GetDigestAlgTag(signerinfo);	n = NSS_CMSAlgArray_GetIndexByAlgTag(sigd->digestAlgorithms, digestalgtag);	if (n < 0 || sigd->digests == NULL || sigd->digests[n] == NULL) {	    /* oops - digest not found */	    PORT_SetError(SEC_ERROR_DIGEST_NOT_FOUND);	    goto loser;	}	/* XXX if our content is anything else but data, we need to force the	 * presence of signed attributes (RFC2630 5.3 "signedAttributes is a	 * collection...") */	/* pass contentType here as we want a contentType attribute */	if ((contentType = NSS_CMSContentInfo_GetContentTypeOID(cinfo)) == NULL)	    goto loser;	/* sign the thing */	rv = NSS_CMSSignerInfo_Sign(signerinfo, sigd->digests[n], contentType);	if (rv != SECSuccess)	    goto loser;	/* while we're at it, count number of certs in certLists */	certlist = NSS_CMSSignerInfo_GetCertList(signerinfo);	if (certlist)	    certcount += certlist->len;    }    /* this is a SET OF, so we need to sort them guys */    rv = NSS_CMSArray_SortByDER((void **)signerinfos, NSSCMSSignerInfoTemplate, NULL);    if (rv != SECSuccess)	goto loser;    /*     * now prepare certs & crls     */    /* count the rest of the certs */    if (sigd->certs != NULL) {	for (ci = 0; sigd->certs[ci] != NULL; ci++)	    certcount++;    }    if (sigd->certLists != NULL) {	for (cli = 0; sigd->certLists[cli] != NULL; cli++)	    certcount += sigd->certLists[cli]->len;    }    if (certcount == 0) {	sigd->rawCerts = NULL;    } else {	/*	 * Combine all of the certs and cert chains into rawcerts.	 * Note: certcount is an upper bound; we may not need that many slots	 * but we will allocate anyway to avoid having to do another pass.	 * (The temporary space saving is not worth it.)	 *	 * XXX ARGH - this NEEDS to be fixed. need to come up with a decent	 *  SetOfDERcertficates implementation	 */	sigd->rawCerts = (SECItem **)PORT_ArenaAlloc(poolp, (certcount + 1) * sizeof(SECItem *));	if (sigd->rawCerts == NULL)	    return SECFailure;	/*	 * XXX Want to check for duplicates and not add *any* cert that is	 * already in the set.  This will be more important when we start	 * dealing with larger sets of certs, dual-key certs (signing and	 * encryption), etc.  For the time being we can slide by...	 *	 * XXX ARGH - this NEEDS to be fixed. need to come up with a decent	 *  SetOfDERcertficates implementation	 */	rci = 0;	if (signerinfos != NULL) {	    for (si = 0; signerinfos[si] != NULL; si++) {		signerinfo = signerinfos[si];		for (ci = 0; ci < signerinfo->certList->len; ci++)		    sigd->rawCerts[rci++] = &(signerinfo->certList->certs[ci]);	    }	}	if (sigd->certs != NULL) {	    for (ci = 0; sigd->certs[ci] != NULL; ci++)		sigd->rawCerts[rci++] = &(sigd->certs[ci]->derCert);	}	if (sigd->certLists != NULL) {	    for (cli = 0; sigd->certLists[cli] != NULL; cli++) {		for (ci = 0; ci < sigd->certLists[cli]->len; ci++)		    sigd->rawCerts[rci++] = &(sigd->certLists[cli]->certs[ci]);	    }	}	sigd->rawCerts[rci] = NULL;	/* this is a SET OF, so we need to sort them guys - we have the DER already, though */	NSS_CMSArray_Sort((void **)sigd->rawCerts, NSS_CMSUtil_DERCompare, NULL, NULL);    }    ret = SECSuccess;loser:    return ret;}SECStatusNSS_CMSSignedData_Decode_BeforeData(NSSCMSSignedData *sigd){    /* set up the digests */    if (sigd->digestAlgorithms != NULL && sigd->digests == NULL) {	/* if digests are already there, do nothing */	sigd->contentInfo.digcx = NSS_CMSDigestContext_StartMultiple(sigd->digestAlgorithms);	if (sigd->contentInfo.digcx == NULL)	    return SECFailure;    }    return SECSuccess;}/* * NSS_CMSSignedData_Decode_AfterData - do all the necessary things to a SignedData *     after all the encapsulated data was passed through the decoder. */SECStatusNSS_CMSSignedData_Decode_AfterData(NSSCMSSignedData *sigd){    /* did we have digest calculation going on? */    if (sigd->contentInfo.digcx) {	if (NSS_CMSDigestContext_FinishMultiple(sigd->contentInfo.digcx, sigd->cmsg->poolp, &(sigd->digests)) != SECSuccess)	    return SECFailure;	/* error has been set by NSS_CMSDigestContext_FinishMultiple */	sigd->contentInfo.digcx = NULL;    }    return SECSuccess;}/* * NSS_CMSSignedData_Decode_AfterEnd - do all the necessary things to a SignedData *     after all decoding is finished. */SECStatusNSS_CMSSignedData_Decode_AfterEnd(NSSCMSSignedData *sigd){    NSSCMSSignerInfo **signerinfos;    int i;    /* set cmsg for all the signerinfos */    signerinfos = sigd->signerInfos;    /* set cmsg for all the signerinfos */    if (signerinfos) {	for (i = 0; signerinfos[i] != NULL; i++)	    signerinfos[i]->cmsg = sigd->cmsg;    }    return SECSuccess;}/*  * NSS_CMSSignedData_GetSignerInfos - retrieve the SignedData's signer list */NSSCMSSignerInfo **NSS_CMSSignedData_GetSignerInfos(NSSCMSSignedData *sigd){    return sigd->signerInfos;}intNSS_CMSSignedData_SignerInfoCount(NSSCMSSignedData *sigd){    return NSS_CMSArray_Count((void **)sigd->signerInfos);}NSSCMSSignerInfo *NSS_CMSSignedData_GetSignerInfo(NSSCMSSignedData *sigd, int i){    return sigd->signerInfos[i];}/*  * NSS_CMSSignedData_GetDigestAlgs - retrieve the SignedData's digest algorithm list */SECAlgorithmID **NSS_CMSSignedData_GetDigestAlgs(NSSCMSSignedData *sigd){    return sigd->digestAlgorithms;}/* * NSS_CMSSignedData_GetContentInfo - return pointer to this signedData's contentinfo */NSSCMSContentInfo *NSS_CMSSignedData_GetContentInfo(NSSCMSSignedData *sigd){    return &(sigd->contentInfo);}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?