]> git.saurik.com Git - apple/security.git/blobdiff - OSX/libsecurity_smime/lib/cmsdigdata.c
Security-57336.1.9.tar.gz
[apple/security.git] / OSX / libsecurity_smime / lib / cmsdigdata.c
diff --git a/OSX/libsecurity_smime/lib/cmsdigdata.c b/OSX/libsecurity_smime/lib/cmsdigdata.c
new file mode 100644 (file)
index 0000000..8fb7d5e
--- /dev/null
@@ -0,0 +1,229 @@
+/*
+ * 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 digestedData methods.
+ */
+
+#include <Security/SecCmsDigestedData.h>
+
+#include <Security/SecCmsContentInfo.h>
+#include <Security/SecCmsDigestContext.h>
+
+#include "cmslocal.h"
+
+#include "secitem.h"
+#include "secoid.h"
+#include <security_asn1/secasn1.h>
+#include <security_asn1/secerr.h>
+
+/*
+ * SecCmsDigestedDataCreate - create a digestedData object (presumably for encoding)
+ *
+ * version will be set by SecCmsDigestedDataEncodeBeforeStart
+ * digestAlg is passed as parameter
+ * contentInfo must be filled by the user
+ * digest will be calculated while encoding
+ */
+SecCmsDigestedDataRef
+SecCmsDigestedDataCreate(SecCmsMessageRef cmsg, SECAlgorithmID *digestalg)
+{
+    void *mark;
+    SecCmsDigestedDataRef digd;
+    PLArenaPool *poolp;
+
+    poolp = cmsg->poolp;
+
+    mark = PORT_ArenaMark(poolp);
+
+    digd = (SecCmsDigestedDataRef)PORT_ArenaZAlloc(poolp, sizeof(SecCmsDigestedData));
+    if (digd == NULL)
+       goto loser;
+
+    digd->cmsg = cmsg;
+
+    if (SECOID_CopyAlgorithmID (poolp, &(digd->digestAlg), digestalg) != SECSuccess)
+       goto loser;
+
+    PORT_ArenaUnmark(poolp, mark);
+    return digd;
+
+loser:
+    PORT_ArenaRelease(poolp, mark);
+    return NULL;
+}
+
+/*
+ * SecCmsDigestedDataDestroy - destroy a digestedData object
+ */
+void
+SecCmsDigestedDataDestroy(SecCmsDigestedDataRef digd)
+{
+    /* everything's in a pool, so don't worry about the storage */
+    SecCmsContentInfoDestroy(&(digd->contentInfo));
+    return;
+}
+
+/*
+ * SecCmsDigestedDataGetContentInfo - return pointer to digestedData object's contentInfo
+ */
+SecCmsContentInfoRef
+SecCmsDigestedDataGetContentInfo(SecCmsDigestedDataRef digd)
+{
+    return &(digd->contentInfo);
+}
+
+/*
+ * SecCmsDigestedDataEncodeBeforeStart - do all the necessary things to a DigestedData
+ *     before encoding begins.
+ *
+ * In particular:
+ *  - set the right version number. The contentInfo's content type must be set up already.
+ */
+OSStatus
+SecCmsDigestedDataEncodeBeforeStart(SecCmsDigestedDataRef digd)
+{
+    unsigned long version;
+    CSSM_DATA_PTR dummy;
+
+    version = SEC_CMS_DIGESTED_DATA_VERSION_DATA;
+    if (SecCmsContentInfoGetContentTypeTag(&(digd->contentInfo)) != SEC_OID_PKCS7_DATA)
+       version = SEC_CMS_DIGESTED_DATA_VERSION_ENCAP;
+
+    dummy = SEC_ASN1EncodeInteger(digd->cmsg->poolp, &(digd->version), version);
+    return (dummy == NULL) ? SECFailure : SECSuccess;
+}
+
+/*
+ * SecCmsDigestedDataEncodeBeforeData - do all the necessary things to a DigestedData
+ *     before the encapsulated data is passed through the encoder.
+ *
+ * In detail:
+ *  - set up the digests if necessary
+ */
+OSStatus
+SecCmsDigestedDataEncodeBeforeData(SecCmsDigestedDataRef digd)
+{
+    /* set up the digests */
+    if (digd->digestAlg.algorithm.Length != 0 && digd->digest.Length == 0) {
+       /* if digest is already there, do nothing */
+       digd->contentInfo.digcx = SecCmsDigestContextStartSingle(&(digd->digestAlg));
+       if (digd->contentInfo.digcx == NULL)
+           return SECFailure;
+    }
+    return SECSuccess;
+}
+
+/*
+ * SecCmsDigestedDataEncodeAfterData - do all the necessary things to a DigestedData
+ *     after all the encapsulated data was passed through the encoder.
+ *
+ * In detail:
+ *  - finish the digests
+ */
+OSStatus
+SecCmsDigestedDataEncodeAfterData(SecCmsDigestedDataRef digd)
+{
+    OSStatus rv = SECSuccess;
+    /* did we have digest calculation going on? */
+    if (digd->contentInfo.digcx) {
+       rv = SecCmsDigestContextFinishSingle(digd->contentInfo.digcx,
+                                            (SecArenaPoolRef)digd->cmsg->poolp, &(digd->digest));
+       /* error has been set by SecCmsDigestContextFinishSingle */
+       digd->contentInfo.digcx = NULL;
+    }
+
+    return rv;
+}
+
+/*
+ * SecCmsDigestedDataDecodeBeforeData - do all the necessary things to a DigestedData
+ *     before the encapsulated data is passed through the encoder.
+ *
+ * In detail:
+ *  - set up the digests if necessary
+ */
+OSStatus
+SecCmsDigestedDataDecodeBeforeData(SecCmsDigestedDataRef digd)
+{
+    /* is there a digest algorithm yet? */
+    if (digd->digestAlg.algorithm.Length == 0)
+       return SECFailure;
+
+    digd->contentInfo.digcx = SecCmsDigestContextStartSingle(&(digd->digestAlg));
+    if (digd->contentInfo.digcx == NULL)
+       return SECFailure;
+
+    return SECSuccess;
+}
+
+/*
+ * SecCmsDigestedDataDecodeAfterData - do all the necessary things to a DigestedData
+ *     after all the encapsulated data was passed through the encoder.
+ *
+ * In detail:
+ *  - finish the digests
+ */
+OSStatus
+SecCmsDigestedDataDecodeAfterData(SecCmsDigestedDataRef digd)
+{
+    OSStatus rv = SECSuccess;
+    /* did we have digest calculation going on? */
+    if (digd->contentInfo.digcx) {
+       rv = SecCmsDigestContextFinishSingle(digd->contentInfo.digcx,
+                                            (SecArenaPoolRef)digd->cmsg->poolp, &(digd->cdigest));
+       /* error has been set by SecCmsDigestContextFinishSingle */
+       digd->contentInfo.digcx = NULL;
+    }
+
+    return rv;
+}
+
+/*
+ * SecCmsDigestedDataDecodeAfterEnd - finalize a digestedData.
+ *
+ * In detail:
+ *  - check the digests for equality
+ */
+OSStatus
+SecCmsDigestedDataDecodeAfterEnd(SecCmsDigestedDataRef digd)
+{
+    /* did we have digest calculation going on? */
+    if (digd->cdigest.Length != 0) {
+       /* XXX comparision btw digest & cdigest */
+       /* XXX set status */
+       /* TODO!!!! */
+    }
+
+    return SECSuccess;
+}