2  * The contents of this file are subject to the Mozilla Public 
   3  * License Version 1.1 (the "License"); you may not use this file 
   4  * except in compliance with the License. You may obtain a copy of 
   5  * the License at http://www.mozilla.org/MPL/ 
   7  * Software distributed under the License is distributed on an "AS 
   8  * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or 
   9  * implied. See the License for the specific language governing 
  10  * rights and limitations under the License. 
  12  * The Original Code is the Netscape security libraries. 
  14  * The Initial Developer of the Original Code is Netscape 
  15  * Communications Corporation.  Portions created by Netscape are  
  16  * Copyright (C) 1994-2000 Netscape Communications Corporation.  All 
  21  * Alternatively, the contents of this file may be used under the 
  22  * terms of the GNU General Public License Version 2 or later (the 
  23  * "GPL"), in which case the provisions of the GPL are applicable  
  24  * instead of those above.  If you wish to allow use of your  
  25  * version of this file only under the terms of the GPL and not to 
  26  * allow others to use your version of this file under the MPL, 
  27  * indicate your decision by deleting the provisions above and 
  28  * replace them with the notice and other provisions required by 
  29  * the GPL.  If you do not delete the provisions above, a recipient 
  30  * may use your version of this file under either the MPL or the 
  43 #include <security_asn1/secasn1.h> 
  44 #include <security_asn1/secerr.h> 
  47  * ------------------------------------------------------------------- 
  48  * XXX The following Attribute stuff really belongs elsewhere. 
  49  * The Attribute type is *not* part of CMS but rather X.501. 
  50  * But for now, since CMS is the only customer of attributes, 
  51  * we define them here.  Once there is a use outside of CMS, 
  52  * then change the attribute types and functions from internal 
  53  * to external naming convention, and move them elsewhere! 
  58  * SecCmsAttributeCreate - create an attribute 
  60  * if value is NULL, the attribute won't have a value. It can be added later 
  61  * with SecCmsAttributeAddValue. 
  64 SecCmsAttributeCreate(PRArenaPool 
*poolp
, SECOidTag oidtag
, CSSM_DATA_PTR value
, Boolean encoded
) 
  66     SecCmsAttribute 
*attr
; 
  67     CSSM_DATA_PTR copiedvalue
; 
  70     PORT_Assert (poolp 
!= NULL
); 
  72     mark 
= PORT_ArenaMark (poolp
); 
  74     attr 
= (SecCmsAttribute 
*)PORT_ArenaZAlloc(poolp
, sizeof(SecCmsAttribute
)); 
  78     attr
->typeTag 
= SECOID_FindOIDByTag(oidtag
); 
  79     if (attr
->typeTag 
== NULL
) 
  82     if (SECITEM_CopyItem(poolp
, &(attr
->type
), &(attr
->typeTag
->oid
)) != SECSuccess
) 
  86         if ((copiedvalue 
= SECITEM_AllocItem(poolp
, NULL
, value
->Length
)) == NULL
) 
  89         if (SECITEM_CopyItem(poolp
, copiedvalue
, value
) != SECSuccess
) 
  92         if (SecCmsArrayAdd(poolp
, (void ***)&(attr
->values
), (void *)copiedvalue
) != SECSuccess
) 
  96     attr
->encoded 
= encoded
; 
  98     PORT_ArenaUnmark (poolp
, mark
); 
 103     PORT_Assert (mark 
!= NULL
); 
 104     PORT_ArenaRelease (poolp
, mark
); 
 109  * SecCmsAttributeAddValue - add another value to an attribute 
 112 SecCmsAttributeAddValue(PLArenaPool 
*poolp
, SecCmsAttribute 
*attr
, CSSM_DATA_PTR value
) 
 114     CSSM_DATA_PTR copiedvalue
; 
 117     PORT_Assert (poolp 
!= NULL
); 
 119     mark 
= PORT_ArenaMark(poolp
); 
 122         if ((copiedvalue 
= SECITEM_AllocItem(poolp
, NULL
, value
->Length
)) == NULL
) 
 125         if (SECITEM_CopyItem(poolp
, copiedvalue
, value
) != SECSuccess
) 
 128         if (SecCmsArrayAdd(poolp
, (void ***)&(attr
->values
), (void *)copiedvalue
) != SECSuccess
) 
 132     PORT_ArenaUnmark(poolp
, mark
); 
 136     PORT_Assert (mark 
!= NULL
); 
 137     PORT_ArenaRelease (poolp
, mark
); 
 142  * SecCmsAttributeGetType - return the OID tag 
 145 SecCmsAttributeGetType(SecCmsAttribute 
*attr
) 
 149     typetag 
= SECOID_FindOID(&(attr
->type
)); 
 151         return SEC_OID_UNKNOWN
; 
 153     return typetag
->offset
; 
 157  * SecCmsAttributeGetValue - return the first attribute value 
 159  * We do some sanity checking first: 
 160  * - Multiple values are *not* expected. 
 161  * - Empty values are *not* expected. 
 164 SecCmsAttributeGetValue(SecCmsAttribute 
*attr
) 
 171     value 
