4 * Copyright 2006 Yuval Fledel
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.
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.
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
24 #define WIN32_NO_STATUS
27 #define SECURITY_WIN32
33 #include "wine/test.h"
35 /* Helper macros to find the size of SECPKG_FUNCTION_TABLE */
36 #define SECPKG_FUNCTION_TABLE_SIZE_1 FIELD_OFFSET(SECPKG_FUNCTION_TABLE, \
38 #define SECPKG_FUNCTION_TABLE_SIZE_2 FIELD_OFFSET(SECPKG_FUNCTION_TABLE, \
39 SetCredentialsAttributes)
40 #define SECPKG_FUNCTION_TABLE_SIZE_3 FIELD_OFFSET(SECPKG_FUNCTION_TABLE, \
41 ChangeAccountPassword)
42 #define SECPKG_FUNCTION_TABLE_SIZE_4 FIELD_OFFSET(SECPKG_FUNCTION_TABLE, \
44 #define SECPKG_FUNCTION_TABLE_SIZE_5 FIELD_OFFSET(SECPKG_FUNCTION_TABLE, \
46 #define SECPKG_FUNCTION_TABLE_SIZE_6 sizeof(SECPKG_FUNCTION_TABLE)
48 static NTSTATUS (NTAPI *pSpLsaModeInitialize)(ULONG, PULONG,
49 PSECPKG_FUNCTION_TABLE*, PULONG);
50 static NTSTATUS (NTAPI *pSpUserModeInitialize)(ULONG, PULONG,
51 PSECPKG_USER_FUNCTION_TABLE*, PULONG);
53 static void testInitialize(void)
55 PSECPKG_USER_FUNCTION_TABLE pUserTables, pUserTables2;
56 PSECPKG_FUNCTION_TABLE pTables, pTables2;
57 ULONG cTables = 0, cUserTables = 0, Version = 0;
60 /* Passing NULL into one of the parameters of SpLsaModeInitialize or
61 SpUserModeInitialize causes a crash. */
63 /* SpLsaModeInitialize does not care about the LSA version. */
64 status = pSpLsaModeInitialize(0, &Version, &pTables2, &cTables);
65 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
67 broken(cTables == 1), /* Win2k */
68 "cTables: %d\n", cTables);
69 ok(pTables2 != NULL,"pTables: %p\n", pTables2);
71 /* We can call it as many times we want. */
72 status = pSpLsaModeInitialize(0x10000, &Version, &pTables, &cTables);
73 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
75 broken(cTables == 1), /* Win2k */
76 "cTables: %d\n", cTables);
77 ok(pTables != NULL, "pTables: %p\n", pTables);
78 /* It will always return the same pointer. */
79 ok(pTables == pTables2, "pTables: %p, pTables2: %p\n", pTables, pTables2);
81 status = pSpLsaModeInitialize(0x23456, &Version, &pTables, &cTables);
82 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
84 broken(cTables == 1), /* Win2k */
85 "cTables: %d\n", cTables);
86 ok(pTables != NULL, "pTables: %p\n", pTables);
87 ok(pTables == pTables2, "pTables: %p, pTables2: %p\n", pTables, pTables2);
89 /* Bad versions to SpUserModeInitialize. Parameters unchanged */
93 status = pSpUserModeInitialize(0, &Version, &pUserTables, &cUserTables);
94 ok(status == STATUS_INVALID_PARAMETER, "status: 0x%x\n", status);
95 ok(Version == 0xdead, "Version: 0x%x\n", Version);
96 ok(cUserTables == 0xdead, "cTables: %d\n", cUserTables);
97 ok(pUserTables == NULL, "pUserTables: %p\n", pUserTables);
99 status = pSpUserModeInitialize(0x20000, &Version, &pUserTables,
101 ok(status == STATUS_INVALID_PARAMETER, "status: 0x%x\n", status);
102 ok(Version == 0xdead, "Version: 0x%x\n", Version);
103 ok(cUserTables == 0xdead, "cTables: %d\n", cUserTables);
104 ok(pUserTables == NULL, "pUserTables: %p\n", pUserTables);
106 /* Good version to SpUserModeInitialize */
107 status = pSpUserModeInitialize(SECPKG_INTERFACE_VERSION, &Version,
108 &pUserTables, &cUserTables);
109 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
110 ok(Version == SECPKG_INTERFACE_VERSION, "Version: 0x%x\n", Version);
111 ok(cUserTables == 2 ||
112 broken(cUserTables == 4), /* Win2k */
113 "cUserTables: %d\n", cUserTables);
114 ok(pUserTables != NULL, "pUserTables: %p\n", pUserTables);
116 /* Initializing user again */
117 status = pSpUserModeInitialize(SECPKG_INTERFACE_VERSION, &Version,
118 &pUserTables2, &cTables);
119 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
120 ok(pUserTables == pUserTables2, "pUserTables: %p, pUserTables2: %p\n",
121 pUserTables, pUserTables2);
124 /* A helper function to find the dispatch table of the next package.
125 Needed because SECPKG_FUNCTION_TABLE's size depend on the version */
126 static PSECPKG_FUNCTION_TABLE getNextSecPkgTable(PSECPKG_FUNCTION_TABLE pTable,
130 PSECPKG_FUNCTION_TABLE pNextTable;
132 if (Version == SECPKG_INTERFACE_VERSION)
133 size = SECPKG_FUNCTION_TABLE_SIZE_1;
134 else if (Version == SECPKG_INTERFACE_VERSION_2)
135 size = SECPKG_FUNCTION_TABLE_SIZE_2;
136 else if (Version == SECPKG_INTERFACE_VERSION_3)
137 size = SECPKG_FUNCTION_TABLE_SIZE_3;
138 else if (Version == SECPKG_INTERFACE_VERSION_4)
139 size = SECPKG_FUNCTION_TABLE_SIZE_4;
140 else if (Version == SECPKG_INTERFACE_VERSION_5)
141 size = SECPKG_FUNCTION_TABLE_SIZE_5;
142 else if (Version == SECPKG_INTERFACE_VERSION_6)
143 size = SECPKG_FUNCTION_TABLE_SIZE_6;
145 ok(FALSE, "Unknown package version 0x%x\n", Version);
149 pNextTable = (PSECPKG_FUNCTION_TABLE)((PBYTE)pTable + size);
150 /* Win7 function tables appear to be SECPKG_INTERFACE_VERSION_6 format,
151 but unfortunately SpLsaModeInitialize returns SECPKG_INTERFACE_VERSION_3.
152 We detect that by comparing the "Initialize" pointer from the old table
153 to the "FreeCredentialsHandle" pointer of the new table. These functions
154 have different numbers of arguments, so they can't possibly point to the
155 same implementation */
156 if (broken((void *) pTable->Initialize == (void *) pNextTable->FreeCredentialsHandle &&
157 pNextTable->FreeCredentialsHandle != NULL))
159 win_skip("Invalid function pointers for next package\n");
166 static void testGetInfo(void)
168 PSECPKG_FUNCTION_TABLE pTables;
169 SecPkgInfoW PackageInfo;
170 ULONG cTables, Version;
173 /* Get the dispatch table */
174 status = pSpLsaModeInitialize(0, &Version, &pTables, &cTables);
175 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
177 /* Passing NULL into ->GetInfo causes a crash. */
179 /* First package: Unified */
180 status = pTables->GetInfo(&PackageInfo);
181 ok(status == STATUS_SUCCESS, "status: 0x%x\n", status);
182 ok(PackageInfo.fCapabilities == 0x107b3, "fCapabilities: 0x%x\n",
183 PackageInfo.fCapabilities);
184 ok(PackageInfo.wVersion == 1, "wVersion: %d\n", PackageInfo.wVersion);
185 ok(PackageInfo.wRPCID == 14, "wRPCID: %d\n", PackageInfo.wRPCID);
186 ok(PackageInfo.cbMaxToken == 0x4000 ||
187 PackageInfo.cbMaxToken == 0x6000, /* Vista */
188 "cbMaxToken: 0x%x\n",
189 PackageInfo.cbMaxToken);
194 win_skip("Second package missing\n");
197 pTables = getNextSecPkgTable(pTables, Version);
200 status = pTables->GetInfo(&PackageInfo);
201 ok(status == STATUS_SUCCESS ||
202 status == SEC_E_UNSUPPORTED_FUNCTION, /* win2k3 */
203 "status: 0x%x\n", status);
205 if (status == STATUS_SUCCESS)
207 ok(PackageInfo.fCapabilities == 0x107b3, "fCapabilities: 0x%x\n",
208 PackageInfo.fCapabilities);
209 ok(PackageInfo.wVersion == 1, "wVersion: %d\n", PackageInfo.wVersion);
210 ok(PackageInfo.wRPCID == 14, "wRPCID: %d\n", PackageInfo.wRPCID);
211 ok(PackageInfo.cbMaxToken == 0x4000 ||
212 PackageInfo.cbMaxToken == 0x6000, /* Win7 */
213 "cbMaxToken: 0x%x\n",
214 PackageInfo.cbMaxToken);
220 HMODULE hMod = LoadLibraryA("schannel.dll");
222 win_skip("schannel.dll not available\n");
226 pSpLsaModeInitialize = (void *)GetProcAddress(hMod, "SpLsaModeInitialize");
227 pSpUserModeInitialize = (void *)GetProcAddress(hMod, "SpUserModeInitialize");
229 if (pSpLsaModeInitialize && pSpUserModeInitialize)
234 else win_skip( "schannel functions not found\n" );