[THEMES]
[reactos.git] / rostests / winetests / crypt32 / str.c
1 /*
2 * Unit test suite for crypt32.dll's Cert*ToStr and CertStrToName functions.
3 *
4 * Copyright 2006 Juan Lang, Aric Stewart for CodeWeavers
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20 //#include <stdio.h>
21 #include <stdarg.h>
22 #include <windef.h>
23 #include <winbase.h>
24 //#include <winerror.h>
25 #include <wincrypt.h>
26
27 #include <wine/test.h>
28
29 typedef struct _CertRDNAttrEncoding {
30 LPCSTR pszObjId;
31 DWORD dwValueType;
32 CERT_RDN_VALUE_BLOB Value;
33 LPCSTR str;
34 BOOL todo;
35 } CertRDNAttrEncoding, *PCertRDNAttrEncoding;
36
37 typedef struct _CertRDNAttrEncodingW {
38 LPCSTR pszObjId;
39 DWORD dwValueType;
40 CERT_RDN_VALUE_BLOB Value;
41 LPCWSTR str;
42 BOOL todo;
43 } CertRDNAttrEncodingW, *PCertRDNAttrEncodingW;
44
45 static BYTE bin1[] = { 0x55, 0x53 };
46 static BYTE bin2[] = { 0x4d, 0x69, 0x6e, 0x6e, 0x65, 0x73, 0x6f, 0x74,
47 0x61 };
48 static BYTE bin3[] = { 0x4d, 0x69, 0x6e, 0x6e, 0x65, 0x61, 0x70, 0x6f,
49 0x6c, 0x69, 0x73 };
50 static BYTE bin4[] = { 0x43, 0x6f, 0x64, 0x65, 0x57, 0x65, 0x61, 0x76,
51 0x65, 0x72, 0x73 };
52 static BYTE bin5[] = { 0x57, 0x69, 0x6e, 0x65, 0x20, 0x44, 0x65, 0x76,
53 0x65, 0x6c, 0x6f, 0x70, 0x6d, 0x65, 0x6e, 0x74 };
54 static BYTE bin6[] = { 0x6c, 0x6f, 0x63, 0x61, 0x6c, 0x68, 0x6f, 0x73,
55 0x74 };
56 static BYTE bin7[] = { 0x61, 0x72, 0x69, 0x63, 0x40, 0x63, 0x6f, 0x64,
57 0x65, 0x77, 0x65, 0x61, 0x76, 0x65, 0x72, 0x73, 0x2e, 0x63, 0x6f, 0x6d };
58 static BYTE bin8[] = {
59 0x65,0x00,0x50,0x00,0x4b,0x00,0x49,0x00,0x20,0x00,0x52,0x00,0x6f,0x00,0x6f,
60 0x00,0x74,0x00,0x20,0x00,0x43,0x00,0x65,0x00,0x72,0x00,0x74,0x00,0x69,0x00,
61 0x66,0x00,0x69,0x00,0x63,0x00,0x61,0x00,0x74,0x00,0x69,0x00,0x6f,0x00,0x6e,
62 0x00,0x20,0x00,0x41,0x00,0x75,0x00,0x74,0x00,0x68,0x00,0x6f,0x00,0x72,0x00,
63 0x69,0x00,0x74,0x00,0x79,0x00 };
64 static BYTE bin9[] = { 0x61, 0x62, 0x63, 0x22, 0x64, 0x65, 0x66 };
65 static BYTE bin10[] = { 0x61, 0x62, 0x63, 0x27, 0x64, 0x65, 0x66 };
66 static BYTE bin11[] = { 0x61, 0x62, 0x63, 0x2c, 0x20, 0x64, 0x65, 0x66 };
67 static BYTE bin12[] = { 0x20, 0x61, 0x62, 0x63, 0x20 };
68 static BYTE bin13[] = { 0x22, 0x64, 0x65, 0x66, 0x22 };
69 static BYTE bin14[] = { 0x31, 0x3b, 0x33 };
70
71 static const BYTE cert[] =
72 {0x30,0x82,0x2,0xbb,0x30,0x82,0x2,0x24,0x2,0x9,0x0,0xe3,0x5a,0x10,0xf1,0xfc,
73 0x4b,0xf3,0xa2,0x30,0xd,0x6,0x9,0x2a,0x86,0x48,0x86,0xf7,0xd,0x1,0x1,0x4,0x5,
74 0x0,0x30,0x81,0xa1,0x31,0xb,0x30,0x9,0x6,0x3,0x55,0x4,0x6,0x13,0x2,0x55,0x53,
75 0x31,0x12,0x30,0x10,0x6,0x3,0x55,0x4,0x8,0x13,0x9,0x4d,0x69,0x6e,0x6e,0x65,
76 0x73,0x6f,0x74,0x61,0x31,0x14,0x30,0x12,0x6,0x3,0x55,0x4,0x7,0x13,0xb,0x4d,
77 0x69,0x6e,0x6e,0x65,0x61,0x70,0x6f,0x6c,0x69,0x73,0x31,0x14,0x30,0x12,0x6,0x3,
78 0x55,0x4,0xa,0x13,0xb,0x43,0x6f,0x64,0x65,0x57,0x65,0x61,0x76,0x65,0x72,0x73,
79 0x31,0x19,0x30,0x17,0x6,0x3,0x55,0x4,0xb,0x13,0x10,0x57,0x69,0x6e,0x65,0x20,
80 0x44,0x65,0x76,0x65,0x6c,0x6f,0x70,0x6d,0x65,0x6e,0x74,0x31,0x12,0x30,0x10,
81 0x6,0x3,0x55,0x4,0x3,0x13,0x9,0x6c,0x6f,0x63,0x61,0x6c,0x68,0x6f,0x73,0x74,
82 0x31,0x23,0x30,0x21,0x6,0x9,0x2a,0x86,0x48,0x86,0xf7,0xd,0x1,0x9,0x1,0x16,
83 0x14,0x61,0x72,0x69,0x63,0x40,0x63,0x6f,0x64,0x65,0x77,0x65,0x61,0x76,0x65,
84 0x72,0x73,0x2e,0x63,0x6f,0x6d,0x30,0x1e,0x17,0xd,0x30,0x36,0x30,0x31,0x32,
85 0x35,0x31,0x33,0x35,0x37,0x32,0x34,0x5a,0x17,0xd,0x30,0x36,0x30,0x32,0x32,
86 0x34,0x31,0x33,0x35,0x37,0x32,0x34,0x5a,0x30,0x81,0xa1,0x31,0xb,0x30,0x9,0x6,
87 0x3,0x55,0x4,0x6,0x13,0x2,0x55,0x53,0x31,0x12,0x30,0x10,0x6,0x3,0x55,0x4,0x8,
88 0x13,0x9,0x4d,0x69,0x6e,0x6e,0x65,0x73,0x6f,0x74,0x61,0x31,0x14,0x30,0x12,0x6,
89 0x3,0x55,0x4,0x7,0x13,0xb,0x4d,0x69,0x6e,0x6e,0x65,0x61,0x70,0x6f,0x6c,0x69,
90 0x73,0x31,0x14,0x30,0x12,0x6,0x3,0x55,0x4,0xa,0x13,0xb,0x43,0x6f,0x64,0x65,
91 0x57,0x65,0x61,0x76,0x65,0x72,0x73,0x31,0x19,0x30,0x17,0x6,0x3,0x55,0x4,0xb,
92 0x13,0x10,0x57,0x69,0x6e,0x65,0x20,0x44,0x65,0x76,0x65,0x6c,0x6f,0x70,0x6d,
93 0x65,0x6e,0x74,0x31,0x12,0x30,0x10,0x6,0x3,0x55,0x4,0x3,0x13,0x9,0x6c,0x6f,
94 0x63,0x61,0x6c,0x68,0x6f,0x73,0x74,0x31,0x23,0x30,0x21,0x6,0x9,0x2a,0x86,0x48,
95 0x86,0xf7,0xd,0x1,0x9,0x1,0x16,0x14,0x61,0x72,0x69,0x63,0x40,0x63,0x6f,0x64,
96 0x65,0x77,0x65,0x61,0x76,0x65,0x72,0x73,0x2e,0x63,0x6f,0x6d,0x30,0x81,0x9f,
97 0x30,0xd,0x6,0x9,0x2a,0x86,0x48,0x86,0xf7,0xd,0x1,0x1,0x1,0x5,0x0,0x3,0x81,
98 0x8d,0x0,0x30,0x81,0x89,0x2,0x81,0x81,0x0,0x9b,0xb5,0x8f,0xaf,0xfb,0x9a,0xaf,
99 0xdc,0xa2,0x4d,0xb1,0xc8,0x72,0x44,0xef,0x79,0x7f,0x28,0xb6,0xfe,0x50,0xdc,
100 0x8a,0xf7,0x11,0x2f,0x90,0x70,0xed,0xa4,0xa9,0xd,0xbf,0x82,0x3e,0x56,0xd8,
101 0x36,0xb6,0x9,0x52,0x83,0xab,0x65,0x95,0x0,0xe2,0xea,0x3c,0x4f,0x85,0xb8,0xc,
102 0x41,0x42,0x77,0x5c,0x9d,0x44,0xeb,0xcf,0x7d,0x60,0x64,0x7a,0x6c,0x4c,0xac,
103 0x4a,0x9a,0x23,0x25,0x15,0xd7,0x92,0xb4,0x10,0xe7,0x95,0xad,0x4b,0x93,0xda,
104 0x6a,0x76,0xe0,0xa5,0xd2,0x13,0x8,0x12,0x30,0x68,0xde,0xb9,0x5b,0x6e,0x2a,
105 0x97,0x43,0xaa,0x7b,0x22,0x33,0x34,0xb1,0xca,0x5d,0x19,0xd8,0x42,0x26,0x45,
106 0xc6,0xe9,0x1d,0xee,0x7,0xc2,0x27,0x95,0x87,0xd8,0x12,0xec,0x4b,0x16,0x9f,0x2,
107 0x3,0x1,0x0,0x1,0x30,0xd,0x6,0x9,0x2a,0x86,0x48,0x86,0xf7,0xd,0x1,0x1,0x4,0x5,
108 0x0,0x3,0x81,0x81,0x0,0x96,0xf9,0xf6,0x6a,0x3d,0xd9,0xca,0x6e,0xd5,0x76,0x73,
109 0xab,0x75,0xc1,0xcc,0x98,0x44,0xc3,0xa9,0x90,0x68,0x88,0x76,0xb9,0xeb,0xb6,
110 0xbe,0x60,0x62,0xb9,0x67,0x1e,0xcc,0xf4,0xe1,0xe7,0x6c,0xc8,0x67,0x3f,0x1d,
111 0xf3,0x68,0x86,0x30,0xee,0xaa,0x92,0x61,0x37,0xd7,0x82,0x90,0x28,0xaa,0x7a,
112 0x18,0x88,0x60,0x14,0x88,0x75,0xc0,0x4a,0x4e,0x7d,0x48,0xe7,0x3,0xa6,0xfd,
113 0xd7,0xce,0x3c,0xe5,0x9b,0xaf,0x2f,0xdc,0xbb,0x7c,0xbd,0x20,0x49,0xd9,0x68,
114 0x37,0xeb,0x5d,0xbb,0xe2,0x6d,0x66,0xe3,0x11,0xc1,0xa7,0x88,0x49,0xc6,0x6f,
115 0x65,0xd3,0xce,0xae,0x26,0x19,0x3,0x2e,0x4f,0x78,0xa5,0xa,0x97,0x7e,0x4f,0xc4,
116 0x91,0x8a,0xf8,0x5,0xef,0x5b,0x3b,0x49,0xbf,0x5f,0x2b};
117
118 static char issuerStr[] =
119 "US, Minnesota, Minneapolis, CodeWeavers, Wine Development, localhost, aric@codeweavers.com";
120 static char issuerStrSemicolon[] =
121 "US; Minnesota; Minneapolis; CodeWeavers; Wine Development; localhost; aric@codeweavers.com";
122 static char issuerStrCRLF[] =
123 "US\r\nMinnesota\r\nMinneapolis\r\nCodeWeavers\r\nWine Development\r\nlocalhost\r\naric@codeweavers.com";
124 static char subjectStr[] =
125 "2.5.4.6=US, 2.5.4.8=Minnesota, 2.5.4.7=Minneapolis, 2.5.4.10=CodeWeavers, 2.5.4.11=Wine Development, 2.5.4.3=localhost, 1.2.840.113549.1.9.1=aric@codeweavers.com";
126 static char subjectStrSemicolon[] =
127 "2.5.4.6=US; 2.5.4.8=Minnesota; 2.5.4.7=Minneapolis; 2.5.4.10=CodeWeavers; 2.5.4.11=Wine Development; 2.5.4.3=localhost; 1.2.840.113549.1.9.1=aric@codeweavers.com";
128 static char subjectStrCRLF[] =
129 "2.5.4.6=US\r\n2.5.4.8=Minnesota\r\n2.5.4.7=Minneapolis\r\n2.5.4.10=CodeWeavers\r\n2.5.4.11=Wine Development\r\n2.5.4.3=localhost\r\n1.2.840.113549.1.9.1=aric@codeweavers.com";
130 static char x500SubjectStr[] = "C=US, S=Minnesota, L=Minneapolis, O=CodeWeavers, OU=Wine Development, CN=localhost, E=aric@codeweavers.com";
131 static char x500SubjectStrSemicolonReverse[] = "E=aric@codeweavers.com; CN=localhost; OU=Wine Development; O=CodeWeavers; L=Minneapolis; S=Minnesota; C=US";
132 static WCHAR issuerStrW[] = {
133 'U','S',',',' ','M','i','n','n','e','s','o','t','a',',',' ','M','i','n','n',
134 'e','a','p','o','l','i','s',',',' ','C','o','d','e','W','e','a','v','e','r',
135 's',',',' ','W','i','n','e',' ','D','e','v','e','l','o','p','m','e','n','t',
136 ',',' ','l','o','c','a','l','h','o','s','t',',',' ','a','r','i','c','@','c',
137 'o','d','e','w','e','a','v','e','r','s','.','c','o','m',0 };
138 static WCHAR issuerStrSemicolonW[] = {
139 'U','S',';',' ','M','i','n','n','e','s','o','t','a',';',' ','M','i','n','n',
140 'e','a','p','o','l','i','s',';',' ','C','o','d','e','W','e','a','v','e','r',
141 's',';',' ','W','i','n','e',' ','D','e','v','e','l','o','p','m','e','n','t',
142 ';',' ','l','o','c','a','l','h','o','s','t',';',' ','a','r','i','c','@','c',
143 'o','d','e','w','e','a','v','e','r','s','.','c','o','m',0 };
144 static WCHAR issuerStrCRLFW[] = {
145 'U','S','\r','\n','M','i','n','n','e','s','o','t','a','\r','\n','M','i','n',
146 'n','e','a','p','o','l','i','s','\r','\n','C','o','d','e','W','e','a','v','e',
147 'r','s','\r','\n','W','i','n','e',' ','D','e','v','e','l','o','p','m','e','n',
148 't','\r','\n','l','o','c','a','l','h','o','s','t','\r','\n','a','r','i','c',
149 '@','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m',0 };
150 static WCHAR subjectStrW[] = {
151 '2','.','5','.','4','.','6','=','U','S',',',' ','2','.','5','.','4','.','8',
152 '=','M','i','n','n','e','s','o','t','a',',',' ','2','.','5','.','4','.','7',
153 '=','M','i','n','n','e','a','p','o','l','i','s',',',' ','2','.','5','.','4',
154 '.','1','0','=','C','o','d','e','W','e','a','v','e','r','s',',',' ','2','.',
155 '5','.','4','.','1','1','=','W','i','n','e',' ','D','e','v','e','l','o','p',
156 'm','e','n','t',',',' ','2','.','5','.','4','.','3','=','l','o','c','a','l',
157 'h','o','s','t',',',' ','1','.','2','.','8','4','0','.','1','1','3','5','4',
158 '9','.','1','.','9','.','1','=','a','r','i','c','@','c','o','d','e','w','e',
159 'a','v','e','r','s','.','c','o','m',0 };
160 static WCHAR subjectStrSemicolonW[] = {
161 '2','.','5','.','4','.','6','=','U','S',';',' ','2','.','5','.','4','.','8',
162 '=','M','i','n','n','e','s','o','t','a',';',' ','2','.','5','.','4','.','7',
163 '=','M','i','n','n','e','a','p','o','l','i','s',';',' ','2','.','5','.','4',
164 '.','1','0','=','C','o','d','e','W','e','a','v','e','r','s',';',' ','2','.',
165 '5','.','4','.','1','1','=','W','i','n','e',' ','D','e','v','e','l','o','p',
166 'm','e','n','t',';',' ','2','.','5','.','4','.','3','=','l','o','c','a','l',
167 'h','o','s','t',';',' ','1','.','2','.','8','4','0','.','1','1','3','5','4',
168 '9','.','1','.','9','.','1','=','a','r','i','c','@','c','o','d','e','w','e',
169 'a','v','e','r','s','.','c','o','m',0 };
170 static WCHAR subjectStrCRLFW[] = {
171 '2','.','5','.','4','.','6','=','U','S','\r','\n','2','.','5','.','4','.','8',
172 '=','M','i','n','n','e','s','o','t','a','\r','\n','2','.','5','.','4','.','7',
173 '=','M','i','n','n','e','a','p','o','l','i','s','\r','\n','2','.','5','.','4',
174 '.','1','0','=','C','o','d','e','W','e','a','v','e','r','s','\r','\n','2','.',
175 '5','.','4','.','1','1','=','W','i','n','e',' ','D','e','v','e','l','o','p',
176 'm','e','n','t','\r','\n','2','.','5','.','4','.','3','=','l','o','c','a','l',
177 'h','o','s','t','\r','\n','1','.','2','.','8','4','0','.','1','1','3','5','4',
178 '9','.','1','.','9','.','1','=','a','r','i','c','@','c','o','d','e','w','e',
179 'a','v','e','r','s','.','c','o','m',0 };
180 static WCHAR x500SubjectStrSemicolonReverseW[] = {
181 'E','=','a','r','i','c','@','c','o','d','e','w','e','a','v','e','r','s','.','c',
182 'o','m',';',' ','C','N','=','l','o','c','a','l','h','o','s','t',';',' ','O','U',
183 '=','W','i','n','e',' ','D','e','v','e','l','o','p','m','e','n','t',';',' ','O',
184 '=','C','o','d','e','W','e','a','v','e','r','s',';',' ','L','=','M','i','n','n',
185 'e','a','p','o','l','i','s',';',' ','S','=','M','i','n','n','e','s','o','t','a',
186 ';',' ','C','=','U','S',0 };
187
188 static HMODULE dll;
189 static DWORD (WINAPI *pCertNameToStrA)(DWORD,LPVOID,DWORD,LPSTR,DWORD);
190 static DWORD (WINAPI *pCertNameToStrW)(DWORD,LPVOID,DWORD,LPWSTR,DWORD);
191 static DWORD (WINAPI *pCertRDNValueToStrA)(DWORD, PCERT_RDN_VALUE_BLOB,
192 LPSTR, DWORD);
193 static DWORD (WINAPI *pCertRDNValueToStrW)(DWORD, PCERT_RDN_VALUE_BLOB,
194 LPWSTR, DWORD);
195 static BOOL (WINAPI *pCertStrToNameA)(DWORD dwCertEncodingType,
196 LPCSTR pszX500, DWORD dwStrType, void *pvReserved, BYTE *pbEncoded,
197 DWORD *pcbEncoded, LPCSTR *ppszError);
198 static BOOL (WINAPI *pCertStrToNameW)(DWORD dwCertEncodingType,
199 LPCWSTR pszX500, DWORD dwStrType, void *pvReserved, BYTE *pbEncoded,
200 DWORD *pcbEncoded, LPCWSTR *ppszError);
201 static DWORD (WINAPI *pCertGetNameStringA)(PCCERT_CONTEXT cert, DWORD type,
202 DWORD flags, void *typePara, LPSTR str, DWORD cch);
203
204
205 static void test_CertRDNValueToStrA(void)
206 {
207 CertRDNAttrEncoding attrs[] = {
208 { "2.5.4.6", CERT_RDN_PRINTABLE_STRING,
209 { sizeof(bin1), bin1 }, "US", FALSE },
210 { "2.5.4.8", CERT_RDN_PRINTABLE_STRING,
211 { sizeof(bin2), bin2 }, "Minnesota", FALSE },
212 { "2.5.4.7", CERT_RDN_PRINTABLE_STRING,
213 { sizeof(bin3), bin3 }, "Minneapolis", FALSE },
214 { "2.5.4.10", CERT_RDN_PRINTABLE_STRING,
215 { sizeof(bin4), bin4 }, "CodeWeavers", FALSE },
216 { "2.5.4.11", CERT_RDN_PRINTABLE_STRING,
217 { sizeof(bin5), bin5 }, "Wine Development", FALSE },
218 { "2.5.4.3", CERT_RDN_PRINTABLE_STRING,
219 { sizeof(bin6), bin6 }, "localhost", FALSE },
220 { "1.2.840.113549.1.9.1", CERT_RDN_IA5_STRING,
221 { sizeof(bin7), bin7 }, "aric@codeweavers.com", FALSE },
222 { "0", CERT_RDN_PRINTABLE_STRING,
223 { sizeof(bin9), bin9 }, "abc\"def", FALSE },
224 { "0", CERT_RDN_PRINTABLE_STRING,
225 { sizeof(bin10), bin10 }, "abc'def", FALSE },
226 { "0", CERT_RDN_PRINTABLE_STRING,
227 { sizeof(bin11), bin11 }, "abc, def", FALSE },
228 { "0", CERT_RDN_PRINTABLE_STRING,
229 { sizeof(bin12), bin12 }, " abc ", FALSE },
230 { "0", CERT_RDN_PRINTABLE_STRING,
231 { sizeof(bin13), bin13 }, "\"def\"", FALSE },
232 { "0", CERT_RDN_PRINTABLE_STRING,
233 { sizeof(bin14), bin14 }, "1;3", FALSE },
234 };
235 DWORD i, ret;
236 char buffer[2000];
237 CERT_RDN_VALUE_BLOB blob = { 0, NULL };
238 static const char ePKI[] = "ePKI Root Certification Authority";
239
240 if (!pCertRDNValueToStrA) return;
241
242 /* This crashes
243 ret = pCertRDNValueToStrA(0, NULL, NULL, 0);
244 */
245 /* With empty input, it generates the empty string */
246 SetLastError(0xdeadbeef);
247 ret = pCertRDNValueToStrA(0, &blob, NULL, 0);
248 ok(ret == 1 && GetLastError() == 0xdeadbeef, "Expected empty string\n");
249 ret = pCertRDNValueToStrA(0, &blob, buffer, sizeof(buffer));
250 ok(ret == 1 && GetLastError() == 0xdeadbeef, "Expected empty string\n");
251 ok(!buffer[0], "Expected empty string\n");
252
253 for (i = 0; i < sizeof(attrs) / sizeof(attrs[0]); i++)
254 {
255 ret = pCertRDNValueToStrA(attrs[i].dwValueType, &attrs[i].Value,
256 buffer, sizeof(buffer));
257 if (attrs[i].todo)
258 {
259 todo_wine
260 ok(ret == strlen(attrs[i].str) + 1, "Expected length %d, got %d\n",
261 lstrlenA(attrs[i].str) + 1, ret);
262 todo_wine
263 ok(!strcmp(buffer, attrs[i].str), "Expected %s, got %s\n",
264 attrs[i].str, buffer);
265 }
266 else
267 {
268 ok(ret == strlen(attrs[i].str) + 1, "Expected length %d, got %d\n",
269 lstrlenA(attrs[i].str) + 1, ret);
270 ok(!strcmp(buffer, attrs[i].str), "Expected %s, got %s\n",
271 attrs[i].str, buffer);
272 }
273 }
274 blob.pbData = bin8;
275 blob.cbData = sizeof(bin8);
276 ret = pCertRDNValueToStrA(CERT_RDN_UTF8_STRING, &blob, buffer,
277 sizeof(buffer));
278 ok(ret == strlen(ePKI) + 1 || broken(ret != strlen(ePKI) + 1),
279 "Expected length %d, got %d\n", lstrlenA(ePKI), ret);
280 if (ret == strlen(ePKI) + 1)
281 ok(!strcmp(buffer, ePKI), "Expected %s, got %s\n", ePKI, buffer);
282 }
283
284 static void test_CertRDNValueToStrW(void)
285 {
286 static const WCHAR usW[] = { 'U','S',0 };
287 static const WCHAR minnesotaW[] = { 'M','i','n','n','e','s','o','t','a',0 };
288 static const WCHAR minneapolisW[] = { 'M','i','n','n','e','a','p','o','l',
289 'i','s',0 };
290 static const WCHAR codeweaversW[] = { 'C','o','d','e','W','e','a','v','e',
291 'r','s',0 };
292 static const WCHAR wineDevW[] = { 'W','i','n','e',' ','D','e','v','e','l',
293 'o','p','m','e','n','t',0 };
294 static const WCHAR localhostW[] = { 'l','o','c','a','l','h','o','s','t',0 };
295 static const WCHAR aricW[] = { 'a','r','i','c','@','c','o','d','e','w','e',
296 'a','v','e','r','s','.','c','o','m',0 };
297 static const WCHAR ePKIW[] = { 'e','P','K','I',' ','R','o','o','t',' ',
298 'C','e','r','t','i','f','i','c','a','t','i','o','n',' ','A','u','t','h',
299 'o','r','i','t','y',0 };
300 static const WCHAR embeddedDoubleQuoteW[] = { 'a','b','c','"','d','e','f',
301 0 };
302 static const WCHAR embeddedSingleQuoteW[] = { 'a','b','c','\'','d','e','f',
303 0 };
304 static const WCHAR embeddedCommaW[] = { 'a','b','c',',',' ','d','e','f',0 };
305 static const WCHAR trailingAndEndingSpaceW[] = { ' ','a','b','c',' ',0 };
306 static const WCHAR enclosingQuotesW[] = { '"','d','e','f','"',0 };
307 static const WCHAR embeddedSemiW[] = { '1',';','3',0 };
308 CertRDNAttrEncodingW attrs[] = {
309 { "2.5.4.6", CERT_RDN_PRINTABLE_STRING,
310 { sizeof(bin1), bin1 }, usW, FALSE },
311 { "2.5.4.8", CERT_RDN_PRINTABLE_STRING,
312 { sizeof(bin2), bin2 }, minnesotaW, FALSE },
313 { "2.5.4.7", CERT_RDN_PRINTABLE_STRING,
314 { sizeof(bin3), bin3 }, minneapolisW, FALSE },
315 { "2.5.4.10", CERT_RDN_PRINTABLE_STRING,
316 { sizeof(bin4), bin4 }, codeweaversW, FALSE },
317 { "2.5.4.11", CERT_RDN_PRINTABLE_STRING,
318 { sizeof(bin5), bin5 }, wineDevW, FALSE },
319 { "2.5.4.3", CERT_RDN_PRINTABLE_STRING,
320 { sizeof(bin6), bin6 }, localhostW, FALSE },
321 { "1.2.840.113549.1.9.1", CERT_RDN_IA5_STRING,
322 { sizeof(bin7), bin7 }, aricW, FALSE },
323 { "0", CERT_RDN_PRINTABLE_STRING,
324 { sizeof(bin9), bin9 }, embeddedDoubleQuoteW, FALSE },
325 { "0", CERT_RDN_PRINTABLE_STRING,
326 { sizeof(bin10), bin10 }, embeddedSingleQuoteW, FALSE },
327 { "0", CERT_RDN_PRINTABLE_STRING,
328 { sizeof(bin11), bin11 }, embeddedCommaW, FALSE },
329 { "0", CERT_RDN_PRINTABLE_STRING,
330 { sizeof(bin12), bin12 }, trailingAndEndingSpaceW, FALSE },
331 { "0", CERT_RDN_PRINTABLE_STRING,
332 { sizeof(bin13), bin13 }, enclosingQuotesW, FALSE },
333 { "0", CERT_RDN_PRINTABLE_STRING,
334 { sizeof(bin14), bin14 }, embeddedSemiW, FALSE },
335 };
336 DWORD i, ret;
337 WCHAR buffer[2000];
338 CERT_RDN_VALUE_BLOB blob = { 0, NULL };
339
340 if (!pCertRDNValueToStrW)
341 {
342 win_skip("CertRDNValueToStrW is not available\n");
343 return;
344 }
345
346 /* This crashes
347 ret = pCertRDNValueToStrW(0, NULL, NULL, 0);
348 */
349 /* With empty input, it generates the empty string */
350 SetLastError(0xdeadbeef);
351 ret = pCertRDNValueToStrW(0, &blob, NULL, 0);
352 ok(ret == 1 && GetLastError() == 0xdeadbeef, "Expected empty string\n");
353 ret = pCertRDNValueToStrW(0, &blob, buffer,
354 sizeof(buffer) / sizeof(buffer[0]));
355 ok(ret == 1 && GetLastError() == 0xdeadbeef, "Expected empty string\n");
356 ok(!buffer[0], "Expected empty string\n");
357
358 for (i = 0; i < sizeof(attrs) / sizeof(attrs[0]); i++)
359 {
360 ret = pCertRDNValueToStrW(attrs[i].dwValueType, &attrs[i].Value,
361 buffer, sizeof(buffer) / sizeof(buffer[0]));
362 if (attrs[i].todo)
363 {
364 todo_wine
365 ok(ret == lstrlenW(attrs[i].str) + 1,
366 "Expected length %d, got %d\n", lstrlenW(attrs[i].str) + 1, ret);
367 todo_wine
368 ok(!lstrcmpW(buffer, attrs[i].str), "Expected %s, got %s\n",
369 wine_dbgstr_w(attrs[i].str), wine_dbgstr_w(buffer));
370 }
371 else
372 {
373 ok(ret == lstrlenW(attrs[i].str) + 1,
374 "Expected length %d, got %d\n", lstrlenW(attrs[i].str) + 1, ret);
375 ok(!lstrcmpW(buffer, attrs[i].str), "Expected %s, got %s\n",
376 wine_dbgstr_w(attrs[i].str), wine_dbgstr_w(buffer));
377 }
378 }
379 blob.pbData = bin8;
380 blob.cbData = sizeof(bin8);
381 ret = pCertRDNValueToStrW(CERT_RDN_UTF8_STRING, &blob, buffer,
382 sizeof(buffer));
383 ok(ret == lstrlenW(ePKIW) + 1 || broken(ret != lstrlenW(ePKIW) + 1),
384 "Expected length %d, got %d\n", lstrlenW(ePKIW), ret);
385 if (ret == lstrlenW(ePKIW) + 1)
386 ok(!lstrcmpW(buffer, ePKIW), "Expected %s, got %s\n",
387 wine_dbgstr_w(ePKIW), wine_dbgstr_w(buffer));
388 }
389
390 static void test_NameToStrConversionA(PCERT_NAME_BLOB pName, DWORD dwStrType,
391 LPCSTR expected, BOOL todo)
392 {
393 char buffer[2000] = { 0 };
394 DWORD i;
395
396 i = pCertNameToStrA(X509_ASN_ENCODING, pName, dwStrType, NULL, 0);
397 if (todo)
398 todo_wine
399 ok(i == strlen(expected) + 1, "Expected %d chars, got %d\n",
400 lstrlenA(expected) + 1, i);
401 else
402 ok(i == strlen(expected) + 1, "Expected %d chars, got %d\n",
403 lstrlenA(expected) + 1, i);
404 i = pCertNameToStrA(X509_ASN_ENCODING,pName, dwStrType, buffer,
405 sizeof(buffer));
406 if (todo)
407 todo_wine
408 ok(i == strlen(expected) + 1, "Expected %d chars, got %d\n",
409 lstrlenA(expected) + 1, i);
410 else
411 ok(i == strlen(expected) + 1, "Expected %d chars, got %d\n",
412 lstrlenA(expected) + 1, i);
413 if (todo)
414 todo_wine
415 ok(!strcmp(buffer, expected), "Expected %s, got %s\n", expected,
416 buffer);
417 else
418 ok(!strcmp(buffer, expected), "Expected %s, got %s\n", expected,
419 buffer);
420 }
421
422 static BYTE encodedSimpleCN[] = {
423 0x30,0x0c,0x31,0x0a,0x30,0x08,0x06,0x03,0x55,0x04,0x03,0x13,0x01,0x31 };
424 static BYTE encodedSingleQuotedCN[] = { 0x30,0x0e,0x31,0x0c,0x30,0x0a,
425 0x06,0x03,0x55,0x04,0x03,0x13,0x03,0x27,0x31,0x27 };
426 static BYTE encodedSpacedCN[] = { 0x30,0x0e,0x31,0x0c,0x30,0x0a,0x06,0x03,
427 0x55,0x04,0x03,0x13,0x03,0x20,0x31,0x20 };
428 static BYTE encodedQuotedCN[] = { 0x30,0x11,0x31,0x0f,0x30,0x0d,0x06,0x03,
429 0x55, 0x04,0x03,0x1e,0x06,0x00,0x22,0x00,0x31,0x00,0x22, };
430 static BYTE encodedMultipleAttrCN[] = { 0x30,0x0e,0x31,0x0c,0x30,0x0a,
431 0x06,0x03,0x55,0x04,0x03,0x13,0x03,0x31,0x2b,0x32 };
432 static BYTE encodedCommaCN[] = {
433 0x30,0x0e,0x31,0x0c,0x30,0x0a,0x06,0x03,0x55,0x04,0x03,0x13,0x03,0x61,0x2c,
434 0x62 };
435 static BYTE encodedEqualCN[] = {
436 0x30,0x0e,0x31,0x0c,0x30,0x0a,0x06,0x03,0x55,0x04,0x03,0x13,0x03,0x61,0x3d,
437 0x62 };
438 static BYTE encodedLessThanCN[] = {
439 0x30,0x0d,0x31,0x0b,0x30,0x09,0x06,0x03,0x55,0x04,0x03,0x1e,0x02,0x00,0x3c
440 };
441 static BYTE encodedGreaterThanCN[] = {
442 0x30,0x0d,0x31,0x0b,0x30,0x09,0x06,0x03,0x55,0x04,0x03,0x1e,0x02,0x00,0x3e
443 };
444 static BYTE encodedHashCN[] = {
445 0x30,0x0d,0x31,0x0b,0x30,0x09,0x06,0x03,0x55,0x04,0x03,0x1e,0x02,0x00,0x23
446 };
447 static BYTE encodedSemiCN[] = {
448 0x30,0x0d,0x31,0x0b,0x30,0x09,0x06,0x03,0x55,0x04,0x03,0x1e,0x02,0x00,0x3b
449 };
450 static BYTE encodedNewlineCN[] = {
451 0x30,0x11,0x31,0x0f,0x30,0x0d,0x06,0x03,0x55,0x04,0x03,0x1e,0x06,0x00,0x61,
452 0x00,0x0a,0x00,0x62 };
453
454 static void test_CertNameToStrA(void)
455 {
456 PCCERT_CONTEXT context;
457 CERT_NAME_BLOB blob;
458
459 if (!pCertNameToStrA)
460 {
461 win_skip("CertNameToStrA is not available\n");
462 return;
463 }
464
465 context = CertCreateCertificateContext(X509_ASN_ENCODING, cert,
466 sizeof(cert));
467 ok(context != NULL, "CertCreateCertificateContext failed: %08x\n",
468 GetLastError());
469 if (context)
470 {
471 DWORD ret;
472
473 /* This crashes
474 ret = pCertNameToStrA(0, NULL, 0, NULL, 0);
475 */
476 /* Test with a bogus encoding type */
477 SetLastError(0xdeadbeef);
478 ret = pCertNameToStrA(0, &context->pCertInfo->Issuer, 0, NULL, 0);
479 ok(ret == 1 && GetLastError() == ERROR_FILE_NOT_FOUND,
480 "Expected retval 1 and ERROR_FILE_NOT_FOUND, got %d - %08x\n",
481 ret, GetLastError());
482 SetLastError(0xdeadbeef);
483 ret = pCertNameToStrA(X509_ASN_ENCODING, &context->pCertInfo->Issuer,
484 0, NULL, 0);
485 ok(ret && GetLastError() == ERROR_SUCCESS,
486 "Expected positive return and ERROR_SUCCESS, got %d - %08x\n",
487 ret, GetLastError());
488
489 test_NameToStrConversionA(&context->pCertInfo->Issuer,
490 CERT_SIMPLE_NAME_STR, issuerStr, FALSE);
491 test_NameToStrConversionA(&context->pCertInfo->Issuer,
492 CERT_SIMPLE_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG,
493 issuerStrSemicolon, FALSE);
494 test_NameToStrConversionA(&context->pCertInfo->Issuer,
495 CERT_SIMPLE_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
496 issuerStrCRLF, FALSE);
497 test_NameToStrConversionA(&context->pCertInfo->Subject,
498 CERT_OID_NAME_STR, subjectStr, FALSE);
499 test_NameToStrConversionA(&context->pCertInfo->Subject,
500 CERT_OID_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG,
501 subjectStrSemicolon, FALSE);
502 test_NameToStrConversionA(&context->pCertInfo->Subject,
503 CERT_OID_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
504 subjectStrCRLF, FALSE);
505 test_NameToStrConversionA(&context->pCertInfo->Subject,
506 CERT_X500_NAME_STR, x500SubjectStr, FALSE);
507 test_NameToStrConversionA(&context->pCertInfo->Subject,
508 CERT_X500_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG | CERT_NAME_STR_REVERSE_FLAG,
509 x500SubjectStrSemicolonReverse, FALSE);
510
511 CertFreeCertificateContext(context);
512 }
513 blob.pbData = encodedSimpleCN;
514 blob.cbData = sizeof(encodedSimpleCN);
515 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=1", FALSE);
516 blob.pbData = encodedSingleQuotedCN;
517 blob.cbData = sizeof(encodedSingleQuotedCN);
518 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN='1'", FALSE);
519 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "'1'", FALSE);
520 blob.pbData = encodedSpacedCN;
521 blob.cbData = sizeof(encodedSpacedCN);
522 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\" 1 \"", FALSE);
523 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\" 1 \"", FALSE);
524 blob.pbData = encodedQuotedCN;
525 blob.cbData = sizeof(encodedQuotedCN);
526 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"\"\"1\"\"\"",
527 FALSE);
528 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"\"\"1\"\"\"",
529 FALSE);
530 blob.pbData = encodedMultipleAttrCN;
531 blob.cbData = sizeof(encodedMultipleAttrCN);
532 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"1+2\"", FALSE);
533 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"1+2\"", FALSE);
534 blob.pbData = encodedCommaCN;
535 blob.cbData = sizeof(encodedCommaCN);
536 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"a,b\"", FALSE);
537 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"a,b\"", FALSE);
538 blob.pbData = encodedEqualCN;
539 blob.cbData = sizeof(encodedEqualCN);
540 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"a=b\"", FALSE);
541 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"a=b\"", FALSE);
542 blob.pbData = encodedLessThanCN;
543 blob.cbData = sizeof(encodedLessThanCN);
544 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"<\"", FALSE);
545 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"<\"", FALSE);
546 blob.pbData = encodedGreaterThanCN;
547 blob.cbData = sizeof(encodedGreaterThanCN);
548 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\">\"", FALSE);
549 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\">\"", FALSE);
550 blob.pbData = encodedHashCN;
551 blob.cbData = sizeof(encodedHashCN);
552 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"#\"", FALSE);
553 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"#\"", FALSE);
554 blob.pbData = encodedSemiCN;
555 blob.cbData = sizeof(encodedSemiCN);
556 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\";\"", FALSE);
557 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\";\"", FALSE);
558 blob.pbData = encodedNewlineCN;
559 blob.cbData = sizeof(encodedNewlineCN);
560 test_NameToStrConversionA(&blob, CERT_X500_NAME_STR, "CN=\"a\nb\"", FALSE);
561 test_NameToStrConversionA(&blob, CERT_SIMPLE_NAME_STR, "\"a\nb\"", FALSE);
562 }
563
564 static void test_NameToStrConversionW(PCERT_NAME_BLOB pName, DWORD dwStrType,
565 LPCWSTR expected, BOOL todo)
566 {
567 WCHAR buffer[2000] = { 0 };
568 DWORD i;
569
570 i = pCertNameToStrW(X509_ASN_ENCODING,pName, dwStrType, NULL, 0);
571 if (todo)
572 todo_wine ok(i == lstrlenW(expected) + 1, "Expected %d chars, got %d\n",
573 lstrlenW(expected) + 1, i);
574 else
575 ok(i == lstrlenW(expected) + 1, "Expected %d chars, got %d\n",
576 lstrlenW(expected) + 1, i);
577 i = pCertNameToStrW(X509_ASN_ENCODING,pName, dwStrType, buffer,
578 sizeof(buffer) / sizeof(buffer[0]));
579 if (todo)
580 todo_wine ok(i == lstrlenW(expected) + 1, "Expected %d chars, got %d\n",
581 lstrlenW(expected) + 1, i);
582 else
583 ok(i == lstrlenW(expected) + 1, "Expected %d chars, got %d\n",
584 lstrlenW(expected) + 1, i);
585 if (todo)
586 todo_wine ok(!lstrcmpW(buffer, expected), "Expected %s, got %s\n",
587 wine_dbgstr_w(expected), wine_dbgstr_w(buffer));
588 else
589 ok(!lstrcmpW(buffer, expected), "Expected %s, got %s\n",
590 wine_dbgstr_w(expected), wine_dbgstr_w(buffer));
591 }
592
593 static const WCHAR simpleCN_W[] = { 'C','N','=','1',0 };
594 static const WCHAR singledQuotedCN_W[] = { 'C','N','=','\'','1','\'',0 };
595 static const WCHAR simpleSingleQuotedCN_W[] = { '\'','1','\'',0 };
596 static const WCHAR spacedCN_W[] = { 'C','N','=','"',' ','1',' ','"',0 };
597 static const WCHAR simpleSpacedCN_W[] = { '"',' ','1',' ','"',0 };
598 static const WCHAR quotedCN_W[] = { 'C','N','=','"','"','"','1','"','"','"',0 };
599 static const WCHAR simpleQuotedCN_W[] = { '"','"','"','1','"','"','"',0 };
600 static const WCHAR multipleAttrCN_W[] = { 'C','N','=','"','1','+','2','"',0 };
601 static const WCHAR simpleMultipleAttrCN_W[] = { '"','1','+','2','"',0 };
602 static const WCHAR commaCN_W[] = { 'C','N','=','"','a',',','b','"',0 };
603 static const WCHAR simpleCommaCN_W[] = { '"','a',',','b','"',0 };
604 static const WCHAR equalCN_W[] = { 'C','N','=','"','a','=','b','"',0 };
605 static const WCHAR simpleEqualCN_W[] = { '"','a','=','b','"',0 };
606 static const WCHAR lessThanCN_W[] = { 'C','N','=','"','<','"',0 };
607 static const WCHAR simpleLessThanCN_W[] = { '"','<','"',0 };
608 static const WCHAR greaterThanCN_W[] = { 'C','N','=','"','>','"',0 };
609 static const WCHAR simpleGreaterThanCN_W[] = { '"','>','"',0 };
610 static const WCHAR hashCN_W[] = { 'C','N','=','"','#','"',0 };
611 static const WCHAR simpleHashCN_W[] = { '"','#','"',0 };
612 static const WCHAR semiCN_W[] = { 'C','N','=','"',';','"',0 };
613 static const WCHAR simpleSemiCN_W[] = { '"',';','"',0 };
614 static const WCHAR newlineCN_W[] = { 'C','N','=','"','a','\n','b','"',0 };
615 static const WCHAR simpleNewlineCN_W[] = { '"','a','\n','b','"',0 };
616
617 static void test_CertNameToStrW(void)
618 {
619 PCCERT_CONTEXT context;
620 CERT_NAME_BLOB blob;
621
622 if (!pCertNameToStrW)
623 {
624 win_skip("CertNameToStrW is not available\n");
625 return;
626 }
627
628 context = CertCreateCertificateContext(X509_ASN_ENCODING, cert,
629 sizeof(cert));
630 ok(context != NULL, "CertCreateCertificateContext failed: %08x\n",
631 GetLastError());
632 if (context)
633 {
634 DWORD ret;
635
636 /* This crashes
637 ret = pCertNameToStrW(0, NULL, 0, NULL, 0);
638 */
639 /* Test with a bogus encoding type */
640 SetLastError(0xdeadbeef);
641 ret = pCertNameToStrW(0, &context->pCertInfo->Issuer, 0, NULL, 0);
642 ok(ret == 1 && GetLastError() == ERROR_FILE_NOT_FOUND,
643 "Expected retval 1 and ERROR_FILE_NOT_FOUND, got %d - %08x\n",
644 ret, GetLastError());
645 SetLastError(0xdeadbeef);
646 ret = pCertNameToStrW(X509_ASN_ENCODING, &context->pCertInfo->Issuer,
647 0, NULL, 0);
648 ok(ret && GetLastError() == ERROR_SUCCESS,
649 "Expected positive return and ERROR_SUCCESS, got %d - %08x\n",
650 ret, GetLastError());
651
652 test_NameToStrConversionW(&context->pCertInfo->Issuer,
653 CERT_SIMPLE_NAME_STR, issuerStrW, FALSE);
654 test_NameToStrConversionW(&context->pCertInfo->Issuer,
655 CERT_SIMPLE_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG,
656 issuerStrSemicolonW, FALSE);
657 test_NameToStrConversionW(&context->pCertInfo->Issuer,
658 CERT_SIMPLE_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
659 issuerStrCRLFW, FALSE);
660 test_NameToStrConversionW(&context->pCertInfo->Subject,
661 CERT_OID_NAME_STR, subjectStrW, FALSE);
662 test_NameToStrConversionW(&context->pCertInfo->Subject,
663 CERT_OID_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG,
664 subjectStrSemicolonW, FALSE);
665 test_NameToStrConversionW(&context->pCertInfo->Subject,
666 CERT_OID_NAME_STR | CERT_NAME_STR_CRLF_FLAG,
667 subjectStrCRLFW, FALSE);
668 test_NameToStrConversionW(&context->pCertInfo->Subject,
669 CERT_X500_NAME_STR | CERT_NAME_STR_SEMICOLON_FLAG | CERT_NAME_STR_REVERSE_FLAG,
670 x500SubjectStrSemicolonReverseW, FALSE);
671
672 CertFreeCertificateContext(context);
673 }
674 blob.pbData = encodedSimpleCN;
675 blob.cbData = sizeof(encodedSimpleCN);
676 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, simpleCN_W, FALSE);
677 blob.pbData = encodedSingleQuotedCN;
678 blob.cbData = sizeof(encodedSingleQuotedCN);
679 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, singledQuotedCN_W,
680 FALSE);
681 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR,
682 simpleSingleQuotedCN_W, FALSE);
683 blob.pbData = encodedSpacedCN;
684 blob.cbData = sizeof(encodedSpacedCN);
685 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, spacedCN_W, FALSE);
686 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleSpacedCN_W,
687 FALSE);
688 blob.pbData = encodedQuotedCN;
689 blob.cbData = sizeof(encodedQuotedCN);
690 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, quotedCN_W,
691 FALSE);
692 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleQuotedCN_W,
693 FALSE);
694 blob.pbData = encodedMultipleAttrCN;
695 blob.cbData = sizeof(encodedMultipleAttrCN);
696 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, multipleAttrCN_W,
697 FALSE);
698 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR,
699 simpleMultipleAttrCN_W, FALSE);
700 blob.pbData = encodedCommaCN;
701 blob.cbData = sizeof(encodedCommaCN);
702 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, commaCN_W, FALSE);
703 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleCommaCN_W,
704 FALSE);
705 blob.pbData = encodedEqualCN;
706 blob.cbData = sizeof(encodedEqualCN);
707 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, equalCN_W, FALSE);
708 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleEqualCN_W,
709 FALSE);
710 blob.pbData = encodedLessThanCN;
711 blob.cbData = sizeof(encodedLessThanCN);
712 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, lessThanCN_W, FALSE);
713 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleLessThanCN_W,
714 FALSE);
715 blob.pbData = encodedGreaterThanCN;
716 blob.cbData = sizeof(encodedGreaterThanCN);
717 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, greaterThanCN_W,
718 FALSE);
719 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR,
720 simpleGreaterThanCN_W, FALSE);
721 blob.pbData = encodedHashCN;
722 blob.cbData = sizeof(encodedHashCN);
723 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, hashCN_W, FALSE);
724 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleHashCN_W,
725 FALSE);
726 blob.pbData = encodedSemiCN;
727 blob.cbData = sizeof(encodedSemiCN);
728 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, semiCN_W, FALSE);
729 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleSemiCN_W,
730 FALSE);
731 blob.pbData = encodedNewlineCN;
732 blob.cbData = sizeof(encodedNewlineCN);
733 test_NameToStrConversionW(&blob, CERT_X500_NAME_STR, newlineCN_W, FALSE);
734 test_NameToStrConversionW(&blob, CERT_SIMPLE_NAME_STR, simpleNewlineCN_W,
735 FALSE);
736 }
737
738 struct StrToNameA
739 {
740 LPCSTR x500;
741 DWORD encodedSize;
742 const BYTE *encoded;
743 };
744
745 static const struct StrToNameA namesA[] = {
746 { "CN=1", sizeof(encodedSimpleCN), encodedSimpleCN },
747 { "CN=\"1\"", sizeof(encodedSimpleCN), encodedSimpleCN },
748 { "CN = \"1\"", sizeof(encodedSimpleCN), encodedSimpleCN },
749 { "CN='1'", sizeof(encodedSingleQuotedCN), encodedSingleQuotedCN },
750 { "CN=\" 1 \"", sizeof(encodedSpacedCN), encodedSpacedCN },
751 { "CN=\"\"\"1\"\"\"", sizeof(encodedQuotedCN), encodedQuotedCN },
752 { "CN=\"1+2\"", sizeof(encodedMultipleAttrCN), encodedMultipleAttrCN },
753 { "CN=\"a,b\"", sizeof(encodedCommaCN), encodedCommaCN },
754 { "CN=\"a=b\"", sizeof(encodedEqualCN), encodedEqualCN },
755 { "CN=\"<\"", sizeof(encodedLessThanCN), encodedLessThanCN },
756 { "CN=\">\"", sizeof(encodedGreaterThanCN), encodedGreaterThanCN },
757 { "CN=\"#\"", sizeof(encodedHashCN), encodedHashCN },
758 { "CN=\";\"", sizeof(encodedSemiCN), encodedSemiCN },
759 };
760
761 static void test_CertStrToNameA(void)
762 {
763 BOOL ret;
764 DWORD size, i;
765 BYTE buf[100];
766
767 if (!pCertStrToNameA)
768 {
769 win_skip("CertStrToNameA is not available\n");
770 return;
771 }
772
773 /* Crash
774 ret = pCertStrToNameA(0, NULL, 0, NULL, NULL, NULL, NULL);
775 */
776 ret = pCertStrToNameA(0, NULL, 0, NULL, NULL, &size, NULL);
777 ok(!ret, "Expected failure\n");
778 ret = pCertStrToNameA(0, "bogus", 0, NULL, NULL, &size, NULL);
779 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
780 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
781 ret = pCertStrToNameA(0, "foo=1", 0, NULL, NULL, &size, NULL);
782 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
783 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
784 ret = pCertStrToNameA(0, "CN=1", 0, NULL, NULL, &size, NULL);
785 ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND,
786 "Expected ERROR_FILE_NOT_FOUND, got %08x\n", GetLastError());
787 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=1", 0, NULL, NULL, &size, NULL);
788 ok(ret, "CertStrToNameA failed: %08x\n", GetLastError());
789 size = sizeof(buf);
790 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=\"\"1\"\"", 0, NULL, buf, &size,
791 NULL);
792 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
793 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
794 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=1+2", 0, NULL, buf,
795 &size, NULL);
796 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
797 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
798 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=1+2", CERT_NAME_STR_NO_PLUS_FLAG, NULL, buf,
799 &size, NULL);
800 ok(ret && GetLastError() == ERROR_SUCCESS,
801 "Expected ERROR_SUCCESS, got %08x\n", GetLastError());
802 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=1,2", CERT_NAME_STR_NO_QUOTING_FLAG, NULL, buf,
803 &size, NULL);
804 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
805 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
806 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=\"1,2;3,4\"", CERT_NAME_STR_NO_QUOTING_FLAG, NULL, buf,
807 &size, NULL);
808 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
809 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
810 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=abc", 0, NULL, buf,
811 &size, NULL);
812 ok(ret && GetLastError() == ERROR_SUCCESS,
813 "Expected ERROR_SUCCESS, got %08x\n", GetLastError());
814 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=abc", CERT_NAME_STR_NO_QUOTING_FLAG, NULL, buf,
815 &size, NULL);
816 ok(ret && GetLastError() == ERROR_SUCCESS,
817 "Expected ERROR_SUCCESS, got %08x\n", GetLastError());
818 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=\"abc\"", 0, NULL, buf,
819 &size, NULL);
820 ok(ret && GetLastError() == ERROR_SUCCESS,
821 "Expected ERROR_SUCCESS, got %08x\n", GetLastError());
822 ret = pCertStrToNameA(X509_ASN_ENCODING, "CN=\"abc\"", CERT_NAME_STR_NO_QUOTING_FLAG, NULL, buf,
823 &size, NULL);
824 ok(!ret && GetLastError() == ERROR_MORE_DATA,
825 "Expected ERROR_MORE_DATA, got %08x\n", GetLastError());
826 for (i = 0; i < sizeof(namesA) / sizeof(namesA[0]); i++)
827 {
828 size = sizeof(buf);
829 ret = pCertStrToNameA(X509_ASN_ENCODING, namesA[i].x500, 0, NULL, buf,
830 &size, NULL);
831 ok(ret, "CertStrToNameA failed on string %s: %08x\n", namesA[i].x500,
832 GetLastError());
833 ok(size == namesA[i].encodedSize,
834 "Expected size %d, got %d\n", namesA[i].encodedSize, size);
835 if (ret)
836 ok(!memcmp(buf, namesA[i].encoded, namesA[i].encodedSize),
837 "Unexpected value for string %s\n", namesA[i].x500);
838 }
839 }
840
841 struct StrToNameW
842 {
843 LPCWSTR x500;
844 DWORD encodedSize;
845 const BYTE *encoded;
846 };
847
848 static const WCHAR badlyQuotedCN_W[] = { 'C','N','=','"','"','1','"','"',0 };
849 static const WCHAR simpleCN2_W[] = { 'C','N','=','"','1','"',0 };
850 static const WCHAR simpleCN3_W[] = { 'C','N',' ','=',' ','"','1','"',0 };
851 static const WCHAR japaneseCN_W[] = { 'C','N','=',0x226f,0x575b,0 };
852 static const BYTE encodedJapaneseCN[] = { 0x30,0x0f,0x31,0x0d,0x30,0x0b,0x06,
853 0x03,0x55,0x04,0x03,0x1e,0x04,0x22,0x6f,0x57,0x5b };
854
855 static const struct StrToNameW namesW[] = {
856 { simpleCN_W, sizeof(encodedSimpleCN), encodedSimpleCN },
857 { simpleCN2_W, sizeof(encodedSimpleCN), encodedSimpleCN },
858 { simpleCN3_W, sizeof(encodedSimpleCN), encodedSimpleCN },
859 { singledQuotedCN_W, sizeof(encodedSingleQuotedCN), encodedSingleQuotedCN },
860 { spacedCN_W, sizeof(encodedSpacedCN), encodedSpacedCN },
861 { quotedCN_W, sizeof(encodedQuotedCN), encodedQuotedCN },
862 { multipleAttrCN_W, sizeof(encodedMultipleAttrCN), encodedMultipleAttrCN },
863 { japaneseCN_W, sizeof(encodedJapaneseCN), encodedJapaneseCN },
864 { commaCN_W, sizeof(encodedCommaCN), encodedCommaCN },
865 { equalCN_W, sizeof(encodedEqualCN), encodedEqualCN },
866 { lessThanCN_W, sizeof(encodedLessThanCN), encodedLessThanCN },
867 { greaterThanCN_W, sizeof(encodedGreaterThanCN), encodedGreaterThanCN },
868 { hashCN_W, sizeof(encodedHashCN), encodedHashCN },
869 { semiCN_W, sizeof(encodedSemiCN), encodedSemiCN },
870 };
871
872 static void test_CertStrToNameW(void)
873 {
874 static const WCHAR bogusW[] = { 'b','o','g','u','s',0 };
875 static const WCHAR fooW[] = { 'f','o','o','=','1',0 };
876 BOOL ret;
877 DWORD size, i;
878 LPCWSTR errorPtr;
879 BYTE buf[100];
880
881 if (!pCertStrToNameW)
882 {
883 win_skip("CertStrToNameW is not available\n");
884 return;
885 }
886
887 /* Crash
888 ret = pCertStrToNameW(0, NULL, 0, NULL, NULL, NULL, NULL);
889 */
890 ret = pCertStrToNameW(0, NULL, 0, NULL, NULL, &size, NULL);
891 ok(!ret, "Expected failure\n");
892 ret = pCertStrToNameW(0, bogusW, 0, NULL, NULL, &size, NULL);
893 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
894 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
895 ret = pCertStrToNameW(0, fooW, 0, NULL, NULL, &size, NULL);
896 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
897 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
898 ret = pCertStrToNameW(0, simpleCN_W, 0, NULL, NULL, &size, NULL);
899 ok(!ret && GetLastError() == ERROR_FILE_NOT_FOUND,
900 "Expected ERROR_FILE_NOT_FOUND, got %08x\n", GetLastError());
901 ret = pCertStrToNameW(X509_ASN_ENCODING, simpleCN_W, 0, NULL, NULL, &size,
902 NULL);
903 ok(ret, "CertStrToNameW failed: %08x\n", GetLastError());
904 size = sizeof(buf);
905 ret = pCertStrToNameW(X509_ASN_ENCODING, badlyQuotedCN_W, 0, NULL, buf,
906 &size, NULL);
907 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
908 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
909 ret = pCertStrToNameW(X509_ASN_ENCODING, badlyQuotedCN_W, 0, NULL, buf,
910 &size, &errorPtr);
911 ok(!ret && GetLastError() == CRYPT_E_INVALID_X500_STRING,
912 "Expected CRYPT_E_INVALID_X500_STRING, got %08x\n", GetLastError());
913 ok(errorPtr && *errorPtr == '1', "Expected first error character was 1\n");
914 for (i = 0; i < sizeof(namesW) / sizeof(namesW[0]); i++)
915 {
916 size = sizeof(buf);
917 ret = pCertStrToNameW(X509_ASN_ENCODING, namesW[i].x500, 0, NULL, buf,
918 &size, NULL);
919 ok(ret, "Index %d: CertStrToNameW failed: %08x\n", i, GetLastError());
920 ok(size == namesW[i].encodedSize,
921 "Index %d: expected size %d, got %d\n", i, namesW[i].encodedSize,
922 size);
923 if (ret)
924 ok(!memcmp(buf, namesW[i].encoded, size),
925 "Index %d: unexpected value\n", i);
926 }
927 }
928
929 static void test_CertGetNameStringA(void)
930 {
931 PCCERT_CONTEXT context;
932
933 if (!pCertGetNameStringA)
934 {
935 win_skip("CertGetNameStringA is not available\n");
936 return;
937 }
938
939 context = CertCreateCertificateContext(X509_ASN_ENCODING, cert,
940 sizeof(cert));
941 ok(context != NULL, "CertCreateCertificateContext failed: %08x\n",
942 GetLastError());
943 if (context)
944 {
945 static const char aric[] = "aric@codeweavers.com";
946 static const char localhost[] = "localhost";
947 DWORD len, type;
948 LPSTR str;
949
950 /* Bad string types/types missing from the cert */
951 len = pCertGetNameStringA(NULL, 0, 0, NULL, NULL, 0);
952 ok(len == 1, "expected 1, got %d\n", len);
953 len = pCertGetNameStringA(context, 0, 0, NULL, NULL, 0);
954 ok(len == 1, "expected 1, got %d\n", len);
955 len = pCertGetNameStringA(context, CERT_NAME_URL_TYPE, 0, NULL, NULL,
956 0);
957 ok(len == 1, "expected 1, got %d\n", len);
958
959 len = pCertGetNameStringA(context, CERT_NAME_EMAIL_TYPE, 0, NULL, NULL,
960 0);
961 ok(len == strlen(aric) + 1, "unexpected length %d\n", len);
962 str = HeapAlloc(GetProcessHeap(), 0, len);
963 if (str)
964 {
965 len = pCertGetNameStringA(context, CERT_NAME_EMAIL_TYPE, 0, NULL,
966 str, len);
967 ok(!strcmp(str, aric), "unexpected value %s\n", str);
968 HeapFree(GetProcessHeap(), 0, str);
969 }
970
971 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, NULL, NULL,
972 0);
973 ok(len == strlen(issuerStr) + 1, "unexpected length %d\n", len);
974 str = HeapAlloc(GetProcessHeap(), 0, len);
975 if (str)
976 {
977 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, NULL,
978 str, len);
979 ok(!strcmp(str, issuerStr), "unexpected value %s\n", str);
980 HeapFree(GetProcessHeap(), 0, str);
981 }
982 type = 0;
983 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, &type, NULL,
984 0);
985 ok(len == strlen(issuerStr) + 1, "unexpected length %d\n", len);
986 str = HeapAlloc(GetProcessHeap(), 0, len);
987 if (str)
988 {
989 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, &type,
990 str, len);
991 ok(!strcmp(str, issuerStr), "unexpected value %s\n", str);
992 HeapFree(GetProcessHeap(), 0, str);
993 }
994 type = CERT_OID_NAME_STR;
995 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, &type, NULL,
996 0);
997 ok(len == strlen(subjectStr) + 1, "unexpected length %d\n", len);
998 str = HeapAlloc(GetProcessHeap(), 0, len);
999 if (str)
1000 {
1001 len = pCertGetNameStringA(context, CERT_NAME_RDN_TYPE, 0, &type,
1002 str, len);
1003 ok(!strcmp(str, subjectStr), "unexpected value %s\n", str);
1004 HeapFree(GetProcessHeap(), 0, str);
1005 }
1006
1007 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0, NULL, NULL,
1008 0);
1009 ok(len == strlen(aric) + 1, "unexpected length %d\n", len);
1010 str = HeapAlloc(GetProcessHeap(), 0, len);
1011 if (str)
1012 {
1013 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0, NULL,
1014 str, len);
1015 ok(!strcmp(str, aric), "unexpected value %s\n", str);
1016 HeapFree(GetProcessHeap(), 0, str);
1017 }
1018 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0,
1019 (void *)szOID_RSA_emailAddr, NULL, 0);
1020 ok(len == strlen(aric) + 1, "unexpected length %d\n", len);
1021 str = HeapAlloc(GetProcessHeap(), 0, len);
1022 if (str)
1023 {
1024 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0,
1025 (void *)szOID_RSA_emailAddr, str, len);
1026 ok(!strcmp(str, aric), "unexpected value %s\n", str);
1027 HeapFree(GetProcessHeap(), 0, str);
1028 }
1029 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0,
1030 (void *)szOID_COMMON_NAME, NULL, 0);
1031 ok(len == strlen(localhost) + 1, "unexpected length %d\n", len);
1032 str = HeapAlloc(GetProcessHeap(), 0, len);
1033 if (str)
1034 {
1035 len = pCertGetNameStringA(context, CERT_NAME_ATTR_TYPE, 0,
1036 (void *)szOID_COMMON_NAME, str, len);
1037 ok(!strcmp(str, localhost), "unexpected value %s\n", str);
1038 HeapFree(GetProcessHeap(), 0, str);
1039 }
1040
1041 len = pCertGetNameStringA(context, CERT_NAME_SIMPLE_DISPLAY_TYPE, 0,
1042 NULL, NULL, 0);
1043 ok(len == strlen(localhost) + 1, "unexpected length %d\n", len);
1044 str = HeapAlloc(GetProcessHeap(), 0, len);
1045 if (str)
1046 {
1047 len = pCertGetNameStringA(context, CERT_NAME_SIMPLE_DISPLAY_TYPE,
1048 0, NULL, str, len);
1049 ok(!strcmp(str, localhost), "unexpected value %s\n", str);
1050 HeapFree(GetProcessHeap(), 0, str);
1051 }
1052
1053 len = pCertGetNameStringA(context, CERT_NAME_FRIENDLY_DISPLAY_TYPE, 0,
1054 NULL, NULL, 0);
1055 ok(len == strlen(localhost) + 1, "unexpected length %d\n", len);
1056 str = HeapAlloc(GetProcessHeap(), 0, len);
1057 if (str)
1058 {
1059 len = pCertGetNameStringA(context, CERT_NAME_FRIENDLY_DISPLAY_TYPE,
1060 0, NULL, str, len);
1061 ok(!strcmp(str, localhost), "unexpected value %s\n", str);
1062 HeapFree(GetProcessHeap(), 0, str);
1063 }
1064
1065 len = pCertGetNameStringA(context, CERT_NAME_DNS_TYPE, 0, NULL, NULL,
1066 0);
1067 ok(len == strlen(localhost) + 1 || broken(len == 1) /* NT4 */,
1068 "unexpected length %d\n", len);
1069 if (len > 1)
1070 {
1071 str = HeapAlloc(GetProcessHeap(), 0, len);
1072 if (str)
1073 {
1074 len = pCertGetNameStringA(context, CERT_NAME_DNS_TYPE, 0, NULL,
1075 str, len);
1076 ok(!strcmp(str, localhost), "unexpected value %s\n", str);
1077 HeapFree(GetProcessHeap(), 0, str);
1078 }
1079 }
1080
1081 CertFreeCertificateContext(context);
1082 }
1083 }
1084
1085 START_TEST(str)
1086 {
1087 dll = GetModuleHandleA("Crypt32.dll");
1088
1089 pCertNameToStrA = (void*)GetProcAddress(dll,"CertNameToStrA");
1090 pCertNameToStrW = (void*)GetProcAddress(dll,"CertNameToStrW");
1091 pCertRDNValueToStrA = (void*)GetProcAddress(dll, "CertRDNValueToStrA");
1092 pCertRDNValueToStrW = (void*)GetProcAddress(dll, "CertRDNValueToStrW");
1093 pCertStrToNameA = (void*)GetProcAddress(dll,"CertStrToNameA");
1094 pCertStrToNameW = (void*)GetProcAddress(dll,"CertStrToNameW");
1095 pCertGetNameStringA = (void*)GetProcAddress(dll, "CertGetNameStringA");
1096
1097 test_CertRDNValueToStrA();
1098 test_CertRDNValueToStrW();
1099 test_CertNameToStrA();
1100 test_CertNameToStrW();
1101 test_CertStrToNameA();
1102 test_CertStrToNameW();
1103 test_CertGetNameStringA();
1104 }