= attr
->values
[0]; 
 173     if (value 
== NULL 
|| value
->Data 
== NULL 
|| value
->Length 
== 0) 
 176     if (attr
->values
[1] != NULL
) 
 183  * SecCmsAttributeCompareValue - compare the attribute's first value against data 
 186 SecCmsAttributeCompareValue(SecCmsAttribute 
*attr
, CSSM_DATA_PTR av
) 
 193     value 
= SecCmsAttributeGetValue(attr
); 
 195     return (value 
!= NULL 
&& value
->Length 
== av
->Length 
&& 
 196         PORT_Memcmp (value
->Data
, av
->Data
, value
->Length
) == 0); 
 200  * templates and functions for separate ASN.1 encoding of attributes 
 202  * used in SecCmsAttributeArrayReorder 
 206  * helper function for dynamic template determination of the attribute value 
 208 static const SecAsn1Template 
* 
 209 cms_attr_choose_attr_value_template(void *src_or_dest
, Boolean encoding
, const char *buf
, size_t len
, void *dest
) 
 211     const SecAsn1Template 
*theTemplate
; 
 212     SecCmsAttribute 
*attribute
; 
 216     PORT_Assert (src_or_dest 
!= NULL
); 
 217     if (src_or_dest 
== NULL
) 
 220     attribute 
= (SecCmsAttribute 
*)src_or_dest
; 
 222     if (encoding 
&& attribute
->encoded
) 
 223         /* we're encoding, and the attribute value is already encoded. */ 
 224         return SEC_ASN1_GET(kSecAsn1AnyTemplate
); 
 226     /* get attribute's typeTag */ 
 227     oiddata 
