2 * Copyright 2005-2008 Juan Lang
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18 * This file implements ASN.1 DER encoding of a limited set of types.
19 * It isn't a full ASN.1 implementation. Microsoft implements BER
20 * encoding of many of the basic types in msasn1.dll, but that interface isn't
21 * implemented, so I implement them here.
24 * "A Layman's Guide to a Subset of ASN.1, BER, and DER", by Burton Kaliski
25 * (available online, look for a PDF copy as the HTML versions tend to have
26 * translation errors.)
28 * RFC3280, http://www.faqs.org/rfcs/rfc3280.html
30 * MSDN, especially "Constants for CryptEncodeObject and CryptDecodeObject"
34 #include "wine/port.h"
41 #define NONAMELESSUNION
47 #include "wine/debug.h"
48 #include "wine/exception.h"
49 #include "wine/unicode.h"
50 #include "crypt32_private.h"
52 WINE_DEFAULT_DEBUG_CHANNEL(cryptasn);
53 WINE_DECLARE_DEBUG_CHANNEL(crypt);
55 typedef BOOL (WINAPI *CryptEncodeObjectFunc)(DWORD, LPCSTR, const void *,
58 /* Prototypes for built-in encoders. They follow the Ex style prototypes.
59 * The dwCertEncodingType and lpszStructType are ignored by the built-in
60 * functions, but the parameters are retained to simplify CryptEncodeObjectEx,
61 * since it must call functions in external DLLs that follow these signatures.
63 BOOL WINAPI CRYPT_AsnEncodeOid(DWORD dwCertEncodingType,
64 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
65 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
66 static BOOL WINAPI CRYPT_AsnEncodeExtensions(DWORD dwCertEncodingType,
67 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
68 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
69 static BOOL WINAPI CRYPT_AsnEncodeSequenceOfAny(DWORD dwCertEncodingType,
70 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
71 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
72 static BOOL WINAPI CRYPT_AsnEncodeBool(DWORD dwCertEncodingType,
73 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
74 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
75 static BOOL WINAPI CRYPT_AsnEncodePubKeyInfo(DWORD dwCertEncodingType,
76 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
77 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
78 static BOOL WINAPI CRYPT_AsnEncodeBits(DWORD dwCertEncodingType,
79 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
80 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
81 static BOOL WINAPI CRYPT_AsnEncodeBitsSwapBytes(DWORD dwCertEncodingType,
82 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
83 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
84 static BOOL WINAPI CRYPT_AsnEncodeInt(DWORD dwCertEncodingType,
85 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
86 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
87 static BOOL WINAPI CRYPT_AsnEncodeInteger(DWORD dwCertEncodingType,
88 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
89 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
90 static BOOL WINAPI CRYPT_AsnEncodeUnsignedInteger(DWORD dwCertEncodingType,
91 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
92 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
93 static BOOL WINAPI CRYPT_AsnEncodeChoiceOfTime(DWORD dwCertEncodingType,
94 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
95 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
96 static BOOL WINAPI CRYPT_AsnEncodeEnhancedKeyUsage(DWORD dwCertEncodingType,
97 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
98 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
99 static BOOL WINAPI CRYPT_AsnEncodePKCSAttributes(DWORD dwCertEncodingType,
100 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
101 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
103 BOOL CRYPT_EncodeEnsureSpace(DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara,
104 BYTE *pbEncoded, DWORD *pcbEncoded, DWORD bytesNeeded)
108 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
110 if (pEncodePara && pEncodePara->pfnAlloc)
111 *(BYTE **)pbEncoded = pEncodePara->pfnAlloc(bytesNeeded);
113 *(BYTE **)pbEncoded = LocalAlloc(0, bytesNeeded);
114 if (!*(BYTE **)pbEncoded)
117 *pcbEncoded = bytesNeeded;
119 else if (bytesNeeded > *pcbEncoded)
121 *pcbEncoded = bytesNeeded;
122 SetLastError(ERROR_MORE_DATA);
126 *pcbEncoded = bytesNeeded;
130 static void CRYPT_FreeSpace(const CRYPT_ENCODE_PARA *pEncodePara, LPVOID pv)
132 if (pEncodePara && pEncodePara->pfnFree)
133 pEncodePara->pfnFree(pv);
138 BOOL CRYPT_EncodeLen(DWORD len, BYTE *pbEncoded, DWORD *pcbEncoded)
140 DWORD bytesNeeded, significantBytes = 0;
148 for (temp = len, significantBytes = sizeof(temp); !(temp & 0xff000000);
149 temp <<= 8, significantBytes--)
151 bytesNeeded = significantBytes + 1;
155 *pcbEncoded = bytesNeeded;
158 if (*pcbEncoded < bytesNeeded)
160 SetLastError(ERROR_MORE_DATA);
164 *pbEncoded = (BYTE)len;
169 *pbEncoded++ = significantBytes | 0x80;
170 for (i = 0; i < significantBytes; i++)
172 *(pbEncoded + significantBytes - i - 1) = (BYTE)(len & 0xff);
176 *pcbEncoded = bytesNeeded;
180 BOOL WINAPI CRYPT_AsnEncodeSequence(DWORD dwCertEncodingType,
181 struct AsnEncodeSequenceItem items[], DWORD cItem, DWORD dwFlags,
182 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
185 DWORD i, dataLen = 0;
187 TRACE("%p, %d, %08x, %p, %p, %d\n", items, cItem, dwFlags, pEncodePara,
188 pbEncoded, *pcbEncoded);
189 for (i = 0, ret = TRUE; ret && i < cItem; i++)
191 ret = items[i].encodeFunc(dwCertEncodingType, NULL,
192 items[i].pvStructInfo, dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL,
193 NULL, &items[i].size);
194 /* Some functions propagate their errors through the size */
196 *pcbEncoded = items[i].size;
197 dataLen += items[i].size;
201 DWORD lenBytes, bytesNeeded;
203 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
204 bytesNeeded = 1 + lenBytes + dataLen;
206 *pcbEncoded = bytesNeeded;
209 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
210 pcbEncoded, bytesNeeded)))
214 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
215 pbEncoded = *(BYTE **)pbEncoded;
217 *out++ = ASN_SEQUENCE;
218 CRYPT_EncodeLen(dataLen, out, &lenBytes);
220 for (i = 0; ret && i < cItem; i++)
222 ret = items[i].encodeFunc(dwCertEncodingType, NULL,
223 items[i].pvStructInfo, dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG,
224 NULL, out, &items[i].size);
225 /* Some functions propagate their errors through the size */
227 *pcbEncoded = items[i].size;
228 out += items[i].size;
230 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
231 CRYPT_FreeSpace(pEncodePara, pbEncoded);
235 TRACE("returning %d (%08x)\n", ret, GetLastError());
239 BOOL WINAPI CRYPT_AsnEncodeConstructed(DWORD dwCertEncodingType,
240 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
241 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
244 const struct AsnConstructedItem *item = pvStructInfo;
247 if ((ret = item->encodeFunc(dwCertEncodingType, lpszStructType,
248 item->pvStructInfo, dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL, NULL, &len)))
250 DWORD dataLen, bytesNeeded;
252 CRYPT_EncodeLen(len, NULL, &dataLen);
253 bytesNeeded = 1 + dataLen + len;
255 *pcbEncoded = bytesNeeded;
256 else if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
257 pbEncoded, pcbEncoded, bytesNeeded)))
261 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
262 pbEncoded = *(BYTE **)pbEncoded;
264 *out++ = ASN_CONTEXT | ASN_CONSTRUCTOR | item->tag;
265 CRYPT_EncodeLen(len, out, &dataLen);
267 ret = item->encodeFunc(dwCertEncodingType, lpszStructType,
268 item->pvStructInfo, dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL,
272 /* Some functions propagate their errors through the size */
274 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
275 CRYPT_FreeSpace(pEncodePara, pbEncoded);
281 /* Some functions propagate their errors through the size */
287 struct AsnEncodeTagSwappedItem
290 const void *pvStructInfo;
291 CryptEncodeObjectExFunc encodeFunc;
294 /* Sort of a wacky hack, it encodes something using the struct
295 * AsnEncodeTagSwappedItem's encodeFunc, then replaces the tag byte with the tag
296 * given in the struct AsnEncodeTagSwappedItem.
298 static BOOL WINAPI CRYPT_AsnEncodeSwapTag(DWORD dwCertEncodingType,
299 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
300 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
303 const struct AsnEncodeTagSwappedItem *item = pvStructInfo;
305 ret = item->encodeFunc(dwCertEncodingType, lpszStructType,
306 item->pvStructInfo, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
307 if (ret && pbEncoded)
308 *pbEncoded = item->tag;
312 static BOOL WINAPI CRYPT_AsnEncodeCertVersion(DWORD dwCertEncodingType,
313 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
314 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
316 const DWORD *ver = pvStructInfo;
319 /* CERT_V1 is not encoded */
327 struct AsnConstructedItem item = { 0, ver, CRYPT_AsnEncodeInt };
329 ret = CRYPT_AsnEncodeConstructed(dwCertEncodingType, X509_INTEGER,
330 &item, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
335 static BOOL WINAPI CRYPT_CopyEncodedBlob(DWORD dwCertEncodingType,
336 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
337 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
339 const CRYPT_DER_BLOB *blob = pvStructInfo;
344 *pcbEncoded = blob->cbData;
349 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
350 pcbEncoded, blob->cbData)))
352 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
353 pbEncoded = *(BYTE **)pbEncoded;
355 memcpy(pbEncoded, blob->pbData, blob->cbData);
356 *pcbEncoded = blob->cbData;
362 static BOOL WINAPI CRYPT_AsnEncodeValidity(DWORD dwCertEncodingType,
363 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
364 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
367 /* This has two filetimes in a row, a NotBefore and a NotAfter */
368 const FILETIME *timePtr = pvStructInfo;
369 struct AsnEncodeSequenceItem items[] = {
370 { timePtr, CRYPT_AsnEncodeChoiceOfTime, 0 },
371 { timePtr + 1, CRYPT_AsnEncodeChoiceOfTime, 0 },
374 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
375 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
380 /* Like CRYPT_AsnEncodeAlgorithmId, but encodes parameters as an asn.1 NULL
383 static BOOL WINAPI CRYPT_AsnEncodeAlgorithmIdWithNullParams(
384 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
385 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
388 const CRYPT_ALGORITHM_IDENTIFIER *algo = pvStructInfo;
389 static const BYTE asn1Null[] = { ASN_NULL, 0 };
390 static const CRYPT_DATA_BLOB nullBlob = { sizeof(asn1Null),
393 struct AsnEncodeSequenceItem items[2] = {
394 { algo->pszObjId, CRYPT_AsnEncodeOid, 0 },
395 { NULL, CRYPT_CopyEncodedBlob, 0 },
398 if (algo->Parameters.cbData)
399 items[1].pvStructInfo = &algo->Parameters;
401 items[1].pvStructInfo = &nullBlob;
402 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
403 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
408 static BOOL WINAPI CRYPT_AsnEncodeAlgorithmId(DWORD dwCertEncodingType,
409 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
410 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
412 const CRYPT_ALGORITHM_IDENTIFIER *algo = pvStructInfo;
414 struct AsnEncodeSequenceItem items[] = {
415 { algo->pszObjId, CRYPT_AsnEncodeOid, 0 },
416 { &algo->Parameters, CRYPT_CopyEncodedBlob, 0 },
419 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
420 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
425 static BOOL WINAPI CRYPT_AsnEncodePubKeyInfo(DWORD dwCertEncodingType,
426 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
427 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
433 const CERT_PUBLIC_KEY_INFO *info = pvStructInfo;
434 struct AsnEncodeSequenceItem items[] = {
435 { &info->Algorithm, CRYPT_AsnEncodeAlgorithmIdWithNullParams, 0 },
436 { &info->PublicKey, CRYPT_AsnEncodeBits, 0 },
439 TRACE("Encoding public key with OID %s\n",
440 debugstr_a(info->Algorithm.pszObjId));
441 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
442 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
447 SetLastError(STATUS_ACCESS_VIOLATION);
454 static BOOL WINAPI CRYPT_AsnEncodeCert(DWORD dwCertEncodingType,
455 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
456 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
462 const CERT_SIGNED_CONTENT_INFO *info = pvStructInfo;
463 struct AsnEncodeSequenceItem items[] = {
464 { &info->ToBeSigned, CRYPT_CopyEncodedBlob, 0 },
465 { &info->SignatureAlgorithm, CRYPT_AsnEncodeAlgorithmId, 0 },
466 { &info->Signature, CRYPT_AsnEncodeBitsSwapBytes, 0 },
469 if (dwFlags & CRYPT_ENCODE_NO_SIGNATURE_BYTE_REVERSAL_FLAG)
470 items[2].encodeFunc = CRYPT_AsnEncodeBits;
471 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
472 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
477 SetLastError(STATUS_ACCESS_VIOLATION);
484 BOOL WINAPI CRYPT_AsnEncodePubKeyInfoNoNull(DWORD dwCertEncodingType,
485 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
486 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
489 const CERT_PUBLIC_KEY_INFO *info = pvStructInfo;
490 struct AsnEncodeSequenceItem items[] = {
491 { &info->Algorithm, CRYPT_AsnEncodeAlgorithmId, 0 },
492 { &info->PublicKey, CRYPT_AsnEncodeBits, 0 },
495 TRACE("Encoding public key with OID %s\n",
496 debugstr_a(info->Algorithm.pszObjId));
497 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
498 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
503 /* Like in Windows, this blithely ignores the validity of the passed-in
504 * CERT_INFO, and just encodes it as-is. The resulting encoded data may not
505 * decode properly, see CRYPT_AsnDecodeCertInfo.
507 static BOOL WINAPI CRYPT_AsnEncodeCertInfo(DWORD dwCertEncodingType,
508 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
509 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
515 const CERT_INFO *info = pvStructInfo;
516 struct AsnEncodeSequenceItem items[10] = {
517 { &info->dwVersion, CRYPT_AsnEncodeCertVersion, 0 },
518 { &info->SerialNumber, CRYPT_AsnEncodeInteger, 0 },
519 { &info->SignatureAlgorithm, CRYPT_AsnEncodeAlgorithmId, 0 },
520 { &info->Issuer, CRYPT_CopyEncodedBlob, 0 },
521 { &info->NotBefore, CRYPT_AsnEncodeValidity, 0 },
522 { &info->Subject, CRYPT_CopyEncodedBlob, 0 },
523 { &info->SubjectPublicKeyInfo, CRYPT_AsnEncodePubKeyInfoNoNull, 0 },
526 struct AsnConstructedItem constructed = { 0 };
527 struct AsnEncodeTagSwappedItem swapped[2] = { { 0 } };
528 DWORD cItem = 7, cSwapped = 0;
530 if (info->IssuerUniqueId.cbData)
532 swapped[cSwapped].tag = ASN_CONTEXT | 1;
533 swapped[cSwapped].pvStructInfo = &info->IssuerUniqueId;
534 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBits;
535 items[cItem].pvStructInfo = &swapped[cSwapped];
536 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
540 if (info->SubjectUniqueId.cbData)
542 swapped[cSwapped].tag = ASN_CONTEXT | 2;
543 swapped[cSwapped].pvStructInfo = &info->SubjectUniqueId;
544 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBits;
545 items[cItem].pvStructInfo = &swapped[cSwapped];
546 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
550 if (info->cExtension)
553 constructed.pvStructInfo = &info->cExtension;
554 constructed.encodeFunc = CRYPT_AsnEncodeExtensions;
555 items[cItem].pvStructInfo = &constructed;
556 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
560 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
561 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
565 SetLastError(STATUS_ACCESS_VIOLATION);
572 static BOOL CRYPT_AsnEncodeCRLEntry(const CRL_ENTRY *entry,
573 BYTE *pbEncoded, DWORD *pcbEncoded)
575 struct AsnEncodeSequenceItem items[3] = {
576 { &entry->SerialNumber, CRYPT_AsnEncodeInteger, 0 },
577 { &entry->RevocationDate, CRYPT_AsnEncodeChoiceOfTime, 0 },
583 TRACE("%p, %p, %p\n", entry, pbEncoded, pcbEncoded);
585 if (entry->cExtension)
587 items[cItem].pvStructInfo = &entry->cExtension;
588 items[cItem].encodeFunc = CRYPT_AsnEncodeExtensions;
592 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items, cItem, 0, NULL,
593 pbEncoded, pcbEncoded);
595 TRACE("returning %d (%08x)\n", ret, GetLastError());
599 static BOOL WINAPI CRYPT_AsnEncodeCRLEntries(DWORD dwCertEncodingType,
600 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
601 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
603 DWORD bytesNeeded, dataLen, lenBytes, i;
604 const CRL_INFO *info = pvStructInfo;
605 const CRL_ENTRY *rgCRLEntry = info->rgCRLEntry;
608 for (i = 0, dataLen = 0; ret && i < info->cCRLEntry; i++)
612 ret = CRYPT_AsnEncodeCRLEntry(&rgCRLEntry[i], NULL, &size);
618 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
619 bytesNeeded = 1 + lenBytes + dataLen;
621 *pcbEncoded = bytesNeeded;
624 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
625 pcbEncoded, bytesNeeded)))
629 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
630 pbEncoded = *(BYTE **)pbEncoded;
632 *out++ = ASN_SEQUENCEOF;
633 CRYPT_EncodeLen(dataLen, out, &lenBytes);
635 for (i = 0; i < info->cCRLEntry; i++)
637 DWORD size = dataLen;
639 ret = CRYPT_AsnEncodeCRLEntry(&rgCRLEntry[i], out, &size);
643 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
644 CRYPT_FreeSpace(pEncodePara, pbEncoded);
651 static BOOL WINAPI CRYPT_AsnEncodeCRLVersion(DWORD dwCertEncodingType,
652 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
653 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
655 const DWORD *ver = pvStructInfo;
658 /* CRL_V1 is not encoded */
665 ret = CRYPT_AsnEncodeInt(dwCertEncodingType, X509_INTEGER, ver,
666 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
670 /* Like in Windows, this blithely ignores the validity of the passed-in
671 * CRL_INFO, and just encodes it as-is. The resulting encoded data may not
672 * decode properly, see CRYPT_AsnDecodeCRLInfo.
674 static BOOL WINAPI CRYPT_AsnEncodeCRLInfo(DWORD dwCertEncodingType,
675 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
676 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
682 const CRL_INFO *info = pvStructInfo;
683 struct AsnEncodeSequenceItem items[7] = {
684 { &info->dwVersion, CRYPT_AsnEncodeCRLVersion, 0 },
685 { &info->SignatureAlgorithm, CRYPT_AsnEncodeAlgorithmId, 0 },
686 { &info->Issuer, CRYPT_CopyEncodedBlob, 0 },
687 { &info->ThisUpdate, CRYPT_AsnEncodeChoiceOfTime, 0 },
690 struct AsnConstructedItem constructed[1] = { { 0 } };
691 DWORD cItem = 4, cConstructed = 0;
693 if (info->NextUpdate.dwLowDateTime || info->NextUpdate.dwHighDateTime)
695 items[cItem].pvStructInfo = &info->NextUpdate;
696 items[cItem].encodeFunc = CRYPT_AsnEncodeChoiceOfTime;
701 items[cItem].pvStructInfo = info;
702 items[cItem].encodeFunc = CRYPT_AsnEncodeCRLEntries;
705 if (info->cExtension)
707 constructed[cConstructed].tag = 0;
708 constructed[cConstructed].pvStructInfo = &info->cExtension;
709 constructed[cConstructed].encodeFunc = CRYPT_AsnEncodeExtensions;
710 items[cItem].pvStructInfo = &constructed[cConstructed];
711 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
716 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
717 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
721 SetLastError(STATUS_ACCESS_VIOLATION);
728 static BOOL CRYPT_AsnEncodeExtension(CERT_EXTENSION *ext, BYTE *pbEncoded,
732 struct AsnEncodeSequenceItem items[3] = {
733 { ext->pszObjId, CRYPT_AsnEncodeOid, 0 },
739 TRACE("%p, %p, %d\n", ext, pbEncoded, *pcbEncoded);
743 items[cItem].pvStructInfo = &ext->fCritical;
744 items[cItem].encodeFunc = CRYPT_AsnEncodeBool;
747 items[cItem].pvStructInfo = &ext->Value;
748 items[cItem].encodeFunc = CRYPT_AsnEncodeOctets;
751 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items, cItem, 0, NULL,
752 pbEncoded, pcbEncoded);
753 TRACE("returning %d (%08x)\n", ret, GetLastError());
757 static BOOL WINAPI CRYPT_AsnEncodeExtensions(DWORD dwCertEncodingType,
758 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
759 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
765 DWORD bytesNeeded, dataLen, lenBytes, i;
766 const CERT_EXTENSIONS *exts = pvStructInfo;
769 for (i = 0, dataLen = 0; ret && i < exts->cExtension; i++)
773 ret = CRYPT_AsnEncodeExtension(&exts->rgExtension[i], NULL, &size);
779 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
780 bytesNeeded = 1 + lenBytes + dataLen;
782 *pcbEncoded = bytesNeeded;
785 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
786 pbEncoded, pcbEncoded, bytesNeeded)))
790 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
791 pbEncoded = *(BYTE **)pbEncoded;
793 *out++ = ASN_SEQUENCEOF;
794 CRYPT_EncodeLen(dataLen, out, &lenBytes);
796 for (i = 0; i < exts->cExtension; i++)
798 DWORD size = dataLen;
800 ret = CRYPT_AsnEncodeExtension(&exts->rgExtension[i],
805 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
806 CRYPT_FreeSpace(pEncodePara, pbEncoded);
813 SetLastError(STATUS_ACCESS_VIOLATION);
820 BOOL WINAPI CRYPT_AsnEncodeOid(DWORD dwCertEncodingType,
821 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
822 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
824 LPCSTR pszObjId = pvStructInfo;
825 DWORD bytesNeeded = 0, lenBytes;
830 TRACE("%s\n", debugstr_a(pszObjId));
837 if (sscanf(pszObjId, "%d.%d%n", &val1, &val2, &firstPos) != 2)
839 SetLastError(CRYPT_E_ASN1_ERROR);
843 firstByte = val1 * 40 + val2;
844 ptr = pszObjId + firstPos;
854 /* note I assume each component is at most 32-bits long in base 2 */
855 if (sscanf(ptr, "%d%n", &val1, &pos) == 1)
857 if (val1 >= 0x10000000)
859 else if (val1 >= 0x200000)
861 else if (val1 >= 0x4000)
863 else if (val1 >= 0x80)
873 SetLastError(CRYPT_E_ASN1_ERROR);
877 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
881 bytesNeeded += 1 + lenBytes;
884 if (*pcbEncoded < bytesNeeded)
886 SetLastError(ERROR_MORE_DATA);
891 *pbEncoded++ = ASN_OBJECTIDENTIFIER;
892 CRYPT_EncodeLen(bytesNeeded - 1 - lenBytes, pbEncoded, &lenBytes);
893 pbEncoded += lenBytes;
899 *pbEncoded++ = firstByte;
900 ptr = pszObjId + firstPos;
903 sscanf(ptr, "%d%n", &val, &pos);
905 unsigned char outBytes[5];
908 if (val >= 0x10000000)
910 else if (val >= 0x200000)
912 else if (val >= 0x4000)
914 else if (val >= 0x80)
918 for (i = numBytes; i > 0; i--)
920 outBytes[i - 1] = val & 0x7f;
923 for (i = 0; i < numBytes - 1; i++)
924 *pbEncoded++ = outBytes[i] | 0x80;
925 *pbEncoded++ = outBytes[i];
934 *pcbEncoded = bytesNeeded;
938 static BOOL CRYPT_AsnEncodeStringCoerce(const CERT_NAME_VALUE *value,
939 BYTE tag, DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
943 LPCSTR str = (LPCSTR)value->Value.pbData;
944 DWORD bytesNeeded, lenBytes, encodedLen;
946 encodedLen = value->Value.cbData ? value->Value.cbData : strlen(str);
947 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
948 bytesNeeded = 1 + lenBytes + encodedLen;
950 *pcbEncoded = bytesNeeded;
953 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
954 pbEncoded, pcbEncoded, bytesNeeded)))
956 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
957 pbEncoded = *(BYTE **)pbEncoded;
959 CRYPT_EncodeLen(encodedLen, pbEncoded, &lenBytes);
960 pbEncoded += lenBytes;
961 memcpy(pbEncoded, str, encodedLen);
967 static BOOL CRYPT_AsnEncodeBMPString(const CERT_NAME_VALUE *value,
968 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
972 LPCWSTR str = (LPCWSTR)value->Value.pbData;
973 DWORD bytesNeeded, lenBytes, strLen;
975 if (value->Value.cbData)
976 strLen = value->Value.cbData / sizeof(WCHAR);
977 else if (value->Value.pbData)
978 strLen = lstrlenW(str);
981 CRYPT_EncodeLen(strLen * 2, NULL, &lenBytes);
982 bytesNeeded = 1 + lenBytes + strLen * 2;
984 *pcbEncoded = bytesNeeded;
987 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
988 pbEncoded, pcbEncoded, bytesNeeded)))
992 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
993 pbEncoded = *(BYTE **)pbEncoded;
994 *pbEncoded++ = ASN_BMPSTRING;
995 CRYPT_EncodeLen(strLen * 2, pbEncoded, &lenBytes);
996 pbEncoded += lenBytes;
997 for (i = 0; i < strLen; i++)
999 *pbEncoded++ = (str[i] & 0xff00) >> 8;
1000 *pbEncoded++ = str[i] & 0x00ff;
1007 static BOOL CRYPT_AsnEncodeUTF8String(const CERT_NAME_VALUE *value,
1008 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
1012 LPCWSTR str = (LPCWSTR)value->Value.pbData;
1013 DWORD bytesNeeded, lenBytes, encodedLen, strLen;
1015 if (value->Value.cbData)
1016 strLen = value->Value.cbData / sizeof(WCHAR);
1018 strLen = strlenW(str);
1021 encodedLen = WideCharToMultiByte(CP_UTF8, 0, str, strLen, NULL, 0, NULL,
1023 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
1024 bytesNeeded = 1 + lenBytes + encodedLen;
1026 *pcbEncoded = bytesNeeded;
1029 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1030 pbEncoded, pcbEncoded, bytesNeeded)))
1032 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1033 pbEncoded = *(BYTE **)pbEncoded;
1034 *pbEncoded++ = ASN_UTF8STRING;
1035 CRYPT_EncodeLen(encodedLen, pbEncoded, &lenBytes);
1036 pbEncoded += lenBytes;
1037 WideCharToMultiByte(CP_UTF8, 0, str, strLen, (LPSTR)pbEncoded,
1038 bytesNeeded - lenBytes - 1, NULL, NULL);
1044 static BOOL WINAPI CRYPT_AsnEncodeNameValue(DWORD dwCertEncodingType,
1045 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1046 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1052 const CERT_NAME_VALUE *value = pvStructInfo;
1054 switch (value->dwValueType)
1056 case CERT_RDN_ANY_TYPE:
1057 /* explicitly disallowed */
1058 SetLastError(E_INVALIDARG);
1061 case CERT_RDN_ENCODED_BLOB:
1062 ret = CRYPT_CopyEncodedBlob(dwCertEncodingType, NULL,
1063 &value->Value, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1065 case CERT_RDN_OCTET_STRING:
1066 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_OCTETSTRING,
1067 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1069 case CERT_RDN_NUMERIC_STRING:
1070 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_NUMERICSTRING,
1071 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1073 case CERT_RDN_PRINTABLE_STRING:
1074 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_PRINTABLESTRING,
1075 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1077 case CERT_RDN_TELETEX_STRING:
1078 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_T61STRING,
1079 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1081 case CERT_RDN_VIDEOTEX_STRING:
1082 ret = CRYPT_AsnEncodeStringCoerce(value,
1083 ASN_VIDEOTEXSTRING, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1085 case CERT_RDN_IA5_STRING:
1086 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_IA5STRING,
1087 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1089 case CERT_RDN_GRAPHIC_STRING:
1090 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_GRAPHICSTRING,
1091 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1093 case CERT_RDN_VISIBLE_STRING:
1094 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_VISIBLESTRING,
1095 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1097 case CERT_RDN_GENERAL_STRING:
1098 ret = CRYPT_AsnEncodeStringCoerce(value, ASN_GENERALSTRING,
1099 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1101 case CERT_RDN_UNIVERSAL_STRING:
1102 FIXME("CERT_RDN_UNIVERSAL_STRING: unimplemented\n");
1103 SetLastError(CRYPT_E_ASN1_CHOICE);
1106 case CERT_RDN_BMP_STRING:
1107 ret = CRYPT_AsnEncodeBMPString(value, dwFlags, pEncodePara,
1108 pbEncoded, pcbEncoded);
1110 case CERT_RDN_UTF8_STRING:
1111 ret = CRYPT_AsnEncodeUTF8String(value, dwFlags, pEncodePara,
1112 pbEncoded, pcbEncoded);
1115 SetLastError(CRYPT_E_ASN1_CHOICE);
1121 SetLastError(STATUS_ACCESS_VIOLATION);
1128 static BOOL CRYPT_AsnEncodeRdnAttr(DWORD dwCertEncodingType,
1129 const CERT_RDN_ATTR *attr, CryptEncodeObjectExFunc nameValueEncodeFunc,
1130 BYTE *pbEncoded, DWORD *pcbEncoded)
1132 DWORD bytesNeeded = 0, lenBytes, size;
1135 ret = CRYPT_AsnEncodeOid(dwCertEncodingType, NULL, attr->pszObjId,
1136 0, NULL, NULL, &size);
1139 bytesNeeded += size;
1140 /* hack: a CERT_RDN_ATTR is identical to a CERT_NAME_VALUE beginning
1141 * with dwValueType, so "cast" it to get its encoded size
1143 ret = nameValueEncodeFunc(dwCertEncodingType, NULL, &attr->dwValueType,
1144 0, NULL, NULL, &size);
1147 bytesNeeded += size;
1148 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
1149 bytesNeeded += 1 + lenBytes;
1152 if (*pcbEncoded < bytesNeeded)
1154 SetLastError(ERROR_MORE_DATA);
1159 *pbEncoded++ = ASN_SEQUENCE;
1160 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, pbEncoded,
1162 pbEncoded += lenBytes;
1163 size = bytesNeeded - 1 - lenBytes;
1164 ret = CRYPT_AsnEncodeOid(dwCertEncodingType, NULL,
1165 attr->pszObjId, 0, NULL, pbEncoded, &size);
1169 size = bytesNeeded - 1 - lenBytes - size;
1170 ret = nameValueEncodeFunc(dwCertEncodingType, NULL,
1171 &attr->dwValueType, 0, NULL, pbEncoded, &size);
1178 *pcbEncoded = bytesNeeded;
1182 /* Have to propagate index of failing character */
1189 static int BLOBComp(const void *l, const void *r)
1191 const CRYPT_DER_BLOB *a = l, *b = r;
1194 if (!(ret = memcmp(a->pbData, b->pbData, min(a->cbData, b->cbData))))
1195 ret = a->cbData - b->cbData;
1199 /* This encodes a SET OF, which in DER must be lexicographically sorted.
1201 static BOOL WINAPI CRYPT_DEREncodeSet(DWORD dwCertEncodingType,
1202 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1203 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1205 const CRYPT_BLOB_ARRAY *set = pvStructInfo;
1206 DWORD bytesNeeded = 0, lenBytes, i;
1209 for (i = 0; i < set->cBlob; i++)
1210 bytesNeeded += set->rgBlob[i].cbData;
1211 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
1212 bytesNeeded += 1 + lenBytes;
1215 *pcbEncoded = bytesNeeded;
1218 else if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1219 pbEncoded, pcbEncoded, bytesNeeded)))
1221 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1222 pbEncoded = *(BYTE **)pbEncoded;
1223 qsort(set->rgBlob, set->cBlob, sizeof(CRYPT_DER_BLOB), BLOBComp);
1224 *pbEncoded++ = ASN_CONSTRUCTOR | ASN_SETOF;
1225 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, pbEncoded, &lenBytes);
1226 pbEncoded += lenBytes;
1227 for (i = 0; i < set->cBlob; i++)
1229 memcpy(pbEncoded, set->rgBlob[i].pbData, set->rgBlob[i].cbData);
1230 pbEncoded += set->rgBlob[i].cbData;
1236 struct DERSetDescriptor
1242 CryptEncodeObjectExFunc encode;
1245 static BOOL WINAPI CRYPT_DEREncodeItemsAsSet(DWORD dwCertEncodingType,
1246 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1247 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1249 const struct DERSetDescriptor *desc = pvStructInfo;
1250 CRYPT_BLOB_ARRAY setOf = { 0, NULL };
1256 setOf.rgBlob = CryptMemAlloc(desc->cItems * sizeof(CRYPT_DER_BLOB));
1261 setOf.cBlob = desc->cItems;
1262 memset(setOf.rgBlob, 0, setOf.cBlob * sizeof(CRYPT_DER_BLOB));
1265 for (i = 0; ret && i < setOf.cBlob; i++)
1267 ret = desc->encode(dwCertEncodingType, lpszStructType,
1268 (const BYTE *)desc->items + i * desc->itemSize + desc->itemOffset,
1269 0, NULL, NULL, &setOf.rgBlob[i].cbData);
1272 setOf.rgBlob[i].pbData = CryptMemAlloc(setOf.rgBlob[i].cbData);
1273 if (!setOf.rgBlob[i].pbData)
1276 ret = desc->encode(dwCertEncodingType, lpszStructType,
1277 (const BYTE *)desc->items + i * desc->itemSize +
1278 desc->itemOffset, 0, NULL, setOf.rgBlob[i].pbData,
1279 &setOf.rgBlob[i].cbData);
1281 /* Some functions propagate their errors through the size */
1283 *pcbEncoded = setOf.rgBlob[i].cbData;
1287 DWORD bytesNeeded = 0, lenBytes;
1289 for (i = 0; i < setOf.cBlob; i++)
1290 bytesNeeded += setOf.rgBlob[i].cbData;
1291 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
1292 bytesNeeded += 1 + lenBytes;
1294 *pcbEncoded = bytesNeeded;
1295 else if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1296 pbEncoded, pcbEncoded, bytesNeeded)))
1298 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1299 pbEncoded = *(BYTE **)pbEncoded;
1300 qsort(setOf.rgBlob, setOf.cBlob, sizeof(CRYPT_DER_BLOB),
1302 *pbEncoded++ = ASN_CONSTRUCTOR | ASN_SETOF;
1303 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, pbEncoded, &lenBytes);
1304 pbEncoded += lenBytes;
1305 for (i = 0; i < setOf.cBlob; i++)
1307 memcpy(pbEncoded, setOf.rgBlob[i].pbData,
1308 setOf.rgBlob[i].cbData);
1309 pbEncoded += setOf.rgBlob[i].cbData;
1313 for (i = 0; i < setOf.cBlob; i++)
1314 CryptMemFree(setOf.rgBlob[i].pbData);
1315 CryptMemFree(setOf.rgBlob);
1319 static BOOL CRYPT_AsnEncodeRdn(DWORD dwCertEncodingType, const CERT_RDN *rdn,
1320 CryptEncodeObjectExFunc nameValueEncodeFunc, BYTE *pbEncoded,
1324 CRYPT_BLOB_ARRAY setOf = { 0, NULL };
1333 setOf.rgBlob = CryptMemAlloc(rdn->cRDNAttr *
1334 sizeof(CRYPT_DER_BLOB));
1339 setOf.cBlob = rdn->cRDNAttr;
1340 memset(setOf.rgBlob, 0, setOf.cBlob * sizeof(CRYPT_DER_BLOB));
1343 for (i = 0; ret && i < rdn->cRDNAttr; i++)
1345 setOf.rgBlob[i].cbData = 0;
1346 ret = CRYPT_AsnEncodeRdnAttr(dwCertEncodingType, &rdn->rgRDNAttr[i],
1347 nameValueEncodeFunc, NULL, &setOf.rgBlob[i].cbData);
1350 setOf.rgBlob[i].pbData = CryptMemAlloc(setOf.rgBlob[i].cbData);
1351 if (!setOf.rgBlob[i].pbData)
1354 ret = CRYPT_AsnEncodeRdnAttr(dwCertEncodingType,
1355 &rdn->rgRDNAttr[i], nameValueEncodeFunc,
1356 setOf.rgBlob[i].pbData, &setOf.rgBlob[i].cbData);
1360 /* Have to propagate index of failing character */
1361 *pcbEncoded = setOf.rgBlob[i].cbData;
1365 ret = CRYPT_DEREncodeSet(X509_ASN_ENCODING, NULL, &setOf, 0, NULL,
1366 pbEncoded, pcbEncoded);
1367 for (i = 0; i < setOf.cBlob; i++)
1368 CryptMemFree(setOf.rgBlob[i].pbData);
1372 SetLastError(STATUS_ACCESS_VIOLATION);
1376 CryptMemFree(setOf.rgBlob);
1380 static BOOL WINAPI CRYPT_AsnEncodeUnicodeNameValue(DWORD dwCertEncodingType,
1381 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1382 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded);
1384 static BOOL WINAPI CRYPT_AsnEncodeOrCopyUnicodeNameValue(
1385 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
1386 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
1389 const CERT_NAME_VALUE *value = pvStructInfo;
1392 if (value->dwValueType == CERT_RDN_ENCODED_BLOB)
1393 ret = CRYPT_CopyEncodedBlob(dwCertEncodingType, NULL, &value->Value,
1394 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1396 ret = CRYPT_AsnEncodeUnicodeNameValue(dwCertEncodingType, NULL, value,
1397 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1401 static BOOL WINAPI CRYPT_AsnEncodeUnicodeName(DWORD dwCertEncodingType,
1402 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1403 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1409 const CERT_NAME_INFO *info = pvStructInfo;
1410 DWORD bytesNeeded = 0, lenBytes, size, i;
1412 TRACE("encoding name with %d RDNs\n", info->cRDN);
1414 for (i = 0; ret && i < info->cRDN; i++)
1416 ret = CRYPT_AsnEncodeRdn(dwCertEncodingType, &info->rgRDN[i],
1417 CRYPT_AsnEncodeOrCopyUnicodeNameValue, NULL, &size);
1419 bytesNeeded += size;
1423 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
1424 bytesNeeded += 1 + lenBytes;
1428 *pcbEncoded = bytesNeeded;
1431 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1432 pbEncoded, pcbEncoded, bytesNeeded)))
1436 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1437 pbEncoded = *(BYTE **)pbEncoded;
1439 *out++ = ASN_SEQUENCEOF;
1440 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
1442 for (i = 0; ret && i < info->cRDN; i++)
1445 ret = CRYPT_AsnEncodeRdn(dwCertEncodingType,
1446 &info->rgRDN[i], CRYPT_AsnEncodeOrCopyUnicodeNameValue,
1451 bytesNeeded -= size;
1456 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
1457 CRYPT_FreeSpace(pEncodePara, pbEncoded);
1464 SetLastError(STATUS_ACCESS_VIOLATION);
1471 static BOOL WINAPI CRYPT_AsnEncodeCTLVersion(DWORD dwCertEncodingType,
1472 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1473 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1475 const DWORD *ver = pvStructInfo;
1478 /* CTL_V1 is not encoded */
1485 ret = CRYPT_AsnEncodeInt(dwCertEncodingType, X509_INTEGER, ver,
1486 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1490 /* Like CRYPT_AsnEncodeAlgorithmId, but encodes parameters as an asn.1 NULL
1491 * if they are empty and the OID is not empty (otherwise omits them.)
1493 static BOOL WINAPI CRYPT_AsnEncodeCTLSubjectAlgorithm(
1494 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
1495 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
1498 const CRYPT_ALGORITHM_IDENTIFIER *algo = pvStructInfo;
1500 struct AsnEncodeSequenceItem items[2] = {
1501 { algo->pszObjId, CRYPT_AsnEncodeOid, 0 },
1507 static const BYTE asn1Null[] = { ASN_NULL, 0 };
1508 static const CRYPT_DATA_BLOB nullBlob = { sizeof(asn1Null),
1511 if (algo->Parameters.cbData)
1512 items[cItem].pvStructInfo = &algo->Parameters;
1514 items[cItem].pvStructInfo = &nullBlob;
1515 items[cItem].encodeFunc = CRYPT_CopyEncodedBlob;
1518 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
1519 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1523 static BOOL CRYPT_AsnEncodeCTLEntry(const CTL_ENTRY *entry,
1524 BYTE *pbEncoded, DWORD *pcbEncoded)
1526 struct AsnEncodeSequenceItem items[2] = {
1527 { &entry->SubjectIdentifier, CRYPT_AsnEncodeOctets, 0 },
1528 { &entry->cAttribute, CRYPT_AsnEncodePKCSAttributes, 0 },
1532 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items,
1533 sizeof(items) / sizeof(items[0]), 0, NULL, pbEncoded, pcbEncoded);
1543 static BOOL WINAPI CRYPT_AsnEncodeCTLEntries(DWORD dwCertEncodingType,
1544 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1545 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1548 DWORD bytesNeeded, dataLen, lenBytes, i;
1549 const struct CTLEntries *entries = pvStructInfo;
1552 for (i = 0, dataLen = 0; ret && i < entries->cEntry; i++)
1556 ret = CRYPT_AsnEncodeCTLEntry(&entries->rgEntry[i], NULL, &size);
1562 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
1563 bytesNeeded = 1 + lenBytes + dataLen;
1565 *pcbEncoded = bytesNeeded;
1568 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1569 pbEncoded, pcbEncoded, bytesNeeded)))
1573 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1574 pbEncoded = *(BYTE **)pbEncoded;
1576 *out++ = ASN_SEQUENCEOF;
1577 CRYPT_EncodeLen(dataLen, out, &lenBytes);
1579 for (i = 0; ret && i < entries->cEntry; i++)
1581 DWORD size = dataLen;
1583 ret = CRYPT_AsnEncodeCTLEntry(&entries->rgEntry[i],
1588 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
1589 CRYPT_FreeSpace(pEncodePara, pbEncoded);
1596 static BOOL WINAPI CRYPT_AsnEncodeCTL(DWORD dwCertEncodingType,
1597 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1598 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1604 const CTL_INFO *info = pvStructInfo;
1605 struct AsnEncodeSequenceItem items[9] = {
1606 { &info->dwVersion, CRYPT_AsnEncodeCTLVersion, 0 },
1607 { &info->SubjectUsage, CRYPT_AsnEncodeEnhancedKeyUsage, 0 },
1609 struct AsnConstructedItem constructed = { 0 };
1612 if (info->ListIdentifier.cbData)
1614 items[cItem].pvStructInfo = &info->ListIdentifier;
1615 items[cItem].encodeFunc = CRYPT_AsnEncodeOctets;
1618 if (info->SequenceNumber.cbData)
1620 items[cItem].pvStructInfo = &info->SequenceNumber;
1621 items[cItem].encodeFunc = CRYPT_AsnEncodeInteger;
1624 items[cItem].pvStructInfo = &info->ThisUpdate;
1625 items[cItem].encodeFunc = CRYPT_AsnEncodeChoiceOfTime;
1627 if (info->NextUpdate.dwLowDateTime || info->NextUpdate.dwHighDateTime)
1629 items[cItem].pvStructInfo = &info->NextUpdate;
1630 items[cItem].encodeFunc = CRYPT_AsnEncodeChoiceOfTime;
1633 items[cItem].pvStructInfo = &info->SubjectAlgorithm;
1634 items[cItem].encodeFunc = CRYPT_AsnEncodeCTLSubjectAlgorithm;
1636 if (info->cCTLEntry)
1638 items[cItem].pvStructInfo = &info->cCTLEntry;
1639 items[cItem].encodeFunc = CRYPT_AsnEncodeCTLEntries;
1642 if (info->cExtension)
1644 constructed.tag = 0;
1645 constructed.pvStructInfo = &info->cExtension;
1646 constructed.encodeFunc = CRYPT_AsnEncodeExtensions;
1647 items[cItem].pvStructInfo = &constructed;
1648 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
1651 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
1652 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1656 SetLastError(STATUS_ACCESS_VIOLATION);
1662 static BOOL CRYPT_AsnEncodeSMIMECapability(DWORD dwCertEncodingType,
1663 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1664 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1670 const CRYPT_SMIME_CAPABILITY *capability = pvStructInfo;
1672 if (!capability->pszObjId)
1673 SetLastError(E_INVALIDARG);
1676 struct AsnEncodeSequenceItem items[] = {
1677 { capability->pszObjId, CRYPT_AsnEncodeOid, 0 },
1678 { &capability->Parameters, CRYPT_CopyEncodedBlob, 0 },
1681 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
1682 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
1688 SetLastError(STATUS_ACCESS_VIOLATION);
1694 static BOOL WINAPI CRYPT_AsnEncodeSMIMECapabilities(DWORD dwCertEncodingType,
1695 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1696 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1702 DWORD bytesNeeded, dataLen, lenBytes, i;
1703 const CRYPT_SMIME_CAPABILITIES *capabilities = pvStructInfo;
1706 for (i = 0, dataLen = 0; ret && i < capabilities->cCapability; i++)
1710 ret = CRYPT_AsnEncodeSMIMECapability(dwCertEncodingType, NULL,
1711 &capabilities->rgCapability[i], 0, NULL, NULL, &size);
1717 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
1718 bytesNeeded = 1 + lenBytes + dataLen;
1720 *pcbEncoded = bytesNeeded;
1723 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
1724 pbEncoded, pcbEncoded, bytesNeeded)))
1728 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1729 pbEncoded = *(BYTE **)pbEncoded;
1731 *out++ = ASN_SEQUENCEOF;
1732 CRYPT_EncodeLen(dataLen, out, &lenBytes);
1734 for (i = 0; i < capabilities->cCapability; i++)
1736 DWORD size = dataLen;
1738 ret = CRYPT_AsnEncodeSMIMECapability(dwCertEncodingType,
1739 NULL, &capabilities->rgCapability[i], 0, NULL,
1744 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
1745 CRYPT_FreeSpace(pEncodePara, pbEncoded);
1752 SetLastError(STATUS_ACCESS_VIOLATION);
1758 static BOOL WINAPI CRYPT_AsnEncodeNoticeNumbers(DWORD dwCertEncodingType,
1759 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1760 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1762 const CERT_POLICY_QUALIFIER_NOTICE_REFERENCE *noticeRef = pvStructInfo;
1763 DWORD bytesNeeded, dataLen, lenBytes, i;
1766 for (i = 0, dataLen = 0; ret && i < noticeRef->cNoticeNumbers; i++)
1770 ret = CRYPT_AsnEncodeInt(dwCertEncodingType, X509_INTEGER,
1771 ¬iceRef->rgNoticeNumbers[i], 0, NULL, NULL, &size);
1777 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
1778 bytesNeeded = 1 + lenBytes + dataLen;
1780 *pcbEncoded = bytesNeeded;
1783 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
1784 pcbEncoded, bytesNeeded)))
1788 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
1789 pbEncoded = *(BYTE **)pbEncoded;
1791 *out++ = ASN_SEQUENCE;
1792 CRYPT_EncodeLen(dataLen, out, &lenBytes);
1794 for (i = 0; i < noticeRef->cNoticeNumbers; i++)
1796 DWORD size = dataLen;
1798 ret = CRYPT_AsnEncodeInt(dwCertEncodingType, X509_INTEGER,
1799 ¬iceRef->rgNoticeNumbers[i], 0, NULL, out, &size);
1803 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
1804 CRYPT_FreeSpace(pEncodePara, pbEncoded);
1811 static BOOL WINAPI CRYPT_AsnEncodeNoticeReference(DWORD dwCertEncodingType,
1812 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1813 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1815 const CERT_POLICY_QUALIFIER_NOTICE_REFERENCE *noticeRef = pvStructInfo;
1817 CERT_NAME_VALUE orgValue = { CERT_RDN_IA5_STRING,
1818 { 0, (LPBYTE)noticeRef->pszOrganization } };
1819 struct AsnEncodeSequenceItem items[] = {
1820 { &orgValue, CRYPT_AsnEncodeNameValue, 0 },
1821 { noticeRef, CRYPT_AsnEncodeNoticeNumbers, 0 },
1824 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
1825 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
1830 static BOOL WINAPI CRYPT_AsnEncodePolicyQualifierUserNotice(
1831 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
1832 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
1839 const CERT_POLICY_QUALIFIER_USER_NOTICE *notice = pvStructInfo;
1840 struct AsnEncodeSequenceItem items[2];
1841 CERT_NAME_VALUE displayTextValue;
1845 if (notice->pNoticeReference)
1847 items[cItem].encodeFunc = CRYPT_AsnEncodeNoticeReference;
1848 items[cItem].pvStructInfo = notice->pNoticeReference;
1851 if (notice->pszDisplayText)
1853 displayTextValue.dwValueType = CERT_RDN_BMP_STRING;
1854 displayTextValue.Value.cbData = 0;
1855 displayTextValue.Value.pbData = (LPBYTE)notice->pszDisplayText;
1856 items[cItem].encodeFunc = CRYPT_AsnEncodeNameValue;
1857 items[cItem].pvStructInfo = &displayTextValue;
1860 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
1861 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1865 SetLastError(STATUS_ACCESS_VIOLATION);
1871 static BOOL WINAPI CRYPT_AsnEncodePKCSAttribute(DWORD dwCertEncodingType,
1872 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1873 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1879 const CRYPT_ATTRIBUTE *attr = pvStructInfo;
1881 if (!attr->pszObjId)
1882 SetLastError(E_INVALIDARG);
1885 struct AsnEncodeSequenceItem items[2] = {
1886 { attr->pszObjId, CRYPT_AsnEncodeOid, 0 },
1887 { &attr->cValue, CRYPT_DEREncodeSet, 0 },
1890 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
1891 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
1897 SetLastError(STATUS_ACCESS_VIOLATION);
1903 static BOOL WINAPI CRYPT_AsnEncodePKCSAttributes(DWORD dwCertEncodingType,
1904 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1905 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1911 const CRYPT_ATTRIBUTES *attributes = pvStructInfo;
1912 struct DERSetDescriptor desc = { attributes->cAttr, attributes->rgAttr,
1913 sizeof(CRYPT_ATTRIBUTE), 0, CRYPT_AsnEncodePKCSAttribute };
1915 ret = CRYPT_DEREncodeItemsAsSet(X509_ASN_ENCODING, lpszStructType,
1916 &desc, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1920 SetLastError(STATUS_ACCESS_VIOLATION);
1926 /* Like CRYPT_AsnEncodePKCSContentInfo, but allows the OID to be NULL */
1927 static BOOL WINAPI CRYPT_AsnEncodePKCSContentInfoInternal(
1928 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
1929 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
1932 const CRYPT_CONTENT_INFO *info = pvStructInfo;
1933 struct AsnEncodeSequenceItem items[2] = {
1934 { info->pszObjId, CRYPT_AsnEncodeOid, 0 },
1937 struct AsnConstructedItem constructed = { 0 };
1940 if (info->Content.cbData)
1942 constructed.tag = 0;
1943 constructed.pvStructInfo = &info->Content;
1944 constructed.encodeFunc = CRYPT_CopyEncodedBlob;
1945 items[cItem].pvStructInfo = &constructed;
1946 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
1949 return CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
1950 cItem, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
1953 BOOL CRYPT_AsnEncodePKCSDigestedData(const CRYPT_DIGESTED_DATA *digestedData,
1954 void *pvData, DWORD *pcbData)
1956 struct AsnEncodeSequenceItem items[] = {
1957 { &digestedData->version, CRYPT_AsnEncodeInt, 0 },
1958 { &digestedData->DigestAlgorithm, CRYPT_AsnEncodeAlgorithmIdWithNullParams,
1960 { &digestedData->ContentInfo, CRYPT_AsnEncodePKCSContentInfoInternal, 0 },
1961 { &digestedData->hash, CRYPT_AsnEncodeOctets, 0 },
1964 return CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items,
1965 sizeof(items) / sizeof(items[0]), 0, NULL, pvData, pcbData);
1968 static BOOL WINAPI CRYPT_AsnEncodePKCSContentInfo(DWORD dwCertEncodingType,
1969 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
1970 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
1976 const CRYPT_CONTENT_INFO *info = pvStructInfo;
1978 if (!info->pszObjId)
1979 SetLastError(E_INVALIDARG);
1981 ret = CRYPT_AsnEncodePKCSContentInfoInternal(dwCertEncodingType,
1982 lpszStructType, pvStructInfo, dwFlags, pEncodePara, pbEncoded,
1987 SetLastError(STATUS_ACCESS_VIOLATION);
1993 static BOOL CRYPT_AsnEncodeUnicodeStringCoerce(const CERT_NAME_VALUE *value,
1994 BYTE tag, DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
1998 LPCWSTR str = (LPCWSTR)value->Value.pbData;
1999 DWORD bytesNeeded, lenBytes, encodedLen;
2001 if (value->Value.cbData)
2002 encodedLen = value->Value.cbData / sizeof(WCHAR);
2004 encodedLen = strlenW(str);
2007 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
2008 bytesNeeded = 1 + lenBytes + encodedLen;
2010 *pcbEncoded = bytesNeeded;
2013 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2014 pbEncoded, pcbEncoded, bytesNeeded)))
2018 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2019 pbEncoded = *(BYTE **)pbEncoded;
2021 CRYPT_EncodeLen(encodedLen, pbEncoded, &lenBytes);
2022 pbEncoded += lenBytes;
2023 for (i = 0; i < encodedLen; i++)
2024 *pbEncoded++ = (BYTE)str[i];
2030 static BOOL CRYPT_AsnEncodeNumericString(const CERT_NAME_VALUE *value,
2031 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
2035 LPCWSTR str = (LPCWSTR)value->Value.pbData;
2036 DWORD bytesNeeded, lenBytes, encodedLen;
2038 if (value->Value.cbData)
2039 encodedLen = value->Value.cbData / sizeof(WCHAR);
2041 encodedLen = strlenW(str);
2044 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
2045 bytesNeeded = 1 + lenBytes + encodedLen;
2047 *pcbEncoded = bytesNeeded;
2050 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2051 pbEncoded, pcbEncoded, bytesNeeded)))
2056 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2057 ptr = *(BYTE **)pbEncoded;
2060 *ptr++ = ASN_NUMERICSTRING;
2061 CRYPT_EncodeLen(encodedLen, ptr, &lenBytes);
2063 for (i = 0; ret && i < encodedLen; i++)
2065 if (isdigitW(str[i]))
2066 *ptr++ = (BYTE)str[i];
2070 SetLastError(CRYPT_E_INVALID_NUMERIC_STRING);
2074 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2075 CRYPT_FreeSpace(pEncodePara, *(BYTE **)pbEncoded);
2081 static inline int isprintableW(WCHAR wc)
2083 return isalnumW(wc) || isspaceW(wc) || wc == '\'' || wc == '(' ||
2084 wc == ')' || wc == '+' || wc == ',' || wc == '-' || wc == '.' ||
2085 wc == '/' || wc == ':' || wc == '=' || wc == '?';
2088 static BOOL CRYPT_AsnEncodePrintableString(const CERT_NAME_VALUE *value,
2089 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
2093 LPCWSTR str = (LPCWSTR)value->Value.pbData;
2094 DWORD bytesNeeded, lenBytes, encodedLen;
2096 if (value->Value.cbData)
2097 encodedLen = value->Value.cbData / sizeof(WCHAR);
2099 encodedLen = strlenW(str);
2102 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
2103 bytesNeeded = 1 + lenBytes + encodedLen;
2105 *pcbEncoded = bytesNeeded;
2108 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2109 pbEncoded, pcbEncoded, bytesNeeded)))
2114 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2115 ptr = *(BYTE **)pbEncoded;
2118 *ptr++ = ASN_PRINTABLESTRING;
2119 CRYPT_EncodeLen(encodedLen, ptr, &lenBytes);
2121 for (i = 0; ret && i < encodedLen; i++)
2123 if (isprintableW(str[i]))
2124 *ptr++ = (BYTE)str[i];
2128 SetLastError(CRYPT_E_INVALID_PRINTABLE_STRING);
2132 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2133 CRYPT_FreeSpace(pEncodePara, *(BYTE **)pbEncoded);
2139 static BOOL CRYPT_AsnEncodeIA5String(const CERT_NAME_VALUE *value,
2140 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
2144 LPCWSTR str = (LPCWSTR)value->Value.pbData;
2145 DWORD bytesNeeded, lenBytes, encodedLen;
2147 if (value->Value.cbData)
2148 encodedLen = value->Value.cbData / sizeof(WCHAR);
2150 encodedLen = strlenW(str);
2153 CRYPT_EncodeLen(encodedLen, NULL, &lenBytes);
2154 bytesNeeded = 1 + lenBytes + encodedLen;
2156 *pcbEncoded = bytesNeeded;
2159 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2160 pbEncoded, pcbEncoded, bytesNeeded)))
2165 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2166 ptr = *(BYTE **)pbEncoded;
2169 *ptr++ = ASN_IA5STRING;
2170 CRYPT_EncodeLen(encodedLen, ptr, &lenBytes);
2172 for (i = 0; ret && i < encodedLen; i++)
2175 *ptr++ = (BYTE)str[i];
2179 SetLastError(CRYPT_E_INVALID_IA5_STRING);
2183 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2184 CRYPT_FreeSpace(pEncodePara, *(BYTE **)pbEncoded);
2190 static BOOL CRYPT_AsnEncodeUniversalString(const CERT_NAME_VALUE *value,
2191 DWORD dwFlags, const CRYPT_ENCODE_PARA *pEncodePara, BYTE *pbEncoded,
2195 LPCWSTR str = (LPCWSTR)value->Value.pbData;
2196 DWORD bytesNeeded, lenBytes, strLen;
2198 /* FIXME: doesn't handle composite characters */
2199 if (value->Value.cbData)
2200 strLen = value->Value.cbData / sizeof(WCHAR);
2202 strLen = strlenW(str);
2205 CRYPT_EncodeLen(strLen * 4, NULL, &lenBytes);
2206 bytesNeeded = 1 + lenBytes + strLen * 4;
2208 *pcbEncoded = bytesNeeded;
2211 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2212 pbEncoded, pcbEncoded, bytesNeeded)))
2216 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2217 pbEncoded = *(BYTE **)pbEncoded;
2218 *pbEncoded++ = ASN_UNIVERSALSTRING;
2219 CRYPT_EncodeLen(strLen * 4, pbEncoded, &lenBytes);
2220 pbEncoded += lenBytes;
2221 for (i = 0; i < strLen; i++)
2225 *pbEncoded++ = (BYTE)((str[i] & 0xff00) >> 8);
2226 *pbEncoded++ = (BYTE)(str[i] & 0x00ff);
2233 static BOOL WINAPI CRYPT_AsnEncodeUnicodeNameValue(DWORD dwCertEncodingType,
2234 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2235 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2241 const CERT_NAME_VALUE *value = pvStructInfo;
2243 switch (value->dwValueType)
2245 case CERT_RDN_ANY_TYPE:
2246 case CERT_RDN_ENCODED_BLOB:
2247 case CERT_RDN_OCTET_STRING:
2248 SetLastError(CRYPT_E_NOT_CHAR_STRING);
2250 case CERT_RDN_NUMERIC_STRING:
2251 ret = CRYPT_AsnEncodeNumericString(value, dwFlags, pEncodePara,
2252 pbEncoded, pcbEncoded);
2254 case CERT_RDN_PRINTABLE_STRING:
2255 ret = CRYPT_AsnEncodePrintableString(value, dwFlags, pEncodePara,
2256 pbEncoded, pcbEncoded);
2258 case CERT_RDN_TELETEX_STRING:
2259 ret = CRYPT_AsnEncodeUnicodeStringCoerce(value, ASN_T61STRING,
2260 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2262 case CERT_RDN_VIDEOTEX_STRING:
2263 ret = CRYPT_AsnEncodeUnicodeStringCoerce(value,
2264 ASN_VIDEOTEXSTRING, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2266 case CERT_RDN_IA5_STRING:
2267 ret = CRYPT_AsnEncodeIA5String(value, dwFlags, pEncodePara,
2268 pbEncoded, pcbEncoded);
2270 case CERT_RDN_GRAPHIC_STRING:
2271 ret = CRYPT_AsnEncodeUnicodeStringCoerce(value, ASN_GRAPHICSTRING,
2272 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2274 case CERT_RDN_VISIBLE_STRING:
2275 ret = CRYPT_AsnEncodeUnicodeStringCoerce(value, ASN_VISIBLESTRING,
2276 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2278 case CERT_RDN_GENERAL_STRING:
2279 ret = CRYPT_AsnEncodeUnicodeStringCoerce(value, ASN_GENERALSTRING,
2280 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2282 case CERT_RDN_UNIVERSAL_STRING:
2283 ret = CRYPT_AsnEncodeUniversalString(value, dwFlags, pEncodePara,
2284 pbEncoded, pcbEncoded);
2286 case CERT_RDN_BMP_STRING:
2287 ret = CRYPT_AsnEncodeBMPString(value, dwFlags, pEncodePara,
2288 pbEncoded, pcbEncoded);
2290 case CERT_RDN_UTF8_STRING:
2291 ret = CRYPT_AsnEncodeUTF8String(value, dwFlags, pEncodePara,
2292 pbEncoded, pcbEncoded);
2295 SetLastError(CRYPT_E_ASN1_CHOICE);
2300 SetLastError(STATUS_ACCESS_VIOLATION);
2306 static BOOL WINAPI CRYPT_AsnEncodeName(DWORD dwCertEncodingType,
2307 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2308 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2314 const CERT_NAME_INFO *info = pvStructInfo;
2315 DWORD bytesNeeded = 0, lenBytes, size, i;
2317 TRACE("encoding name with %d RDNs\n", info->cRDN);
2319 for (i = 0; ret && i < info->cRDN; i++)
2321 ret = CRYPT_AsnEncodeRdn(dwCertEncodingType, &info->rgRDN[i],
2322 CRYPT_AsnEncodeNameValue, NULL, &size);
2324 bytesNeeded += size;
2326 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
2327 bytesNeeded += 1 + lenBytes;
2331 *pcbEncoded = bytesNeeded;
2334 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2335 pbEncoded, pcbEncoded, bytesNeeded)))
2339 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2340 pbEncoded = *(BYTE **)pbEncoded;
2342 *out++ = ASN_SEQUENCEOF;
2343 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
2345 for (i = 0; ret && i < info->cRDN; i++)
2348 ret = CRYPT_AsnEncodeRdn(dwCertEncodingType,
2349 &info->rgRDN[i], CRYPT_AsnEncodeNameValue, out, &size);
2353 bytesNeeded -= size;
2356 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2357 CRYPT_FreeSpace(pEncodePara, pbEncoded);
2364 SetLastError(STATUS_ACCESS_VIOLATION);
2371 static BOOL WINAPI CRYPT_AsnEncodeBool(DWORD dwCertEncodingType,
2372 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2373 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2375 BOOL val = *(const BOOL *)pvStructInfo, ret;
2384 else if (*pcbEncoded < 3)
2387 SetLastError(ERROR_MORE_DATA);
2393 *pbEncoded++ = ASN_BOOL;
2395 *pbEncoded++ = val ? 0xff : 0;
2398 TRACE("returning %d (%08x)\n", ret, GetLastError());
2402 static BOOL WINAPI CRYPT_AsnEncodeAltNameEntry(DWORD dwCertEncodingType,
2403 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2404 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2406 const CERT_ALT_NAME_ENTRY *entry = pvStructInfo;
2412 switch (entry->dwAltNameChoice)
2414 case CERT_ALT_NAME_RFC822_NAME:
2415 case CERT_ALT_NAME_DNS_NAME:
2416 case CERT_ALT_NAME_URL:
2417 tag = ASN_CONTEXT | (entry->dwAltNameChoice - 1);
2418 if (entry->u.pwszURL)
2422 /* Not + 1: don't encode the NULL-terminator */
2423 dataLen = lstrlenW(entry->u.pwszURL);
2424 for (i = 0; ret && i < dataLen; i++)
2426 if (entry->u.pwszURL[i] > 0x7f)
2428 SetLastError(CRYPT_E_INVALID_IA5_STRING);
2437 case CERT_ALT_NAME_DIRECTORY_NAME:
2438 tag = ASN_CONTEXT | ASN_CONSTRUCTOR | (entry->dwAltNameChoice - 1);
2439 dataLen = entry->u.DirectoryName.cbData;
2441 case CERT_ALT_NAME_IP_ADDRESS:
2442 tag = ASN_CONTEXT | (entry->dwAltNameChoice - 1);
2443 dataLen = entry->u.IPAddress.cbData;
2445 case CERT_ALT_NAME_REGISTERED_ID:
2447 struct AsnEncodeTagSwappedItem swapped =
2448 { ASN_CONTEXT | (entry->dwAltNameChoice - 1), entry->u.pszRegisteredID,
2449 CRYPT_AsnEncodeOid };
2451 return CRYPT_AsnEncodeSwapTag(0, NULL, &swapped, 0, NULL, pbEncoded,
2454 case CERT_ALT_NAME_OTHER_NAME:
2455 FIXME("name type %d unimplemented\n", entry->dwAltNameChoice);
2458 SetLastError(E_INVALIDARG);
2463 DWORD bytesNeeded, lenBytes;
2465 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
2466 bytesNeeded = 1 + dataLen + lenBytes;
2468 *pcbEncoded = bytesNeeded;
2469 else if (*pcbEncoded < bytesNeeded)
2471 SetLastError(ERROR_MORE_DATA);
2472 *pcbEncoded = bytesNeeded;
2478 CRYPT_EncodeLen(dataLen, pbEncoded, &lenBytes);
2479 pbEncoded += lenBytes;
2480 switch (entry->dwAltNameChoice)
2482 case CERT_ALT_NAME_RFC822_NAME:
2483 case CERT_ALT_NAME_DNS_NAME:
2484 case CERT_ALT_NAME_URL:
2488 for (i = 0; i < dataLen; i++)
2489 *pbEncoded++ = (BYTE)entry->u.pwszURL[i];
2492 case CERT_ALT_NAME_DIRECTORY_NAME:
2493 memcpy(pbEncoded, entry->u.DirectoryName.pbData, dataLen);
2495 case CERT_ALT_NAME_IP_ADDRESS:
2496 memcpy(pbEncoded, entry->u.IPAddress.pbData, dataLen);
2500 *pcbEncoded = bytesNeeded;
2503 TRACE("returning %d (%08x)\n", ret, GetLastError());
2507 static BOOL WINAPI CRYPT_AsnEncodeAuthorityKeyId(DWORD dwCertEncodingType,
2508 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2509 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2515 const CERT_AUTHORITY_KEY_ID_INFO *info = pvStructInfo;
2516 struct AsnEncodeSequenceItem items[3] = { { 0 } };
2517 struct AsnEncodeTagSwappedItem swapped[3] = { { 0 } };
2518 struct AsnConstructedItem constructed = { 0 };
2519 DWORD cItem = 0, cSwapped = 0;
2521 if (info->KeyId.cbData)
2523 swapped[cSwapped].tag = ASN_CONTEXT | 0;
2524 swapped[cSwapped].pvStructInfo = &info->KeyId;
2525 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeOctets;
2526 items[cItem].pvStructInfo = &swapped[cSwapped];
2527 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
2531 if (info->CertIssuer.cbData)
2533 constructed.tag = 1;
2534 constructed.pvStructInfo = &info->CertIssuer;
2535 constructed.encodeFunc = CRYPT_CopyEncodedBlob;
2536 items[cItem].pvStructInfo = &constructed;
2537 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
2540 if (info->CertSerialNumber.cbData)
2542 swapped[cSwapped].tag = ASN_CONTEXT | 2;
2543 swapped[cSwapped].pvStructInfo = &info->CertSerialNumber;
2544 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInteger;
2545 items[cItem].pvStructInfo = &swapped[cSwapped];
2546 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
2550 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items, cItem, dwFlags,
2551 pEncodePara, pbEncoded, pcbEncoded);
2555 SetLastError(STATUS_ACCESS_VIOLATION);
2562 static BOOL WINAPI CRYPT_AsnEncodeAltName(DWORD dwCertEncodingType,
2563 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2564 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2570 const CERT_ALT_NAME_INFO *info = pvStructInfo;
2571 DWORD bytesNeeded, dataLen, lenBytes, i;
2574 /* FIXME: should check that cAltEntry is not bigger than 0xff, since we
2575 * can't encode an erroneous entry index if it's bigger than this.
2577 for (i = 0, dataLen = 0; ret && i < info->cAltEntry; i++)
2581 ret = CRYPT_AsnEncodeAltNameEntry(dwCertEncodingType, NULL,
2582 &info->rgAltEntry[i], 0, NULL, NULL, &len);
2585 else if (GetLastError() == CRYPT_E_INVALID_IA5_STRING)
2587 /* CRYPT_AsnEncodeAltNameEntry encoded the index of
2588 * the bad character, now set the index of the bad
2591 *pcbEncoded = (BYTE)i <<
2592 CERT_ALT_NAME_ENTRY_ERR_INDEX_SHIFT | len;
2597 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
2598 bytesNeeded = 1 + lenBytes + dataLen;
2601 *pcbEncoded = bytesNeeded;
2606 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2607 pbEncoded, pcbEncoded, bytesNeeded)))
2611 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2612 pbEncoded = *(BYTE **)pbEncoded;
2614 *out++ = ASN_SEQUENCEOF;
2615 CRYPT_EncodeLen(dataLen, out, &lenBytes);
2617 for (i = 0; ret && i < info->cAltEntry; i++)
2619 DWORD len = dataLen;
2621 ret = CRYPT_AsnEncodeAltNameEntry(dwCertEncodingType,
2622 NULL, &info->rgAltEntry[i], 0, NULL, out, &len);
2629 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2630 CRYPT_FreeSpace(pEncodePara, pbEncoded);
2637 SetLastError(STATUS_ACCESS_VIOLATION);
2644 static BOOL WINAPI CRYPT_AsnEncodeAuthorityKeyId2(DWORD dwCertEncodingType,
2645 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2646 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2652 const CERT_AUTHORITY_KEY_ID2_INFO *info = pvStructInfo;
2653 struct AsnEncodeSequenceItem items[3] = { { 0 } };
2654 struct AsnEncodeTagSwappedItem swapped[3] = { { 0 } };
2655 DWORD cItem = 0, cSwapped = 0;
2657 if (info->KeyId.cbData)
2659 swapped[cSwapped].tag = ASN_CONTEXT | 0;
2660 swapped[cSwapped].pvStructInfo = &info->KeyId;
2661 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeOctets;
2662 items[cItem].pvStructInfo = &swapped[cSwapped];
2663 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
2667 if (info->AuthorityCertIssuer.cAltEntry)
2669 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 1;
2670 swapped[cSwapped].pvStructInfo = &info->AuthorityCertIssuer;
2671 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeAltName;
2672 items[cItem].pvStructInfo = &swapped[cSwapped];
2673 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
2677 if (info->AuthorityCertSerialNumber.cbData)
2679 swapped[cSwapped].tag = ASN_CONTEXT | 2;
2680 swapped[cSwapped].pvStructInfo = &info->AuthorityCertSerialNumber;
2681 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInteger;
2682 items[cItem].pvStructInfo = &swapped[cSwapped];
2683 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
2687 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items, cItem, dwFlags,
2688 pEncodePara, pbEncoded, pcbEncoded);
2692 SetLastError(STATUS_ACCESS_VIOLATION);
2699 static BOOL CRYPT_AsnEncodeAccessDescription(
2700 const CERT_ACCESS_DESCRIPTION *descr, BYTE *pbEncoded, DWORD *pcbEncoded)
2702 struct AsnEncodeSequenceItem items[] = {
2703 { descr->pszAccessMethod, CRYPT_AsnEncodeOid, 0 },
2704 { &descr->AccessLocation, CRYPT_AsnEncodeAltNameEntry, 0 },
2707 if (!descr->pszAccessMethod)
2709 SetLastError(E_INVALIDARG);
2712 return CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items,
2713 sizeof(items) / sizeof(items[0]), 0, NULL, pbEncoded, pcbEncoded);
2716 static BOOL WINAPI CRYPT_AsnEncodeAuthorityInfoAccess(DWORD dwCertEncodingType,
2717 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2718 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2724 DWORD bytesNeeded, dataLen, lenBytes, i;
2725 const CERT_AUTHORITY_INFO_ACCESS *info = pvStructInfo;
2728 for (i = 0, dataLen = 0; ret && i < info->cAccDescr; i++)
2732 ret = CRYPT_AsnEncodeAccessDescription(&info->rgAccDescr[i], NULL,
2739 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
2740 bytesNeeded = 1 + lenBytes + dataLen;
2742 *pcbEncoded = bytesNeeded;
2745 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2746 pbEncoded, pcbEncoded, bytesNeeded)))
2750 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2751 pbEncoded = *(BYTE **)pbEncoded;
2753 *out++ = ASN_SEQUENCEOF;
2754 CRYPT_EncodeLen(dataLen, out, &lenBytes);
2756 for (i = 0; i < info->cAccDescr; i++)
2758 DWORD size = dataLen;
2760 ret = CRYPT_AsnEncodeAccessDescription(
2761 &info->rgAccDescr[i], out, &size);
2765 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2766 CRYPT_FreeSpace(pEncodePara, pbEncoded);
2773 SetLastError(STATUS_ACCESS_VIOLATION);
2780 static BOOL WINAPI CRYPT_AsnEncodeBasicConstraints(DWORD dwCertEncodingType,
2781 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2782 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2788 const CERT_BASIC_CONSTRAINTS_INFO *info = pvStructInfo;
2789 struct AsnEncodeSequenceItem items[3] = {
2790 { &info->SubjectType, CRYPT_AsnEncodeBits, 0 },
2795 if (info->fPathLenConstraint)
2797 items[cItem].pvStructInfo = &info->dwPathLenConstraint;
2798 items[cItem].encodeFunc = CRYPT_AsnEncodeInt;
2801 if (info->cSubtreesConstraint)
2803 items[cItem].pvStructInfo = &info->cSubtreesConstraint;
2804 items[cItem].encodeFunc = CRYPT_AsnEncodeSequenceOfAny;
2807 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
2808 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2812 SetLastError(STATUS_ACCESS_VIOLATION);
2819 static BOOL WINAPI CRYPT_AsnEncodeBasicConstraints2(DWORD dwCertEncodingType,
2820 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2821 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2827 const CERT_BASIC_CONSTRAINTS2_INFO *info = pvStructInfo;
2828 struct AsnEncodeSequenceItem items[2] = { { 0 } };
2833 items[cItem].pvStructInfo = &info->fCA;
2834 items[cItem].encodeFunc = CRYPT_AsnEncodeBool;
2837 if (info->fPathLenConstraint)
2839 items[cItem].pvStructInfo = &info->dwPathLenConstraint;
2840 items[cItem].encodeFunc = CRYPT_AsnEncodeInt;
2843 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
2844 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
2848 SetLastError(STATUS_ACCESS_VIOLATION);
2855 static BOOL WINAPI CRYPT_AsnEncodeCertPolicyQualifiers(DWORD dwCertEncodingType,
2856 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2857 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2859 const CERT_POLICY_INFO *info = pvStructInfo;
2862 if (!info->cPolicyQualifier)
2869 struct AsnEncodeSequenceItem items[2] = {
2870 { NULL, CRYPT_AsnEncodeOid, 0 },
2871 { NULL, CRYPT_CopyEncodedBlob, 0 },
2873 DWORD bytesNeeded = 0, lenBytes, size, i;
2876 for (i = 0; ret && i < info->cPolicyQualifier; i++)
2878 items[0].pvStructInfo =
2879 info->rgPolicyQualifier[i].pszPolicyQualifierId;
2880 items[1].pvStructInfo = &info->rgPolicyQualifier[i].Qualifier;
2881 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
2882 sizeof(items) / sizeof(items[0]),
2883 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL, NULL, &size);
2885 bytesNeeded += size;
2887 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
2888 bytesNeeded += 1 + lenBytes;
2892 *pcbEncoded = bytesNeeded;
2895 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2896 pbEncoded, pcbEncoded, bytesNeeded)))
2900 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2901 pbEncoded = *(BYTE **)pbEncoded;
2903 *out++ = ASN_SEQUENCEOF;
2904 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
2906 for (i = 0; ret && i < info->cPolicyQualifier; i++)
2908 items[0].pvStructInfo =
2909 info->rgPolicyQualifier[i].pszPolicyQualifierId;
2910 items[1].pvStructInfo =
2911 &info->rgPolicyQualifier[i].Qualifier;
2913 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
2914 sizeof(items) / sizeof(items[0]),
2915 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL, out, &size);
2919 bytesNeeded -= size;
2922 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
2923 CRYPT_FreeSpace(pEncodePara, pbEncoded);
2931 static BOOL CRYPT_AsnEncodeCertPolicy(DWORD dwCertEncodingType,
2932 const CERT_POLICY_INFO *info, DWORD dwFlags, BYTE *pbEncoded,
2935 struct AsnEncodeSequenceItem items[2] = {
2936 { info->pszPolicyIdentifier, CRYPT_AsnEncodeOid, 0 },
2937 { info, CRYPT_AsnEncodeCertPolicyQualifiers, 0 },
2941 if (!info->pszPolicyIdentifier)
2943 SetLastError(E_INVALIDARG);
2946 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
2947 sizeof(items) / sizeof(items[0]), dwFlags, NULL, pbEncoded, pcbEncoded);
2951 static BOOL WINAPI CRYPT_AsnEncodeCertPolicies(DWORD dwCertEncodingType,
2952 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
2953 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
2959 const CERT_POLICIES_INFO *info = pvStructInfo;
2960 DWORD bytesNeeded = 0, lenBytes, size, i;
2963 for (i = 0; ret && i < info->cPolicyInfo; i++)
2965 ret = CRYPT_AsnEncodeCertPolicy(dwCertEncodingType,
2966 &info->rgPolicyInfo[i], dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL,
2969 bytesNeeded += size;
2971 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
2972 bytesNeeded += 1 + lenBytes;
2976 *pcbEncoded = bytesNeeded;
2979 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
2980 pbEncoded, pcbEncoded, bytesNeeded)))
2984 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
2985 pbEncoded = *(BYTE **)pbEncoded;
2987 *out++ = ASN_SEQUENCEOF;
2988 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
2990 for (i = 0; ret && i < info->cPolicyInfo; i++)
2993 ret = CRYPT_AsnEncodeCertPolicy(dwCertEncodingType,
2994 &info->rgPolicyInfo[i],
2995 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, out, &size);
2999 bytesNeeded -= size;
3002 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
3003 CRYPT_FreeSpace(pEncodePara, pbEncoded);
3010 SetLastError(STATUS_ACCESS_VIOLATION);
3016 static BOOL CRYPT_AsnEncodeCertPolicyMapping(DWORD dwCertEncodingType,
3017 const CERT_POLICY_MAPPING *mapping, DWORD dwFlags, BYTE *pbEncoded,
3020 struct AsnEncodeSequenceItem items[] = {
3021 { mapping->pszIssuerDomainPolicy, CRYPT_AsnEncodeOid, 0 },
3022 { mapping->pszSubjectDomainPolicy, CRYPT_AsnEncodeOid, 0 },
3025 if (!mapping->pszIssuerDomainPolicy || !mapping->pszSubjectDomainPolicy)
3027 SetLastError(E_INVALIDARG);
3030 return CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
3031 sizeof(items) / sizeof(items[0]), dwFlags, NULL, pbEncoded, pcbEncoded);
3034 static BOOL WINAPI CRYPT_AsnEncodeCertPolicyMappings(DWORD dwCertEncodingType,
3035 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3036 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3042 const CERT_POLICY_MAPPINGS_INFO *info = pvStructInfo;
3043 DWORD bytesNeeded = 0, lenBytes, size, i;
3046 for (i = 0; ret && i < info->cPolicyMapping; i++)
3048 ret = CRYPT_AsnEncodeCertPolicyMapping(dwCertEncodingType,
3049 &info->rgPolicyMapping[i], dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG,
3052 bytesNeeded += size;
3054 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
3055 bytesNeeded += 1 + lenBytes;
3059 *pcbEncoded = bytesNeeded;
3062 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
3063 pbEncoded, pcbEncoded, bytesNeeded)))
3067 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3068 pbEncoded = *(BYTE **)pbEncoded;
3070 *out++ = ASN_SEQUENCEOF;
3071 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
3073 for (i = 0; ret && i < info->cPolicyMapping; i++)
3076 ret = CRYPT_AsnEncodeCertPolicyMapping(
3077 dwCertEncodingType, &info->rgPolicyMapping[i],
3078 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, out, &size);
3082 bytesNeeded -= size;
3085 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
3086 CRYPT_FreeSpace(pEncodePara, pbEncoded);
3093 SetLastError(STATUS_ACCESS_VIOLATION);
3099 static BOOL WINAPI CRYPT_AsnEncodeCertPolicyConstraints(
3100 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
3101 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
3108 const CERT_POLICY_CONSTRAINTS_INFO *info = pvStructInfo;
3109 struct AsnEncodeSequenceItem items[2];
3110 struct AsnEncodeTagSwappedItem swapped[2];
3111 DWORD cItem = 0, cSwapped = 0;
3113 if (info->fRequireExplicitPolicy)
3115 swapped[cSwapped].tag = ASN_CONTEXT | 0;
3116 swapped[cSwapped].pvStructInfo =
3117 &info->dwRequireExplicitPolicySkipCerts;
3118 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInt;
3119 items[cItem].pvStructInfo = &swapped[cSwapped];
3120 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3124 if (info->fInhibitPolicyMapping)
3126 swapped[cSwapped].tag = ASN_CONTEXT | 1;
3127 swapped[cSwapped].pvStructInfo =
3128 &info->dwInhibitPolicyMappingSkipCerts;
3129 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInt;
3130 items[cItem].pvStructInfo = &swapped[cSwapped];
3131 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3135 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
3136 dwFlags, NULL, pbEncoded, pcbEncoded);
3140 SetLastError(STATUS_ACCESS_VIOLATION);
3146 static BOOL WINAPI CRYPT_AsnEncodeRsaPubKey(DWORD dwCertEncodingType,
3147 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3148 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3154 const BLOBHEADER *hdr = pvStructInfo;
3156 if (hdr->bType != PUBLICKEYBLOB)
3158 SetLastError(E_INVALIDARG);
3163 const RSAPUBKEY *rsaPubKey = (const RSAPUBKEY *)
3164 ((const BYTE *)pvStructInfo + sizeof(BLOBHEADER));
3165 CRYPT_INTEGER_BLOB blob = { rsaPubKey->bitlen / 8,
3166 (BYTE *)pvStructInfo + sizeof(BLOBHEADER) + sizeof(RSAPUBKEY) };
3167 struct AsnEncodeSequenceItem items[] = {
3168 { &blob, CRYPT_AsnEncodeUnsignedInteger, 0 },
3169 { &rsaPubKey->pubexp, CRYPT_AsnEncodeInt, 0 },
3172 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
3173 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
3179 SetLastError(STATUS_ACCESS_VIOLATION);
3186 BOOL WINAPI CRYPT_AsnEncodeOctets(DWORD dwCertEncodingType,
3187 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3188 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3194 const CRYPT_DATA_BLOB *blob = pvStructInfo;
3195 DWORD bytesNeeded, lenBytes;
3197 TRACE("(%d, %p), %08x, %p, %p, %d\n", blob->cbData, blob->pbData,
3198 dwFlags, pEncodePara, pbEncoded, *pcbEncoded);
3200 CRYPT_EncodeLen(blob->cbData, NULL, &lenBytes);
3201 bytesNeeded = 1 + lenBytes + blob->cbData;
3204 *pcbEncoded = bytesNeeded;
3209 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3210 pcbEncoded, bytesNeeded)))
3212 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3213 pbEncoded = *(BYTE **)pbEncoded;
3214 *pbEncoded++ = ASN_OCTETSTRING;
3215 CRYPT_EncodeLen(blob->cbData, pbEncoded, &lenBytes);
3216 pbEncoded += lenBytes;
3218 memcpy(pbEncoded, blob->pbData, blob->cbData);
3224 SetLastError(STATUS_ACCESS_VIOLATION);
3228 TRACE("returning %d (%08x)\n", ret, GetLastError());
3232 static BOOL WINAPI CRYPT_AsnEncodeBits(DWORD dwCertEncodingType,
3233 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3234 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3240 const CRYPT_BIT_BLOB *blob = pvStructInfo;
3241 DWORD bytesNeeded, lenBytes, dataBytes;
3244 /* yep, MS allows cUnusedBits to be >= 8 */
3245 if (!blob->cUnusedBits)
3247 dataBytes = blob->cbData;
3250 else if (blob->cbData * 8 > blob->cUnusedBits)
3252 dataBytes = (blob->cbData * 8 - blob->cUnusedBits) / 8 + 1;
3253 unusedBits = blob->cUnusedBits >= 8 ? blob->cUnusedBits / 8 :
3261 CRYPT_EncodeLen(dataBytes + 1, NULL, &lenBytes);
3262 bytesNeeded = 1 + lenBytes + dataBytes + 1;
3265 *pcbEncoded = bytesNeeded;
3270 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3271 pcbEncoded, bytesNeeded)))
3273 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3274 pbEncoded = *(BYTE **)pbEncoded;
3275 *pbEncoded++ = ASN_BITSTRING;
3276 CRYPT_EncodeLen(dataBytes + 1, pbEncoded, &lenBytes);
3277 pbEncoded += lenBytes;
3278 *pbEncoded++ = unusedBits;
3281 BYTE mask = 0xff << unusedBits;
3285 memcpy(pbEncoded, blob->pbData, dataBytes - 1);
3286 pbEncoded += dataBytes - 1;
3288 *pbEncoded = *(blob->pbData + dataBytes - 1) & mask;
3295 SetLastError(STATUS_ACCESS_VIOLATION);
3302 static BOOL WINAPI CRYPT_AsnEncodeBitsSwapBytes(DWORD dwCertEncodingType,
3303 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3304 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3310 const CRYPT_BIT_BLOB *blob = pvStructInfo;
3311 CRYPT_BIT_BLOB newBlob = { blob->cbData, NULL, blob->cUnusedBits };
3316 newBlob.pbData = CryptMemAlloc(newBlob.cbData);
3321 for (i = 0; i < newBlob.cbData; i++)
3322 newBlob.pbData[newBlob.cbData - i - 1] = blob->pbData[i];
3328 ret = CRYPT_AsnEncodeBits(dwCertEncodingType, lpszStructType,
3329 &newBlob, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
3330 CryptMemFree(newBlob.pbData);
3334 SetLastError(STATUS_ACCESS_VIOLATION);
3341 static BOOL WINAPI CRYPT_AsnEncodeInt(DWORD dwCertEncodingType,
3342 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3343 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3345 CRYPT_INTEGER_BLOB blob = { sizeof(INT), (BYTE *)pvStructInfo };
3347 return CRYPT_AsnEncodeInteger(dwCertEncodingType, X509_MULTI_BYTE_INTEGER,
3348 &blob, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
3351 static BOOL WINAPI CRYPT_AsnEncodeInteger(DWORD dwCertEncodingType,
3352 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3353 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3359 DWORD significantBytes, lenBytes, bytesNeeded;
3362 const CRYPT_INTEGER_BLOB *blob = pvStructInfo;
3364 significantBytes = blob->cbData;
3365 if (significantBytes)
3367 if (blob->pbData[significantBytes - 1] & 0x80)
3369 /* negative, lop off leading (little-endian) 0xffs */
3370 for (; significantBytes > 0 &&
3371 blob->pbData[significantBytes - 1] == 0xff; significantBytes--)
3373 if (blob->pbData[significantBytes - 1] < 0x80)
3381 /* positive, lop off leading (little-endian) zeroes */
3382 for (; significantBytes > 0 &&
3383 !blob->pbData[significantBytes - 1]; significantBytes--)
3385 if (significantBytes == 0)
3386 significantBytes = 1;
3387 if (blob->pbData[significantBytes - 1] > 0x7f)
3395 CRYPT_EncodeLen(significantBytes + 1, NULL, &lenBytes);
3397 CRYPT_EncodeLen(significantBytes, NULL, &lenBytes);
3398 bytesNeeded = 1 + lenBytes + significantBytes;
3403 *pcbEncoded = bytesNeeded;
3408 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3409 pcbEncoded, bytesNeeded)))
3411 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3412 pbEncoded = *(BYTE **)pbEncoded;
3413 *pbEncoded++ = ASN_INTEGER;
3416 CRYPT_EncodeLen(significantBytes + 1, pbEncoded, &lenBytes);
3417 pbEncoded += lenBytes;
3418 *pbEncoded++ = padByte;
3422 CRYPT_EncodeLen(significantBytes, pbEncoded, &lenBytes);
3423 pbEncoded += lenBytes;
3425 for (; significantBytes > 0; significantBytes--)
3426 *(pbEncoded++) = blob->pbData[significantBytes - 1];
3432 SetLastError(STATUS_ACCESS_VIOLATION);
3439 static BOOL WINAPI CRYPT_AsnEncodeUnsignedInteger(DWORD dwCertEncodingType,
3440 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3441 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3447 DWORD significantBytes, lenBytes, bytesNeeded;
3449 const CRYPT_INTEGER_BLOB *blob = pvStructInfo;
3451 significantBytes = blob->cbData;
3452 if (significantBytes)
3454 /* positive, lop off leading (little-endian) zeroes */
3455 for (; significantBytes > 0 && !blob->pbData[significantBytes - 1];
3458 if (significantBytes == 0)
3459 significantBytes = 1;
3460 if (blob->pbData[significantBytes - 1] > 0x7f)
3464 CRYPT_EncodeLen(significantBytes + 1, NULL, &lenBytes);
3466 CRYPT_EncodeLen(significantBytes, NULL, &lenBytes);
3467 bytesNeeded = 1 + lenBytes + significantBytes;
3472 *pcbEncoded = bytesNeeded;
3477 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3478 pcbEncoded, bytesNeeded)))
3480 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3481 pbEncoded = *(BYTE **)pbEncoded;
3482 *pbEncoded++ = ASN_INTEGER;
3485 CRYPT_EncodeLen(significantBytes + 1, pbEncoded, &lenBytes);
3486 pbEncoded += lenBytes;
3491 CRYPT_EncodeLen(significantBytes, pbEncoded, &lenBytes);
3492 pbEncoded += lenBytes;
3494 for (; significantBytes > 0; significantBytes--)
3495 *(pbEncoded++) = blob->pbData[significantBytes - 1];
3501 SetLastError(STATUS_ACCESS_VIOLATION);
3508 static BOOL WINAPI CRYPT_AsnEncodeEnumerated(DWORD dwCertEncodingType,
3509 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3510 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3512 CRYPT_INTEGER_BLOB blob;
3515 /* Encode as an unsigned integer, then change the tag to enumerated */
3516 blob.cbData = sizeof(DWORD);
3517 blob.pbData = (BYTE *)pvStructInfo;
3518 ret = CRYPT_AsnEncodeUnsignedInteger(dwCertEncodingType,
3519 X509_MULTI_BYTE_UINT, &blob, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
3520 if (ret && pbEncoded)
3522 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3523 pbEncoded = *(BYTE **)pbEncoded;
3524 pbEncoded[0] = ASN_ENUMERATED;
3529 static BOOL WINAPI CRYPT_AsnEncodeUtcTime(DWORD dwCertEncodingType,
3530 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3531 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3538 /* sorry, magic number: enough for tag, len, YYMMDDHHMMSSZ\0. I use a
3539 * temporary buffer because the output buffer is not NULL-terminated.
3542 static const DWORD bytesNeeded = sizeof(buf) - 1;
3546 *pcbEncoded = bytesNeeded;
3551 /* Sanity check the year, this is a two-digit year format */
3552 ret = FileTimeToSystemTime(pvStructInfo, &sysTime);
3553 if (ret && (sysTime.wYear < 1950 || sysTime.wYear > 2050))
3555 SetLastError(CRYPT_E_BAD_ENCODE);
3560 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
3561 pbEncoded, pcbEncoded, bytesNeeded)))
3563 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3564 pbEncoded = *(BYTE **)pbEncoded;
3565 buf[0] = ASN_UTCTIME;
3566 buf[1] = bytesNeeded - 2;
3567 snprintf(buf + 2, sizeof(buf) - 2,
3568 "%02d%02d%02d%02d%02d%02dZ", sysTime.wYear >= 2000 ?
3569 sysTime.wYear - 2000 : sysTime.wYear - 1900,
3570 sysTime.wMonth, sysTime.wDay, sysTime.wHour,
3571 sysTime.wMinute, sysTime.wSecond);
3572 memcpy(pbEncoded, buf, bytesNeeded);
3579 SetLastError(STATUS_ACCESS_VIOLATION);
3586 static BOOL CRYPT_AsnEncodeGeneralizedTime(DWORD dwCertEncodingType,
3587 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3588 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3595 /* sorry, magic number: enough for tag, len, YYYYMMDDHHMMSSZ\0. I use a
3596 * temporary buffer because the output buffer is not NULL-terminated.
3599 static const DWORD bytesNeeded = sizeof(buf) - 1;
3603 *pcbEncoded = bytesNeeded;
3608 ret = FileTimeToSystemTime(pvStructInfo, &sysTime);
3610 ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3611 pcbEncoded, bytesNeeded);
3614 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3615 pbEncoded = *(BYTE **)pbEncoded;
3616 buf[0] = ASN_GENERALTIME;
3617 buf[1] = bytesNeeded - 2;
3618 snprintf(buf + 2, sizeof(buf) - 2, "%04d%02d%02d%02d%02d%02dZ",
3619 sysTime.wYear, sysTime.wMonth, sysTime.wDay, sysTime.wHour,
3620 sysTime.wMinute, sysTime.wSecond);
3621 memcpy(pbEncoded, buf, bytesNeeded);
3627 SetLastError(STATUS_ACCESS_VIOLATION);
3634 static BOOL WINAPI CRYPT_AsnEncodeChoiceOfTime(DWORD dwCertEncodingType,
3635 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3636 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3644 /* Check the year, if it's in the UTCTime range call that encode func */
3645 if (!FileTimeToSystemTime(pvStructInfo, &sysTime))
3647 if (sysTime.wYear >= 1950 && sysTime.wYear <= 2050)
3648 ret = CRYPT_AsnEncodeUtcTime(dwCertEncodingType, lpszStructType,
3649 pvStructInfo, dwFlags, pEncodePara, pbEncoded, pcbEncoded);
3651 ret = CRYPT_AsnEncodeGeneralizedTime(dwCertEncodingType,
3652 lpszStructType, pvStructInfo, dwFlags, pEncodePara, pbEncoded,
3657 SetLastError(STATUS_ACCESS_VIOLATION);
3664 static BOOL WINAPI CRYPT_AsnEncodeSequenceOfAny(DWORD dwCertEncodingType,
3665 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3666 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3672 DWORD bytesNeeded, dataLen, lenBytes, i;
3673 const CRYPT_SEQUENCE_OF_ANY *seq = pvStructInfo;
3675 for (i = 0, dataLen = 0; i < seq->cValue; i++)
3676 dataLen += seq->rgValue[i].cbData;
3677 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
3678 bytesNeeded = 1 + lenBytes + dataLen;
3681 *pcbEncoded = bytesNeeded;
3686 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara, pbEncoded,
3687 pcbEncoded, bytesNeeded)))
3689 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3690 pbEncoded = *(BYTE **)pbEncoded;
3691 *pbEncoded++ = ASN_SEQUENCEOF;
3692 CRYPT_EncodeLen(dataLen, pbEncoded, &lenBytes);
3693 pbEncoded += lenBytes;
3694 for (i = 0; i < seq->cValue; i++)
3696 memcpy(pbEncoded, seq->rgValue[i].pbData,
3697 seq->rgValue[i].cbData);
3698 pbEncoded += seq->rgValue[i].cbData;
3705 SetLastError(STATUS_ACCESS_VIOLATION);
3712 static BOOL CRYPT_AsnEncodeDistPoint(const CRL_DIST_POINT *distPoint,
3713 BYTE *pbEncoded, DWORD *pcbEncoded)
3716 struct AsnEncodeSequenceItem items[3] = { { 0 } };
3717 struct AsnConstructedItem constructed = { 0 };
3718 struct AsnEncodeTagSwappedItem swapped[3] = { { 0 } };
3719 DWORD cItem = 0, cSwapped = 0;
3721 switch (distPoint->DistPointName.dwDistPointNameChoice)
3723 case CRL_DIST_POINT_NO_NAME:
3726 case CRL_DIST_POINT_FULL_NAME:
3727 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 0;
3728 swapped[cSwapped].pvStructInfo = &distPoint->DistPointName.u.FullName;
3729 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeAltName;
3730 constructed.tag = 0;
3731 constructed.pvStructInfo = &swapped[cSwapped];
3732 constructed.encodeFunc = CRYPT_AsnEncodeSwapTag;
3733 items[cItem].pvStructInfo = &constructed;
3734 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
3738 case CRL_DIST_POINT_ISSUER_RDN_NAME:
3739 FIXME("unimplemented for CRL_DIST_POINT_ISSUER_RDN_NAME\n");
3745 if (ret && distPoint->ReasonFlags.cbData)
3747 swapped[cSwapped].tag = ASN_CONTEXT | 1;
3748 swapped[cSwapped].pvStructInfo = &distPoint->ReasonFlags;
3749 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBits;
3750 items[cItem].pvStructInfo = &swapped[cSwapped];
3751 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3755 if (ret && distPoint->CRLIssuer.cAltEntry)
3757 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 2;
3758 swapped[cSwapped].pvStructInfo = &distPoint->CRLIssuer;
3759 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeAltName;
3760 items[cItem].pvStructInfo = &swapped[cSwapped];
3761 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3766 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items, cItem, 0, NULL,
3767 pbEncoded, pcbEncoded);
3771 static BOOL WINAPI CRYPT_AsnEncodeCRLDistPoints(DWORD dwCertEncodingType,
3772 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3773 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3779 const CRL_DIST_POINTS_INFO *info = pvStructInfo;
3781 if (!info->cDistPoint)
3783 SetLastError(E_INVALIDARG);
3788 DWORD bytesNeeded, dataLen, lenBytes, i;
3791 for (i = 0, dataLen = 0; ret && i < info->cDistPoint; i++)
3795 ret = CRYPT_AsnEncodeDistPoint(&info->rgDistPoint[i], NULL,
3799 else if (GetLastError() == CRYPT_E_INVALID_IA5_STRING)
3801 /* Have to propagate index of failing character */
3807 CRYPT_EncodeLen(dataLen, NULL, &lenBytes);
3808 bytesNeeded = 1 + lenBytes + dataLen;
3811 *pcbEncoded = bytesNeeded;
3816 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
3817 pbEncoded, pcbEncoded, bytesNeeded)))
3821 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3822 pbEncoded = *(BYTE **)pbEncoded;
3824 *out++ = ASN_SEQUENCEOF;
3825 CRYPT_EncodeLen(dataLen, out, &lenBytes);
3827 for (i = 0; ret && i < info->cDistPoint; i++)
3829 DWORD len = dataLen;
3831 ret = CRYPT_AsnEncodeDistPoint(
3832 &info->rgDistPoint[i], out, &len);
3839 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
3840 CRYPT_FreeSpace(pEncodePara, pbEncoded);
3848 SetLastError(STATUS_ACCESS_VIOLATION);
3855 static BOOL WINAPI CRYPT_AsnEncodeEnhancedKeyUsage(DWORD dwCertEncodingType,
3856 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3857 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3863 const CERT_ENHKEY_USAGE *usage = pvStructInfo;
3864 DWORD bytesNeeded = 0, lenBytes, size, i;
3867 for (i = 0; ret && i < usage->cUsageIdentifier; i++)
3869 ret = CRYPT_AsnEncodeOid(dwCertEncodingType, NULL,
3870 usage->rgpszUsageIdentifier[i],
3871 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL, NULL, &size);
3873 bytesNeeded += size;
3875 CRYPT_EncodeLen(bytesNeeded, NULL, &lenBytes);
3876 bytesNeeded += 1 + lenBytes;
3880 *pcbEncoded = bytesNeeded;
3883 if ((ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
3884 pbEncoded, pcbEncoded, bytesNeeded)))
3888 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
3889 pbEncoded = *(BYTE **)pbEncoded;
3891 *out++ = ASN_SEQUENCEOF;
3892 CRYPT_EncodeLen(bytesNeeded - lenBytes - 1, out, &lenBytes);
3894 for (i = 0; ret && i < usage->cUsageIdentifier; i++)
3897 ret = CRYPT_AsnEncodeOid(dwCertEncodingType, NULL,
3898 usage->rgpszUsageIdentifier[i],
3899 dwFlags & ~CRYPT_ENCODE_ALLOC_FLAG, NULL, out, &size);
3903 bytesNeeded -= size;
3906 if (!ret && (dwFlags & CRYPT_ENCODE_ALLOC_FLAG))
3907 CRYPT_FreeSpace(pEncodePara, pbEncoded);
3914 SetLastError(STATUS_ACCESS_VIOLATION);
3921 static BOOL WINAPI CRYPT_AsnEncodeIssuingDistPoint(DWORD dwCertEncodingType,
3922 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
3923 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
3929 const CRL_ISSUING_DIST_POINT *point = pvStructInfo;
3930 struct AsnEncodeSequenceItem items[6] = { { 0 } };
3931 struct AsnConstructedItem constructed = { 0 };
3932 struct AsnEncodeTagSwappedItem swapped[5] = { { 0 } };
3933 DWORD cItem = 0, cSwapped = 0;
3936 switch (point->DistPointName.dwDistPointNameChoice)
3938 case CRL_DIST_POINT_NO_NAME:
3941 case CRL_DIST_POINT_FULL_NAME:
3942 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 0;
3943 swapped[cSwapped].pvStructInfo = &point->DistPointName.u.FullName;
3944 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeAltName;
3945 constructed.tag = 0;
3946 constructed.pvStructInfo = &swapped[cSwapped];
3947 constructed.encodeFunc = CRYPT_AsnEncodeSwapTag;
3948 items[cItem].pvStructInfo = &constructed;
3949 items[cItem].encodeFunc = CRYPT_AsnEncodeConstructed;
3954 SetLastError(E_INVALIDARG);
3957 if (ret && point->fOnlyContainsUserCerts)
3959 swapped[cSwapped].tag = ASN_CONTEXT | 1;
3960 swapped[cSwapped].pvStructInfo = &point->fOnlyContainsUserCerts;
3961 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBool;
3962 items[cItem].pvStructInfo = &swapped[cSwapped];
3963 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3967 if (ret && point->fOnlyContainsCACerts)
3969 swapped[cSwapped].tag = ASN_CONTEXT | 2;
3970 swapped[cSwapped].pvStructInfo = &point->fOnlyContainsCACerts;
3971 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBool;
3972 items[cItem].pvStructInfo = &swapped[cSwapped];
3973 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3977 if (ret && point->OnlySomeReasonFlags.cbData)
3979 swapped[cSwapped].tag = ASN_CONTEXT | 3;
3980 swapped[cSwapped].pvStructInfo = &point->OnlySomeReasonFlags;
3981 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBits;
3982 items[cItem].pvStructInfo = &swapped[cSwapped];
3983 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3987 if (ret && point->fIndirectCRL)
3989 swapped[cSwapped].tag = ASN_CONTEXT | 4;
3990 swapped[cSwapped].pvStructInfo = &point->fIndirectCRL;
3991 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeBool;
3992 items[cItem].pvStructInfo = &swapped[cSwapped];
3993 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
3998 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
3999 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
4003 SetLastError(STATUS_ACCESS_VIOLATION);
4010 static BOOL CRYPT_AsnEncodeGeneralSubtree(DWORD dwCertEncodingType,
4011 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4012 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4015 const CERT_GENERAL_SUBTREE *subtree = pvStructInfo;
4016 struct AsnEncodeSequenceItem items[3] = {
4017 { &subtree->Base, CRYPT_AsnEncodeAltNameEntry, 0 },
4020 struct AsnEncodeTagSwappedItem swapped[2] = { { 0 } };
4021 DWORD cItem = 1, cSwapped = 0;
4023 if (subtree->dwMinimum)
4025 swapped[cSwapped].tag = ASN_CONTEXT | 0;
4026 swapped[cSwapped].pvStructInfo = &subtree->dwMinimum;
4027 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInt;
4028 items[cItem].pvStructInfo = &swapped[cSwapped];
4029 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4033 if (subtree->fMaximum)
4035 swapped[cSwapped].tag = ASN_CONTEXT | 1;
4036 swapped[cSwapped].pvStructInfo = &subtree->dwMaximum;
4037 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeInt;
4038 items[cItem].pvStructInfo = &swapped[cSwapped];
4039 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4043 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem, dwFlags,
4044 pEncodePara, pbEncoded, pcbEncoded);
4048 static BOOL WINAPI CRYPT_AsnEncodeNameConstraints(DWORD dwCertEncodingType,
4049 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4050 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4053 CRYPT_BLOB_ARRAY permitted = { 0, NULL }, excluded = { 0, NULL };
4055 TRACE("%p\n", pvStructInfo);
4059 const CERT_NAME_CONSTRAINTS_INFO *constraints = pvStructInfo;
4060 struct AsnEncodeSequenceItem items[2] = { { 0 } };
4061 struct AsnEncodeTagSwappedItem swapped[2] = { { 0 } };
4062 DWORD i, cItem = 0, cSwapped = 0;
4065 if (constraints->cPermittedSubtree)
4067 permitted.rgBlob = CryptMemAlloc(
4068 constraints->cPermittedSubtree * sizeof(CRYPT_DER_BLOB));
4069 if (permitted.rgBlob)
4071 permitted.cBlob = constraints->cPermittedSubtree;
4072 memset(permitted.rgBlob, 0,
4073 permitted.cBlob * sizeof(CRYPT_DER_BLOB));
4074 for (i = 0; ret && i < permitted.cBlob; i++)
4075 ret = CRYPT_AsnEncodeGeneralSubtree(dwCertEncodingType,
4076 NULL, &constraints->rgPermittedSubtree[i],
4077 CRYPT_ENCODE_ALLOC_FLAG, NULL,
4078 (BYTE *)&permitted.rgBlob[i].pbData,
4079 &permitted.rgBlob[i].cbData);
4082 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 0;
4083 swapped[cSwapped].pvStructInfo = &permitted;
4084 swapped[cSwapped].encodeFunc = CRYPT_DEREncodeSet;
4085 items[cItem].pvStructInfo = &swapped[cSwapped];
4086 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4094 if (constraints->cExcludedSubtree)
4096 excluded.rgBlob = CryptMemAlloc(
4097 constraints->cExcludedSubtree * sizeof(CRYPT_DER_BLOB));
4098 if (excluded.rgBlob)
4100 excluded.cBlob = constraints->cExcludedSubtree;
4101 memset(excluded.rgBlob, 0,
4102 excluded.cBlob * sizeof(CRYPT_DER_BLOB));
4103 for (i = 0; ret && i < excluded.cBlob; i++)
4104 ret = CRYPT_AsnEncodeGeneralSubtree(dwCertEncodingType,
4105 NULL, &constraints->rgExcludedSubtree[i],
4106 CRYPT_ENCODE_ALLOC_FLAG, NULL,
4107 (BYTE *)&excluded.rgBlob[i].pbData,
4108 &excluded.rgBlob[i].cbData);
4111 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 1;
4112 swapped[cSwapped].pvStructInfo = &excluded;
4113 swapped[cSwapped].encodeFunc = CRYPT_DEREncodeSet;
4114 items[cItem].pvStructInfo = &swapped[cSwapped];
4115 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4124 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
4125 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
4126 for (i = 0; i < permitted.cBlob; i++)
4127 LocalFree(permitted.rgBlob[i].pbData);
4128 for (i = 0; i < excluded.cBlob; i++)
4129 LocalFree(excluded.rgBlob[i].pbData);
4133 SetLastError(STATUS_ACCESS_VIOLATION);
4136 CryptMemFree(permitted.rgBlob);
4137 CryptMemFree(excluded.rgBlob);
4138 TRACE("returning %d\n", ret);
4142 static BOOL WINAPI CRYPT_AsnEncodeIssuerSerialNumber(
4143 DWORD dwCertEncodingType, LPCSTR lpszStructType, const void *pvStructInfo,
4144 DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded,
4148 const CERT_ISSUER_SERIAL_NUMBER *issuerSerial = pvStructInfo;
4149 struct AsnEncodeSequenceItem items[] = {
4150 { &issuerSerial->Issuer, CRYPT_CopyEncodedBlob, 0 },
4151 { &issuerSerial->SerialNumber, CRYPT_AsnEncodeInteger, 0 },
4154 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
4155 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
4160 static BOOL WINAPI CRYPT_AsnEncodePKCSSignerInfo(DWORD dwCertEncodingType,
4161 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4162 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4166 if (!(dwCertEncodingType & PKCS_7_ASN_ENCODING))
4168 SetLastError(E_INVALIDARG);
4174 const CMSG_SIGNER_INFO *info = pvStructInfo;
4176 if (!info->Issuer.cbData)
4177 SetLastError(E_INVALIDARG);
4180 struct AsnEncodeSequenceItem items[7] = {
4181 { &info->dwVersion, CRYPT_AsnEncodeInt, 0 },
4182 { &info->Issuer, CRYPT_AsnEncodeIssuerSerialNumber, 0 },
4183 { &info->HashAlgorithm, CRYPT_AsnEncodeAlgorithmIdWithNullParams,
4186 struct AsnEncodeTagSwappedItem swapped[2] = { { 0 } };
4187 DWORD cItem = 3, cSwapped = 0;
4189 if (info->AuthAttrs.cAttr)
4191 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 0;
4192 swapped[cSwapped].pvStructInfo = &info->AuthAttrs;
4193 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodePKCSAttributes;
4194 items[cItem].pvStructInfo = &swapped[cSwapped];
4195 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4199 items[cItem].pvStructInfo = &info->HashEncryptionAlgorithm;
4200 items[cItem].encodeFunc = CRYPT_AsnEncodeAlgorithmIdWithNullParams;
4202 items[cItem].pvStructInfo = &info->EncryptedHash;
4203 items[cItem].encodeFunc = CRYPT_AsnEncodeOctets;
4205 if (info->UnauthAttrs.cAttr)
4207 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 1;
4208 swapped[cSwapped].pvStructInfo = &info->UnauthAttrs;
4209 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodePKCSAttributes;
4210 items[cItem].pvStructInfo = &swapped[cSwapped];
4211 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4215 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
4216 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
4221 SetLastError(STATUS_ACCESS_VIOLATION);
4227 static BOOL WINAPI CRYPT_AsnEncodeCMSSignerInfo(DWORD dwCertEncodingType,
4228 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4229 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4233 if (!(dwCertEncodingType & PKCS_7_ASN_ENCODING))
4235 SetLastError(E_INVALIDARG);
4241 const CMSG_CMS_SIGNER_INFO *info = pvStructInfo;
4243 if (info->SignerId.dwIdChoice != CERT_ID_ISSUER_SERIAL_NUMBER &&
4244 info->SignerId.dwIdChoice != CERT_ID_KEY_IDENTIFIER)
4245 SetLastError(E_INVALIDARG);
4246 else if (info->SignerId.dwIdChoice == CERT_ID_ISSUER_SERIAL_NUMBER &&
4247 !info->SignerId.u.IssuerSerialNumber.Issuer.cbData)
4248 SetLastError(E_INVALIDARG);
4251 struct AsnEncodeSequenceItem items[7] = {
4252 { &info->dwVersion, CRYPT_AsnEncodeInt, 0 },
4254 struct AsnEncodeTagSwappedItem swapped[3] = { { 0 } };
4255 DWORD cItem = 1, cSwapped = 0;
4257 if (info->SignerId.dwIdChoice == CERT_ID_ISSUER_SERIAL_NUMBER)
4259 items[cItem].pvStructInfo =
4260 &info->SignerId.u.IssuerSerialNumber.Issuer;
4261 items[cItem].encodeFunc =
4262 CRYPT_AsnEncodeIssuerSerialNumber;
4267 swapped[cSwapped].tag = ASN_CONTEXT | 0;
4268 swapped[cSwapped].pvStructInfo = &info->SignerId.u.KeyId;
4269 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodeOctets;
4270 items[cItem].pvStructInfo = &swapped[cSwapped];
4271 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4275 items[cItem].pvStructInfo = &info->HashAlgorithm;
4276 items[cItem].encodeFunc = CRYPT_AsnEncodeAlgorithmIdWithNullParams;
4278 if (info->AuthAttrs.cAttr)
4280 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 0;
4281 swapped[cSwapped].pvStructInfo = &info->AuthAttrs;
4282 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodePKCSAttributes;
4283 items[cItem].pvStructInfo = &swapped[cSwapped];
4284 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4288 items[cItem].pvStructInfo = &info->HashEncryptionAlgorithm;
4289 items[cItem].encodeFunc = CRYPT_AsnEncodeAlgorithmIdWithNullParams;
4291 items[cItem].pvStructInfo = &info->EncryptedHash;
4292 items[cItem].encodeFunc = CRYPT_AsnEncodeOctets;
4294 if (info->UnauthAttrs.cAttr)
4296 swapped[cSwapped].tag = ASN_CONTEXT | ASN_CONSTRUCTOR | 1;
4297 swapped[cSwapped].pvStructInfo = &info->UnauthAttrs;
4298 swapped[cSwapped].encodeFunc = CRYPT_AsnEncodePKCSAttributes;
4299 items[cItem].pvStructInfo = &swapped[cSwapped];
4300 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4304 ret = CRYPT_AsnEncodeSequence(dwCertEncodingType, items, cItem,
4305 dwFlags, pEncodePara, pbEncoded, pcbEncoded);
4310 SetLastError(STATUS_ACCESS_VIOLATION);
4316 BOOL CRYPT_AsnEncodeCMSSignedInfo(CRYPT_SIGNED_INFO *signedInfo, void *pvData,
4319 struct AsnEncodeSequenceItem items[7] = {
4320 { &signedInfo->version, CRYPT_AsnEncodeInt, 0 },
4322 struct DERSetDescriptor digestAlgorithmsSet = { 0 }, certSet = { 0 };
4323 struct DERSetDescriptor crlSet = { 0 }, signerSet = { 0 };
4324 struct AsnEncodeTagSwappedItem swapped[2] = { { 0 } };
4325 DWORD cItem = 1, cSwapped = 0;
4328 if (signedInfo->cSignerInfo)
4330 digestAlgorithmsSet.cItems = signedInfo->cSignerInfo;
4331 digestAlgorithmsSet.items = signedInfo->rgSignerInfo;
4332 digestAlgorithmsSet.itemSize = sizeof(CMSG_CMS_SIGNER_INFO);
4333 digestAlgorithmsSet.itemOffset =
4334 offsetof(CMSG_CMS_SIGNER_INFO, HashAlgorithm);
4335 digestAlgorithmsSet.encode = CRYPT_AsnEncodeAlgorithmIdWithNullParams;
4336 items[cItem].pvStructInfo = &digestAlgorithmsSet;
4337 items[cItem].encodeFunc = CRYPT_DEREncodeItemsAsSet;
4340 items[cItem].pvStructInfo = &signedInfo->content;
4341 items[cItem].encodeFunc = CRYPT_AsnEncodePKCSContentInfoInternal;
4343 if (signedInfo->cCertEncoded)
4345 certSet.cItems = signedInfo->cCertEncoded;
4346 certSet.items = signedInfo->rgCertEncoded;
4347 certSet.itemSize = sizeof(CERT_BLOB);
4348 certSet.itemOffset = 0;
4349 certSet.encode = CRYPT_CopyEncodedBlob;
4350 swapped[cSwapped].tag = ASN_CONSTRUCTOR | ASN_CONTEXT | 0;
4351 swapped[cSwapped].pvStructInfo = &certSet;
4352 swapped[cSwapped].encodeFunc = CRYPT_DEREncodeItemsAsSet;
4353 items[cItem].pvStructInfo = &swapped[cSwapped];
4354 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4358 if (signedInfo->cCrlEncoded)
4360 crlSet.cItems = signedInfo->cCrlEncoded;
4361 crlSet.items = signedInfo->rgCrlEncoded;
4362 crlSet.itemSize = sizeof(CRL_BLOB);
4363 crlSet.itemOffset = 0;
4364 crlSet.encode = CRYPT_CopyEncodedBlob;
4365 swapped[cSwapped].tag = ASN_CONSTRUCTOR | ASN_CONTEXT | 1;
4366 swapped[cSwapped].pvStructInfo = &crlSet;
4367 swapped[cSwapped].encodeFunc = CRYPT_DEREncodeItemsAsSet;
4368 items[cItem].pvStructInfo = &swapped[cSwapped];
4369 items[cItem].encodeFunc = CRYPT_AsnEncodeSwapTag;
4373 if (ret && signedInfo->cSignerInfo)
4375 signerSet.cItems = signedInfo->cSignerInfo;
4376 signerSet.items = signedInfo->rgSignerInfo;
4377 signerSet.itemSize = sizeof(CMSG_CMS_SIGNER_INFO);
4378 signerSet.itemOffset = 0;
4379 signerSet.encode = CRYPT_AsnEncodeCMSSignerInfo;
4380 items[cItem].pvStructInfo = &signerSet;
4381 items[cItem].encodeFunc = CRYPT_DEREncodeItemsAsSet;
4385 ret = CRYPT_AsnEncodeSequence(X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
4386 items, cItem, 0, NULL, pvData, pcbData);
4391 static BOOL WINAPI CRYPT_AsnEncodeRecipientInfo(DWORD dwCertEncodingType,
4392 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4393 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4395 const CMSG_KEY_TRANS_RECIPIENT_INFO *info = pvStructInfo;
4396 struct AsnEncodeSequenceItem items[] = {
4397 { &info->dwVersion, CRYPT_AsnEncodeInt, 0 },
4398 { &info->RecipientId.u.IssuerSerialNumber,
4399 CRYPT_AsnEncodeIssuerSerialNumber, 0 },
4400 { &info->KeyEncryptionAlgorithm,
4401 CRYPT_AsnEncodeAlgorithmIdWithNullParams, 0 },
4402 { &info->EncryptedKey, CRYPT_AsnEncodeOctets, 0 },
4405 return CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
4406 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
4410 static BOOL WINAPI CRYPT_AsnEncodeEncryptedContentInfo(DWORD dwCertEncodingType,
4411 LPCSTR lpszStructType, const void *pvStructInfo, DWORD dwFlags,
4412 PCRYPT_ENCODE_PARA pEncodePara, BYTE *pbEncoded, DWORD *pcbEncoded)
4414 const CRYPT_ENCRYPTED_CONTENT_INFO *info = pvStructInfo;
4415 struct AsnEncodeTagSwappedItem swapped = { ASN_CONTEXT | 0,
4416 &info->encryptedContent, CRYPT_AsnEncodeOctets };
4417 struct AsnEncodeSequenceItem items[] = {
4418 { info->contentType, CRYPT_AsnEncodeOid, 0 },
4419 { &info->contentEncryptionAlgorithm,
4420 CRYPT_AsnEncodeAlgorithmIdWithNullParams, 0 },
4421 { &swapped, CRYPT_AsnEncodeSwapTag, 0 },
4424 return CRYPT_AsnEncodeSequence(dwCertEncodingType, items,
4425 sizeof(items) / sizeof(items[0]), dwFlags, pEncodePara, pbEncoded,
4429 BOOL CRYPT_AsnEncodePKCSEnvelopedData(const CRYPT_ENVELOPED_DATA *envelopedData,
4430 void *pvData, DWORD *pcbData)
4432 struct DERSetDescriptor recipientInfosSet = { envelopedData->cRecipientInfo,
4433 envelopedData->rgRecipientInfo, sizeof(CMSG_KEY_TRANS_RECIPIENT_INFO), 0,
4434 CRYPT_AsnEncodeRecipientInfo };
4435 struct AsnEncodeSequenceItem items[] = {
4436 { &envelopedData->version, CRYPT_AsnEncodeInt, 0 },
4437 { &recipientInfosSet, CRYPT_DEREncodeItemsAsSet, 0 },
4438 { &envelopedData->encryptedContentInfo,
4439 CRYPT_AsnEncodeEncryptedContentInfo, 0 },
4442 return CRYPT_AsnEncodeSequence(X509_ASN_ENCODING, items,
4443 sizeof(items) / sizeof(items[0]), 0, NULL, pvData, pcbData);
4446 static CryptEncodeObjectExFunc CRYPT_GetBuiltinEncoder(DWORD dwCertEncodingType,
4447 LPCSTR lpszStructType)
4449 CryptEncodeObjectExFunc encodeFunc = NULL;
4451 if ((dwCertEncodingType & CERT_ENCODING_TYPE_MASK) != X509_ASN_ENCODING
4452 && (dwCertEncodingType & CMSG_ENCODING_TYPE_MASK) != PKCS_7_ASN_ENCODING)
4454 SetLastError(ERROR_FILE_NOT_FOUND);
4458 if (IS_INTOID(lpszStructType))
4460 switch (LOWORD(lpszStructType))
4462 case LOWORD(X509_CERT):
4463 encodeFunc = CRYPT_AsnEncodeCert;
4465 case LOWORD(X509_CERT_TO_BE_SIGNED):
4466 encodeFunc = CRYPT_AsnEncodeCertInfo;
4468 case LOWORD(X509_CERT_CRL_TO_BE_SIGNED):
4469 encodeFunc = CRYPT_AsnEncodeCRLInfo;
4471 case LOWORD(X509_EXTENSIONS):
4472 encodeFunc = CRYPT_AsnEncodeExtensions;
4474 case LOWORD(X509_NAME_VALUE):
4475 encodeFunc = CRYPT_AsnEncodeNameValue;
4477 case LOWORD(X509_NAME):
4478 encodeFunc = CRYPT_AsnEncodeName;
4480 case LOWORD(X509_PUBLIC_KEY_INFO):
4481 encodeFunc = CRYPT_AsnEncodePubKeyInfo;
4483 case LOWORD(X509_AUTHORITY_KEY_ID):
4484 encodeFunc = CRYPT_AsnEncodeAuthorityKeyId;
4486 case LOWORD(X509_ALTERNATE_NAME):
4487 encodeFunc = CRYPT_AsnEncodeAltName;
4489 case LOWORD(X509_BASIC_CONSTRAINTS):
4490 encodeFunc = CRYPT_AsnEncodeBasicConstraints;
4492 case LOWORD(X509_BASIC_CONSTRAINTS2):
4493 encodeFunc = CRYPT_AsnEncodeBasicConstraints2;
4495 case LOWORD(X509_CERT_POLICIES):
4496 encodeFunc = CRYPT_AsnEncodeCertPolicies;
4498 case LOWORD(RSA_CSP_PUBLICKEYBLOB):
4499 encodeFunc = CRYPT_AsnEncodeRsaPubKey;
4501 case LOWORD(X509_UNICODE_NAME):
4502 encodeFunc = CRYPT_AsnEncodeUnicodeName;
4504 case LOWORD(PKCS_CONTENT_INFO):
4505 encodeFunc = CRYPT_AsnEncodePKCSContentInfo;
4507 case LOWORD(PKCS_ATTRIBUTE):
4508 encodeFunc = CRYPT_AsnEncodePKCSAttribute;
4510 case LOWORD(X509_UNICODE_NAME_VALUE):
4511 encodeFunc = CRYPT_AsnEncodeUnicodeNameValue;
4513 case LOWORD(X509_OCTET_STRING):
4514 encodeFunc = CRYPT_AsnEncodeOctets;
4516 case LOWORD(X509_BITS):
4517 case LOWORD(X509_KEY_USAGE):
4518 encodeFunc = CRYPT_AsnEncodeBits;
4520 case LOWORD(X509_INTEGER):
4521 encodeFunc = CRYPT_AsnEncodeInt;
4523 case LOWORD(X509_MULTI_BYTE_INTEGER):
4524 encodeFunc = CRYPT_AsnEncodeInteger;
4526 case LOWORD(X509_MULTI_BYTE_UINT):
4527 encodeFunc = CRYPT_AsnEncodeUnsignedInteger;
4529 case LOWORD(X509_ENUMERATED):
4530 encodeFunc = CRYPT_AsnEncodeEnumerated;
4532 case LOWORD(X509_CHOICE_OF_TIME):
4533 encodeFunc = CRYPT_AsnEncodeChoiceOfTime;
4535 case LOWORD(X509_AUTHORITY_KEY_ID2):
4536 encodeFunc = CRYPT_AsnEncodeAuthorityKeyId2;
4538 case LOWORD(X509_AUTHORITY_INFO_ACCESS):
4539 encodeFunc = CRYPT_AsnEncodeAuthorityInfoAccess;
4541 case LOWORD(X509_SEQUENCE_OF_ANY):
4542 encodeFunc = CRYPT_AsnEncodeSequenceOfAny;
4544 case LOWORD(PKCS_UTC_TIME):
4545 encodeFunc = CRYPT_AsnEncodeUtcTime;
4547 case LOWORD(X509_CRL_DIST_POINTS):
4548 encodeFunc = CRYPT_AsnEncodeCRLDistPoints;
4550 case LOWORD(X509_ENHANCED_KEY_USAGE):
4551 encodeFunc = CRYPT_AsnEncodeEnhancedKeyUsage;
4553 case LOWORD(PKCS_CTL):
4554 encodeFunc = CRYPT_AsnEncodeCTL;
4556 case LOWORD(PKCS_SMIME_CAPABILITIES):
4557 encodeFunc = CRYPT_AsnEncodeSMIMECapabilities;
4559 case LOWORD(X509_PKIX_POLICY_QUALIFIER_USERNOTICE):
4560 encodeFunc = CRYPT_AsnEncodePolicyQualifierUserNotice;
4562 case LOWORD(PKCS_ATTRIBUTES):
4563 encodeFunc = CRYPT_AsnEncodePKCSAttributes;
4565 case LOWORD(X509_ISSUING_DIST_POINT):
4566 encodeFunc = CRYPT_AsnEncodeIssuingDistPoint;
4568 case LOWORD(X509_NAME_CONSTRAINTS):
4569 encodeFunc = CRYPT_AsnEncodeNameConstraints;
4571 case LOWORD(X509_POLICY_MAPPINGS):
4572 encodeFunc = CRYPT_AsnEncodeCertPolicyMappings;
4574 case LOWORD(X509_POLICY_CONSTRAINTS):
4575 encodeFunc = CRYPT_AsnEncodeCertPolicyConstraints;
4577 case LOWORD(PKCS7_SIGNER_INFO):
4578 encodeFunc = CRYPT_AsnEncodePKCSSignerInfo;
4580 case LOWORD(CMS_SIGNER_INFO):
4581 encodeFunc = CRYPT_AsnEncodeCMSSignerInfo;
4585 else if (!strcmp(lpszStructType, szOID_CERT_EXTENSIONS))
4586 encodeFunc = CRYPT_AsnEncodeExtensions;
4587 else if (!strcmp(lpszStructType, szOID_RSA_signingTime))
4588 encodeFunc = CRYPT_AsnEncodeUtcTime;
4589 else if (!strcmp(lpszStructType, szOID_RSA_SMIMECapabilities))
4590 encodeFunc = CRYPT_AsnEncodeUtcTime;
4591 else if (!strcmp(lpszStructType, szOID_AUTHORITY_KEY_IDENTIFIER))
4592 encodeFunc = CRYPT_AsnEncodeAuthorityKeyId;
4593 else if (!strcmp(lpszStructType, szOID_LEGACY_POLICY_MAPPINGS))
4594 encodeFunc = CRYPT_AsnEncodeCertPolicyMappings;
4595 else if (!strcmp(lpszStructType, szOID_AUTHORITY_KEY_IDENTIFIER2))
4596 encodeFunc = CRYPT_AsnEncodeAuthorityKeyId2;
4597 else if (!strcmp(lpszStructType, szOID_CRL_REASON_CODE))
4598 encodeFunc = CRYPT_AsnEncodeEnumerated;
4599 else if (!strcmp(lpszStructType, szOID_KEY_USAGE))
4600 encodeFunc = CRYPT_AsnEncodeBits;
4601 else if (!strcmp(lpszStructType, szOID_SUBJECT_KEY_IDENTIFIER))
4602 encodeFunc = CRYPT_AsnEncodeOctets;
4603 else if (!strcmp(lpszStructType, szOID_BASIC_CONSTRAINTS))
4604 encodeFunc = CRYPT_AsnEncodeBasicConstraints;
4605 else if (!strcmp(lpszStructType, szOID_BASIC_CONSTRAINTS2))
4606 encodeFunc = CRYPT_AsnEncodeBasicConstraints2;
4607 else if (!strcmp(lpszStructType, szOID_ISSUER_ALT_NAME))
4608 encodeFunc = CRYPT_AsnEncodeAltName;
4609 else if (!strcmp(lpszStructType, szOID_ISSUER_ALT_NAME2))
4610 encodeFunc = CRYPT_AsnEncodeAltName;
4611 else if (!strcmp(lpszStructType, szOID_NEXT_UPDATE_LOCATION))
4612 encodeFunc = CRYPT_AsnEncodeAltName;
4613 else if (!strcmp(lpszStructType, szOID_SUBJECT_ALT_NAME))
4614 encodeFunc = CRYPT_AsnEncodeAltName;
4615 else if (!strcmp(lpszStructType, szOID_SUBJECT_ALT_NAME2))
4616 encodeFunc = CRYPT_AsnEncodeAltName;
4617 else if (!strcmp(lpszStructType, szOID_CRL_DIST_POINTS))
4618 encodeFunc = CRYPT_AsnEncodeCRLDistPoints;
4619 else if (!strcmp(lpszStructType, szOID_CERT_POLICIES))
4620 encodeFunc = CRYPT_AsnEncodeCertPolicies;
4621 else if (!strcmp(lpszStructType, szOID_POLICY_MAPPINGS))
4622 encodeFunc = CRYPT_AsnEncodeCertPolicyMappings;
4623 else if (!strcmp(lpszStructType, szOID_POLICY_CONSTRAINTS))
4624 encodeFunc = CRYPT_AsnEncodeCertPolicyConstraints;
4625 else if (!strcmp(lpszStructType, szOID_ENHANCED_KEY_USAGE))
4626 encodeFunc = CRYPT_AsnEncodeEnhancedKeyUsage;
4627 else if (!strcmp(lpszStructType, szOID_ISSUING_DIST_POINT))
4628 encodeFunc = CRYPT_AsnEncodeIssuingDistPoint;
4629 else if (!strcmp(lpszStructType, szOID_NAME_CONSTRAINTS))
4630 encodeFunc = CRYPT_AsnEncodeNameConstraints;
4631 else if (!strcmp(lpszStructType, szOID_AUTHORITY_INFO_ACCESS))
4632 encodeFunc = CRYPT_AsnEncodeAuthorityInfoAccess;
4633 else if (!strcmp(lpszStructType, szOID_PKIX_POLICY_QUALIFIER_USERNOTICE))
4634 encodeFunc = CRYPT_AsnEncodePolicyQualifierUserNotice;
4635 else if (!strcmp(lpszStructType, szOID_CTL))
4636 encodeFunc = CRYPT_AsnEncodeCTL;
4640 static CryptEncodeObjectFunc CRYPT_LoadEncoderFunc(DWORD dwCertEncodingType,
4641 LPCSTR lpszStructType, HCRYPTOIDFUNCADDR *hFunc)
4643 static HCRYPTOIDFUNCSET set = NULL;
4644 CryptEncodeObjectFunc encodeFunc = NULL;
4647 set = CryptInitOIDFunctionSet(CRYPT_OID_ENCODE_OBJECT_FUNC, 0);
4648 CryptGetOIDFunctionAddress(set, dwCertEncodingType, lpszStructType, 0,
4649 (void **)&encodeFunc, hFunc);
4653 static CryptEncodeObjectExFunc CRYPT_LoadEncoderExFunc(DWORD dwCertEncodingType,
4654 LPCSTR lpszStructType, HCRYPTOIDFUNCADDR *hFunc)
4656 static HCRYPTOIDFUNCSET set = NULL;
4657 CryptEncodeObjectExFunc encodeFunc = NULL;
4660 set = CryptInitOIDFunctionSet(CRYPT_OID_ENCODE_OBJECT_EX_FUNC, 0);
4661 CryptGetOIDFunctionAddress(set, dwCertEncodingType, lpszStructType, 0,
4662 (void **)&encodeFunc, hFunc);
4666 BOOL WINAPI CryptEncodeObject(DWORD dwCertEncodingType, LPCSTR lpszStructType,
4667 const void *pvStructInfo, BYTE *pbEncoded, DWORD *pcbEncoded)
4670 HCRYPTOIDFUNCADDR hFunc = NULL;
4671 CryptEncodeObjectFunc pCryptEncodeObject = NULL;
4672 CryptEncodeObjectExFunc pCryptEncodeObjectEx = NULL;
4674 TRACE_(crypt)("(0x%08x, %s, %p, %p, %p)\n", dwCertEncodingType,
4675 debugstr_a(lpszStructType), pvStructInfo, pbEncoded,
4678 if (!pbEncoded && !pcbEncoded)
4680 SetLastError(ERROR_INVALID_PARAMETER);
4684 if (!(pCryptEncodeObjectEx = CRYPT_GetBuiltinEncoder(dwCertEncodingType,
4687 TRACE_(crypt)("OID %s not found or unimplemented, looking for DLL\n",
4688 debugstr_a(lpszStructType));
4689 pCryptEncodeObject = CRYPT_LoadEncoderFunc(dwCertEncodingType,
4690 lpszStructType, &hFunc);
4691 if (!pCryptEncodeObject)
4692 pCryptEncodeObjectEx = CRYPT_LoadEncoderExFunc(dwCertEncodingType,
4693 lpszStructType, &hFunc);
4695 if (pCryptEncodeObject)
4696 ret = pCryptEncodeObject(dwCertEncodingType, lpszStructType,
4697 pvStructInfo, pbEncoded, pcbEncoded);
4698 else if (pCryptEncodeObjectEx)
4699 ret = pCryptEncodeObjectEx(dwCertEncodingType, lpszStructType,
4700 pvStructInfo, 0, NULL, pbEncoded, pcbEncoded);
4702 CryptFreeOIDFunctionAddress(hFunc, 0);
4703 TRACE_(crypt)("returning %d\n", ret);
4707 BOOL WINAPI CryptEncodeObjectEx(DWORD dwCertEncodingType, LPCSTR lpszStructType,
4708 const void *pvStructInfo, DWORD dwFlags, PCRYPT_ENCODE_PARA pEncodePara,
4709 void *pvEncoded, DWORD *pcbEncoded)
4712 HCRYPTOIDFUNCADDR hFunc = NULL;
4713 CryptEncodeObjectExFunc encodeFunc = NULL;
4715 TRACE_(crypt)("(0x%08x, %s, %p, 0x%08x, %p, %p, %p)\n", dwCertEncodingType,
4716 debugstr_a(lpszStructType), pvStructInfo, dwFlags, pEncodePara,
4717 pvEncoded, pcbEncoded);
4719 if (!pvEncoded && !pcbEncoded)
4721 SetLastError(ERROR_INVALID_PARAMETER);
4725 SetLastError(NOERROR);
4726 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG) {
4728 SetLastError(ERROR_INVALID_PARAMETER);
4731 *(BYTE **)pvEncoded = NULL;
4733 encodeFunc = CRYPT_GetBuiltinEncoder(dwCertEncodingType, lpszStructType);
4736 TRACE_(crypt)("OID %s not found or unimplemented, looking for DLL\n",
4737 debugstr_a(lpszStructType));
4738 encodeFunc = CRYPT_LoadEncoderExFunc(dwCertEncodingType, lpszStructType,
4742 ret = encodeFunc(dwCertEncodingType, lpszStructType, pvStructInfo,
4743 dwFlags, pEncodePara, pvEncoded, pcbEncoded);
4746 CryptEncodeObjectFunc pCryptEncodeObject =
4747 CRYPT_LoadEncoderFunc(dwCertEncodingType, lpszStructType, &hFunc);
4749 if (pCryptEncodeObject)
4751 if (dwFlags & CRYPT_ENCODE_ALLOC_FLAG)
4753 ret = pCryptEncodeObject(dwCertEncodingType, lpszStructType,
4754 pvStructInfo, NULL, pcbEncoded);
4755 if (ret && (ret = CRYPT_EncodeEnsureSpace(dwFlags, pEncodePara,
4756 pvEncoded, pcbEncoded, *pcbEncoded)))
4757 ret = pCryptEncodeObject(dwCertEncodingType,
4758 lpszStructType, pvStructInfo, *(BYTE **)pvEncoded,
4762 ret = pCryptEncodeObject(dwCertEncodingType, lpszStructType,
4763 pvStructInfo, pvEncoded, pcbEncoded);
4767 CryptFreeOIDFunctionAddress(hFunc, 0);
4768 TRACE_(crypt)("returning %d\n", ret);
4772 BOOL WINAPI PFXExportCertStore(HCERTSTORE hStore, CRYPT_DATA_BLOB *pPFX,
4773 LPCWSTR szPassword, DWORD dwFlags)
4775 return PFXExportCertStoreEx(hStore, pPFX, szPassword, NULL, dwFlags);
4778 BOOL WINAPI PFXExportCertStoreEx(HCERTSTORE hStore, CRYPT_DATA_BLOB *pPFX,
4779 LPCWSTR szPassword, void *pvReserved, DWORD dwFlags)
4781 FIXME_(crypt)("(%p, %p, %p, %p, %08x): stub\n", hStore, pPFX, szPassword,
4782 pvReserved, dwFlags);
4786 BOOL WINAPI CryptExportPublicKeyInfo(HCRYPTPROV_OR_NCRYPT_KEY_HANDLE hCryptProv, DWORD dwKeySpec,
4787 DWORD dwCertEncodingType, PCERT_PUBLIC_KEY_INFO pInfo, DWORD *pcbInfo)
4789 return CryptExportPublicKeyInfoEx(hCryptProv, dwKeySpec, dwCertEncodingType,
4790 NULL, 0, NULL, pInfo, pcbInfo);
4793 static BOOL WINAPI CRYPT_ExportRsaPublicKeyInfoEx(HCRYPTPROV_OR_NCRYPT_KEY_HANDLE hCryptProv,
4794 DWORD dwKeySpec, DWORD dwCertEncodingType, LPSTR pszPublicKeyObjId,
4795 DWORD dwFlags, void *pvAuxInfo, PCERT_PUBLIC_KEY_INFO pInfo, DWORD *pcbInfo)
4799 static CHAR oid[] = szOID_RSA_RSA;
4801 TRACE_(crypt)("(%08lx, %d, %08x, %s, %08x, %p, %p, %d)\n", hCryptProv,
4802 dwKeySpec, dwCertEncodingType, debugstr_a(pszPublicKeyObjId), dwFlags,
4803 pvAuxInfo, pInfo, pInfo ? *pcbInfo : 0);
4805 if (!pszPublicKeyObjId)
4806 pszPublicKeyObjId = oid;
4807 if ((ret = CryptGetUserKey(hCryptProv, dwKeySpec, &key)))
4811 ret = CryptExportKey(key, 0, PUBLICKEYBLOB, 0, NULL, &keySize);
4814 LPBYTE pubKey = CryptMemAlloc(keySize);
4818 ret = CryptExportKey(key, 0, PUBLICKEYBLOB, 0, pubKey,
4822 DWORD encodedLen = 0;
4824 ret = CryptEncodeObject(dwCertEncodingType,
4825 RSA_CSP_PUBLICKEYBLOB, pubKey, NULL, &encodedLen);
4828 DWORD sizeNeeded = sizeof(CERT_PUBLIC_KEY_INFO) +
4829 strlen(pszPublicKeyObjId) + 1 + encodedLen;
4832 *pcbInfo = sizeNeeded;
4833 else if (*pcbInfo < sizeNeeded)
4835 SetLastError(ERROR_MORE_DATA);
4836 *pcbInfo = sizeNeeded;
4841 *pcbInfo = sizeNeeded;
4842 pInfo->Algorithm.pszObjId = (char *)pInfo +
4843 sizeof(CERT_PUBLIC_KEY_INFO);
4844 lstrcpyA(pInfo->Algorithm.pszObjId,
4846 pInfo->Algorithm.Parameters.cbData = 0;
4847 pInfo->Algorithm.Parameters.pbData = NULL;
4848 pInfo->PublicKey.pbData =
4849 (BYTE *)pInfo->Algorithm.pszObjId
4850 + lstrlenA(pInfo->Algorithm.pszObjId) + 1;
4851 pInfo->PublicKey.cbData = encodedLen;
4852 pInfo->PublicKey.cUnusedBits = 0;
4853 ret = CryptEncodeObject(dwCertEncodingType,
4854 RSA_CSP_PUBLICKEYBLOB, pubKey,
4855 pInfo->PublicKey.pbData, &pInfo->PublicKey.cbData);
4859 CryptMemFree(pubKey);
4864 CryptDestroyKey(key);
4869 typedef BOOL (WINAPI *ExportPublicKeyInfoExFunc)(HCRYPTPROV_OR_NCRYPT_KEY_HANDLE hCryptProv,
4870 DWORD dwKeySpec, DWORD dwCertEncodingType, LPSTR pszPublicKeyObjId,
4871 DWORD dwFlags, void *pvAuxInfo, PCERT_PUBLIC_KEY_INFO pInfo, DWORD *pcbInfo);
4873 BOOL WINAPI CryptExportPublicKeyInfoEx(HCRYPTPROV_OR_NCRYPT_KEY_HANDLE hCryptProv, DWORD dwKeySpec,
4874 DWORD dwCertEncodingType, LPSTR pszPublicKeyObjId, DWORD dwFlags,
4875 void *pvAuxInfo, PCERT_PUBLIC_KEY_INFO pInfo, DWORD *pcbInfo)
4877 static HCRYPTOIDFUNCSET set = NULL;
4879 ExportPublicKeyInfoExFunc exportFunc = NULL;
4880 HCRYPTOIDFUNCADDR hFunc = NULL;
4882 TRACE_(crypt)("(%08lx, %d, %08x, %s, %08x, %p, %p, %d)\n", hCryptProv,
4883 dwKeySpec, dwCertEncodingType, debugstr_a(pszPublicKeyObjId), dwFlags,
4884 pvAuxInfo, pInfo, pInfo ? *pcbInfo : 0);
4888 SetLastError(ERROR_INVALID_PARAMETER);
4892 if (pszPublicKeyObjId)
4895 set = CryptInitOIDFunctionSet(CRYPT_OID_EXPORT_PUBLIC_KEY_INFO_FUNC,
4897 CryptGetOIDFunctionAddress(set, dwCertEncodingType, pszPublicKeyObjId,
4898 0, (void **)&exportFunc, &hFunc);
4901 exportFunc = CRYPT_ExportRsaPublicKeyInfoEx;
4902 ret = exportFunc(hCryptProv, dwKeySpec, dwCertEncodingType,
4903 pszPublicKeyObjId, dwFlags, pvAuxInfo, pInfo, pcbInfo);
4905 CryptFreeOIDFunctionAddress(hFunc, 0);
4909 BOOL WINAPI CryptImportPublicKeyInfo(HCRYPTPROV hCryptProv,
4910 DWORD dwCertEncodingType, PCERT_PUBLIC_KEY_INFO pInfo, HCRYPTKEY *phKey)
4912 return CryptImportPublicKeyInfoEx(hCryptProv, dwCertEncodingType, pInfo,
4916 static BOOL WINAPI CRYPT_ImportRsaPublicKeyInfoEx(HCRYPTPROV hCryptProv,
4917 DWORD dwCertEncodingType, PCERT_PUBLIC_KEY_INFO pInfo, ALG_ID aiKeyAlg,
4918 DWORD dwFlags, void *pvAuxInfo, HCRYPTKEY *phKey)
4921 DWORD pubKeySize = 0;
4923 TRACE_(crypt)("(%08lx, %08x, %p, %08x, %08x, %p, %p)\n", hCryptProv,
4924 dwCertEncodingType, pInfo, aiKeyAlg, dwFlags, pvAuxInfo, phKey);
4926 ret = CryptDecodeObject(dwCertEncodingType, RSA_CSP_PUBLICKEYBLOB,
4927 pInfo->PublicKey.pbData, pInfo->PublicKey.cbData, 0, NULL, &pubKeySize);
4930 LPBYTE pubKey = CryptMemAlloc(pubKeySize);
4934 ret = CryptDecodeObject(dwCertEncodingType, RSA_CSP_PUBLICKEYBLOB,
4935 pInfo->PublicKey.pbData, pInfo->PublicKey.cbData, 0, pubKey,
4940 ((BLOBHEADER*)pubKey)->aiKeyAlg = aiKeyAlg;
4941 ret = CryptImportKey(hCryptProv, pubKey, pubKeySize, 0, 0,
4944 CryptMemFree(pubKey);
4952 typedef BOOL (WINAPI *ImportPublicKeyInfoExFunc)(HCRYPTPROV hCryptProv,
4953 DWORD dwCertEncodingType, PCERT_PUBLIC_KEY_INFO pInfo, ALG_ID aiKeyAlg,
4954 DWORD dwFlags, void *pvAuxInfo, HCRYPTKEY *phKey);
4956 BOOL WINAPI CryptImportPublicKeyInfoEx(HCRYPTPROV hCryptProv,
4957 DWORD dwCertEncodingType, PCERT_PUBLIC_KEY_INFO pInfo, ALG_ID aiKeyAlg,
4958 DWORD dwFlags, void *pvAuxInfo, HCRYPTKEY *phKey)
4960 static HCRYPTOIDFUNCSET set = NULL;
4962 ImportPublicKeyInfoExFunc importFunc = NULL;
4963 HCRYPTOIDFUNCADDR hFunc = NULL;
4965 TRACE_(crypt)("(%08lx, %08x, %p, %08x, %08x, %p, %p)\n", hCryptProv,
4966 dwCertEncodingType, pInfo, aiKeyAlg, dwFlags, pvAuxInfo, phKey);
4969 set = CryptInitOIDFunctionSet(CRYPT_OID_IMPORT_PUBLIC_KEY_INFO_FUNC, 0);
4970 CryptGetOIDFunctionAddress(set, dwCertEncodingType,
4971 pInfo->Algorithm.pszObjId, 0, (void **)&importFunc, &hFunc);
4973 importFunc = CRYPT_ImportRsaPublicKeyInfoEx;
4974 ret = importFunc(hCryptProv, dwCertEncodingType, pInfo, aiKeyAlg, dwFlags,
4977 CryptFreeOIDFunctionAddress(hFunc, 0);