= attribute
->typeTag
; 
 228     if (oiddata 
== NULL
) { 
 229         oiddata 
= SECOID_FindOID(&attribute
->type
); 
 230         attribute
->typeTag 
= oiddata
; 
 233     if (oiddata 
== NULL
) { 
 234         /* still no OID tag? OID is unknown then. en/decode value as ANY. */ 
 236         theTemplate 
= SEC_ASN1_GET(kSecAsn1AnyTemplate
); 
 238         switch (oiddata
->offset
) { 
 239         case SEC_OID_PKCS9_SMIME_CAPABILITIES
: 
 240         case SEC_OID_SMIME_ENCRYPTION_KEY_PREFERENCE
: 
 241         case SEC_OID_APPLE_HASH_AGILITY_V2
: 
 242             /* these guys need to stay DER-encoded */ 
 244             /* same goes for OIDs that are not handled here */ 
 246             theTemplate 
= SEC_ASN1_GET(kSecAsn1AnyTemplate
); 
 248             /* otherwise choose proper template */ 
 249         case SEC_OID_PKCS9_EMAIL_ADDRESS
: 
 250         case SEC_OID_RFC1274_MAIL
: 
 251         case SEC_OID_PKCS9_UNSTRUCTURED_NAME
: 
 253             theTemplate 
= SEC_ASN1_GET(kSecAsn1IA5StringTemplate
); 
 255         case SEC_OID_PKCS9_CONTENT_TYPE
: 
 257             theTemplate 
= SEC_ASN1_GET(kSecAsn1ObjectIDTemplate
); 
 259         case SEC_OID_PKCS9_MESSAGE_DIGEST
: 
 260         case SEC_OID_APPLE_HASH_AGILITY
: 
 262             theTemplate 
= SEC_ASN1_GET(kSecAsn1OctetStringTemplate
); 
 264         case SEC_OID_PKCS9_SIGNING_TIME
: 
 265         case SEC_OID_APPLE_EXPIRATION_TIME
: 
 267             theTemplate 
= SEC_ASN1_GET(kSecAsn1UTCTimeTemplate
); // @@@ This should be a choice between UTCTime and GeneralizedTime -- mb 
 269           /* XXX Want other types here, too */ 
 275          * If we are encoding and we think we have an already-encoded value, 
 276          * then the code which initialized this attribute should have set 
 277          * the "encoded" property to true (and we would have returned early, 
 278          * up above).  No devastating error, but that code should be fixed. 
 279          * (It could indicate that the resulting encoded bytes are wrong.) 
 281         PORT_Assert (!encoded
); 
 284          * We are decoding; record whether the resulting value is 
 285          * still encoded or not. 
 287         attribute
->encoded 
= encoded
; 
 292 static const SecAsn1TemplateChooserPtr cms_attr_chooser
 
 293         = cms_attr_choose_attr_value_template
; 
 295 const SecAsn1Template nss_cms_attribute_template
[] = { 
 297           0, NULL
, sizeof(SecCmsAttribute
) }, 
 298     { SEC_ASN1_OBJECT_ID
, 
 299           offsetof(SecCmsAttribute
,type
) }, 
 300     { SEC_ASN1_DYNAMIC 
| SEC_ASN1_SET_OF
, 
 301           offsetof(SecCmsAttribute
,values
), 
 306 const SecAsn1Template nss_cms_set_of_attribute_template
[] = { 
 307     { SEC_ASN1_SET_OF
, 0, nss_cms_attribute_template 
} 
 310 /* ============================================================================= 
 311  * Attribute Array methods 
 315  * SecCmsAttributeArrayEncode - encode an Attribute array as SET OF Attributes 
 317  * If you are wondering why this routine does not reorder the attributes 
 318  * first, and might be tempted to make it do so, see the comment by the 
 319  * call to ReorderAttributes in cmsencode.c.  (Or, see who else calls this 
 320  * and think long and hard about the implications of making it always 
 321  * do the reordering.) 
 324 SecCmsAttributeArrayEncode(PRArenaPool 
*poolp
, SecCmsAttribute 
***attrs
, CSSM_DATA_PTR dest
) 
 326     return SEC_ASN1EncodeItem (poolp
, dest
, (void *)attrs
, nss_cms_set_of_attribute_template
); 
 330  * SecCmsAttributeArrayReorder - sort attribute array by attribute's DER encoding 
 332  * make sure that the order of the attributes guarantees valid DER (which must be 
 333  * in lexigraphically ascending order for a SET OF); if reordering is necessary it 
 334  * will be done in place (in attrs). 
 337 SecCmsAttributeArrayReorder(SecCmsAttribute 
**attrs
) 
 339     return SecCmsArraySortByDER((void **)attrs
, nss_cms_attribute_template
, NULL
); 
 343  * SecCmsAttributeArrayFindAttrByOidTag - look through a set of attributes and 
 344  * find one that matches the specified object ID. 
 346  * If "only" is true, then make sure that there is not more than one attribute 
 347  * of the same type.  Otherwise, just return the first one found. (XXX Does 
 348  * anybody really want that first-found behavior?  It was like that when I found it...) 
 351 SecCmsAttributeArrayFindAttrByOidTag(SecCmsAttribute 
**attrs
, SECOidTag oidtag
, Boolean only
) 
 354     SecCmsAttribute 
*attr1
, *attr2
; 
 359     oid 
= SECOID_FindOIDByTag(oidtag
); 
 363     while ((attr1 
= *attrs
++) != NULL
) { 
 364         if (attr1
->type
.Length 
== oid
->oid
.Length 
&& PORT_Memcmp (attr1
->type
.Data
, 
 366                                                             oid
->oid
.Length
) == 0) 
 376     while ((attr2 
= *attrs
++) != NULL
) { 
 377         if (attr2
->type
.Length 
== oid
->oid
.Length 
&& PORT_Memcmp (attr2
->type
.Data
, 
 379                                                             oid
->oid
.Length
) == 0) 
 390  * SecCmsAttributeArrayAddAttr - add an attribute to an 
 391  * array of attributes.  
 394 SecCmsAttributeArrayAddAttr(PLArenaPool 
*poolp
, SecCmsAttribute 
***attrs
, SecCmsAttribute 
*attr
) 
 396     SecCmsAttribute 
*oattr
; 
 400     mark 
= PORT_ArenaMark(poolp
); 
 402     /* find oidtag of attr */ 
 403     type 
= SecCmsAttributeGetType(attr
); 
 405     /* see if we have one already */ 
 406     oattr 
= SecCmsAttributeArrayFindAttrByOidTag(*attrs
, type
, PR_FALSE
); 
 407     PORT_Assert (oattr 
== NULL
); 
 409         goto loser
;     /* XXX or would it be better to replace it? */ 
 411     /* no, shove it in */ 
 412     if (SecCmsArrayAdd(poolp
, (void ***)attrs
, (void *)attr
) != SECSuccess
) 
 415     PORT_ArenaUnmark(poolp
, mark
); 
 419     PORT_ArenaRelease(poolp
, mark
); 
 424  * SecCmsAttributeArraySetAttr - set an attribute's value in a set of attributes 
 427 SecCmsAttributeArraySetAttr(PLArenaPool 
*poolp
, SecCmsAttribute 
***attrs
, SECOidTag type
, CSSM_DATA_PTR value
, Boolean encoded
) 
 429     SecCmsAttribute 
*attr
; 
 432     mark 
= PORT_ArenaMark(poolp
); 
 434     /* see if we have one already */ 
 435     attr 
= SecCmsAttributeArrayFindAttrByOidTag(*attrs
, type
, PR_FALSE
); 
 437         /* not found? create one! */ 
 438         attr 
= SecCmsAttributeCreate(poolp
, type
, value
, encoded
); 
 441         /* and add it to the list */ 
 442         if (SecCmsArrayAdd(poolp
, (void ***)attrs
, (void *)attr
) != SECSuccess
) 
 445         /* found, shove it in */ 
 446         /* XXX we need a decent memory model @#$#$!#!!! */ 
 447         attr
->values
[0] = value
; 
 448         attr
->encoded 
= encoded
; 
 451     PORT_ArenaUnmark (poolp
, mark
); 
 455     PORT_ArenaRelease (poolp
, mark
);