2 * Implementation of userenv.dll
4 * Copyright 2006 Mike McCormack for CodeWeavers
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
34 #include "wine/debug.h"
35 #include "wine/unicode.h"
37 WINE_DEFAULT_DEBUG_CHANNEL( userenv
);
39 BOOL WINAPI
DllMain(HINSTANCE hinstDLL
, DWORD fdwReason
, LPVOID lpvReserved
)
41 TRACE("%p %d %p\n", hinstDLL
, fdwReason
, lpvReserved
);
45 case DLL_WINE_PREATTACH
:
46 return FALSE
; /* prefer native version */
47 case DLL_PROCESS_ATTACH
:
48 DisableThreadLibraryCalls(hinstDLL
);
54 static BOOL
get_reg_value(WCHAR
*env
, HKEY hkey
, const WCHAR
*name
, WCHAR
*val
, DWORD size
)
56 DWORD type
, res_size
=0;
58 if (RegQueryValueExW(hkey
, name
, 0, &type
, NULL
, &res_size
) != ERROR_SUCCESS
)
66 return RegQueryValueExW(hkey
, name
, 0, NULL
, (BYTE
*)val
, &size
) == ERROR_SUCCESS
;
68 else if (type
== REG_EXPAND_SZ
)
70 UNICODE_STRING us_buf
, us_expanded
;
71 WCHAR
*buf
= HeapAlloc(GetProcessHeap(), 0, res_size
);
75 if (RegQueryValueExW(hkey
, name
, 0, NULL
, (BYTE
*)buf
, &res_size
) != ERROR_SUCCESS
)
77 HeapFree(GetProcessHeap(), 0, buf
);
81 RtlInitUnicodeString(&us_buf
, buf
);
82 us_expanded
.Buffer
= val
;
83 us_expanded
.MaximumLength
= size
;
84 if (RtlExpandEnvironmentStrings_U(env
, &us_buf
, &us_expanded
, &size
) != STATUS_SUCCESS
)
86 HeapFree(GetProcessHeap(), 0, buf
);
90 HeapFree(GetProcessHeap(), 0, buf
);
97 static void set_registry_variables(WCHAR
**env
, HKEY hkey
, DWORD type
, BOOL set_path
)
99 static const WCHAR SystemRootW
[] = {'S','y','s','t','e','m','R','o','o','t',0};
100 static const WCHAR SystemDriveW
[] = {'S','y','s','t','e','m','D','r','i','v','e',0};
101 static const WCHAR PATHW
[] = {'P','A','T','H'};
103 UNICODE_STRING us_name
, us_value
;
104 WCHAR name
[1024], value
[1024];
105 DWORD ret
, index
, size
;
107 for (index
= 0; ; index
++)
109 size
= sizeof(name
)/sizeof(WCHAR
);
110 ret
= RegEnumValueW(hkey
, index
, name
, &size
, NULL
, NULL
, NULL
, NULL
);
111 if (ret
!= ERROR_SUCCESS
)
114 if (!memicmpW(name
, SystemRootW
, sizeof(SystemRootW
)/sizeof(WCHAR
)))
116 if (!memicmpW(name
, SystemDriveW
, sizeof(SystemDriveW
)/sizeof(WCHAR
)))
119 RtlInitUnicodeString(&us_name
, name
);
120 us_value
.Buffer
= value
;
121 us_value
.MaximumLength
= sizeof(value
);
122 if (!memicmpW(name
, PATHW
, sizeof(PATHW
)/sizeof(WCHAR
)) &&
123 !RtlQueryEnvironmentVariable_U(*env
, &us_name
, &us_value
))
128 size
= strlenW(value
)+1;
129 if (!get_reg_value(*env
, hkey
, name
, value
+size
,
130 sizeof(value
)-size
*sizeof(WCHAR
)))
134 RtlInitUnicodeString(&us_value
, value
);
135 RtlSetEnvironmentVariable(env
, &us_name
, &us_value
);
139 if (!get_reg_value(*env
, hkey
, name
, value
, sizeof(value
)))
145 RtlInitUnicodeString(&us_value
, value
);
146 RtlSetEnvironmentVariable(env
, &us_name
, &us_value
);
150 static void set_wow64_environment(WCHAR
**env
)
152 static const WCHAR versionW
[] = {'S','o','f','t','w','a','r','e','\\',
153 'M','i','c','r','o','s','o','f','t','\\',
154 'W','i','n','d','o','w','s','\\',
155 'C','u','r','r','e','n','t','V','e','r','s','i','o','n',0};
156 static const WCHAR progdirW
[] = {'P','r','o','g','r','a','m','F','i','l','e','s','D','i','r',0};
157 static const WCHAR progdir86W
[] = {'P','r','o','g','r','a','m','F','i','l','e','s','D','i','r',' ','(','x','8','6',')',0};
158 static const WCHAR progfilesW
[] = {'P','r','o','g','r','a','m','F','i','l','e','s',0};
159 static const WCHAR progw6432W
[] = {'P','r','o','g','r','a','m','W','6','4','3','2',0};
160 static const WCHAR commondirW
[] = {'C','o','m','m','o','n','F','i','l','e','s','D','i','r',0};
161 static const WCHAR commondir86W
[] = {'C','o','m','m','o','n','F','i','l','e','s','D','i','r',' ','(','x','8','6',')',0};
162 static const WCHAR commonfilesW
[] = {'C','o','m','m','o','n','P','r','o','g','r','a','m','F','i','l','e','s',0};
163 static const WCHAR commonw6432W
[] = {'C','o','m','m','o','n','P','r','o','g','r','a','m','W','6','4','3','2',0};
165 UNICODE_STRING nameW
, valueW
;
168 BOOL is_win64
= (sizeof(void *) > sizeof(int));
171 IsWow64Process( GetCurrentProcess(), &is_wow64
);
173 if (RegOpenKeyExW(HKEY_LOCAL_MACHINE
, versionW
, 0,
174 KEY_READ
|KEY_WOW64_64KEY
, &hkey
))
177 /* set the ProgramFiles variables */
179 if (get_reg_value(*env
, hkey
, progdirW
, buf
, sizeof(buf
)))
181 if (is_win64
|| is_wow64
)
183 RtlInitUnicodeString(&nameW
, progw6432W
);
184 RtlInitUnicodeString(&valueW
, buf
);
185 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
187 if (is_win64
|| !is_wow64
)
189 RtlInitUnicodeString(&nameW
, progfilesW
);
190 RtlInitUnicodeString(&valueW
, buf
);
191 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
194 if (is_wow64
&& get_reg_value(*env
, hkey
, progdir86W
, buf
, sizeof(buf
)))
196 RtlInitUnicodeString(&nameW
, progfilesW
);
197 RtlInitUnicodeString(&valueW
, buf
);
198 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
201 /* set the CommonProgramFiles variables */
203 if (get_reg_value(*env
, hkey
, commondirW
, buf
, sizeof(buf
)))
205 if (is_win64
|| is_wow64
)
207 RtlInitUnicodeString(&nameW
, commonw6432W
);
208 RtlInitUnicodeString(&valueW
, buf
);
209 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
211 if (is_win64
|| !is_wow64
)
213 RtlInitUnicodeString(&nameW
, commonfilesW
);
214 RtlInitUnicodeString(&valueW
, buf
);
215 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
218 if (is_wow64
&& get_reg_value(*env
, hkey
, commondir86W
, buf
, sizeof(buf
)))
220 RtlInitUnicodeString(&nameW
, commonfilesW
);
221 RtlInitUnicodeString(&valueW
, buf
);
222 RtlSetEnvironmentVariable(env
, &nameW
, &valueW
);
228 BOOL WINAPI
CreateEnvironmentBlock( LPVOID
* lpEnvironment
,
229 HANDLE hToken
, BOOL bInherit
)
231 static const WCHAR env_keyW
[] = {'S','y','s','t','e','m','\\',
232 'C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\',
233 'C','o','n','t','r','o','l','\\',
234 'S','e','s','s','i','o','n',' ','M','a','n','a','g','e','r','\\',
235 'E','n','v','i','r','o','n','m','e','n','t',0};
236 static const WCHAR profile_keyW
[] = {'S','o','f','t','w','a','r','e','\\',
237 'M','i','c','r','o','s','o','f','t','\\',
238 'W','i','n','d','o','w','s',' ','N','T','\\',
239 'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
240 'P','r','o','f','i','l','e','L','i','s','t',0};
241 static const WCHAR envW
[] = {'E','n','v','i','r','o','n','m','e','n','t',0};
242 static const WCHAR volatile_envW
[] = {'V','o','l','a','t','i','l','e',' ','E','n','v','i','r','o','n','m','e','n','t',0};
243 static const WCHAR ProfilesDirectoryW
[] = {'P','r','o','f','i','l','e','s','D','i','r','e','c','t','o','r','y',0};
245 static const WCHAR SystemRootW
[] = {'S','y','s','t','e','m','R','o','o','t',0};
246 static const WCHAR SystemDriveW
[] = {'S','y','s','t','e','m','D','r','i','v','e',0};
247 static const WCHAR AllUsersProfileW
[] = {'A','l','l','U','s','e','r','s','P','r','o','f','i','l','e',0};
248 static const WCHAR ALLUSERSPROFILEW
[] = {'A','L','L','U','S','E','R','S','P','R','O','F','I','L','E',0};
249 static const WCHAR USERNAMEW
[] = {'U','S','E','R','N','A','M','E',0};
250 static const WCHAR USERPROFILEW
[] = {'U','S','E','R','P','R','O','F','I','L','E',0};
251 static const WCHAR DefaultW
[] = {'D','e','f','a','u','l','t',0};
252 static const WCHAR COMPUTERNAMEW
[] = {'C','O','M','P','U','T','E','R','N','A','M','E',0};
254 WCHAR
*env
, buf
[UNICODE_STRING_MAX_CHARS
], profiles_dir
[MAX_PATH
];
255 UNICODE_STRING us_name
, us_val
;
259 TRACE("%p %p %d\n", lpEnvironment
, hToken
, bInherit
);
264 if (RegOpenKeyExW(HKEY_LOCAL_MACHINE
, env_keyW
, 0, KEY_READ
, &hkey
) != ERROR_SUCCESS
)
267 if (RtlCreateEnvironment(bInherit
, &env
) != STATUS_SUCCESS
)
273 if (!GetEnvironmentVariableW(SystemRootW
, buf
, UNICODE_STRING_MAX_CHARS
))
275 if (!get_reg_value(env
, hkey
, SystemRootW
, buf
, UNICODE_STRING_MAX_CHARS
))
278 WARN("SystemRoot variable not set\n");
281 RtlInitUnicodeString(&us_name
, SystemRootW
);
282 RtlInitUnicodeString(&us_val
, buf
);
283 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
285 if (!GetEnvironmentVariableW(SystemDriveW
, buf
, UNICODE_STRING_MAX_CHARS
))
287 if (!get_reg_value(env
, hkey
, SystemRootW
, buf
, UNICODE_STRING_MAX_CHARS
))
290 WARN("SystemDrive variable not set\n");
293 RtlInitUnicodeString(&us_name
, SystemDriveW
);
294 RtlInitUnicodeString(&us_val
, buf
);
295 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
297 set_registry_variables(&env
, hkey
, REG_SZ
, !bInherit
);
298 set_registry_variables(&env
, hkey
, REG_EXPAND_SZ
, !bInherit
);
300 if (RegOpenKeyExW(hkey
, envW
, 0, KEY_READ
, &hsubkey
) == ERROR_SUCCESS
)
302 set_registry_variables(&env
, hsubkey
, REG_SZ
, !bInherit
);
303 set_registry_variables(&env
, hsubkey
, REG_EXPAND_SZ
, !bInherit
);
304 RegCloseKey(hsubkey
);
307 if (RegOpenKeyExW(hkey
, volatile_envW
, 0, KEY_READ
, &hsubkey
) == ERROR_SUCCESS
)
309 set_registry_variables(&env
, hsubkey
, REG_SZ
, !bInherit
);
310 set_registry_variables(&env
, hsubkey
, REG_EXPAND_SZ
, !bInherit
);
311 RegCloseKey(hsubkey
);
315 if (RegOpenKeyExW(HKEY_LOCAL_MACHINE
, profile_keyW
, 0, KEY_READ
, &hkey
) == ERROR_SUCCESS
)
317 if (get_reg_value(env
, hkey
, ProfilesDirectoryW
, profiles_dir
, MAX_PATH
-sizeof(WCHAR
)))
319 len
= strlenW(profiles_dir
);
320 if (profiles_dir
[len
-1] != '\\')
322 profiles_dir
[len
++] = '\\';
323 profiles_dir
[len
] = '\0';
326 memcpy(buf
, profiles_dir
, len
*sizeof(WCHAR
));
327 if (get_reg_value(env
, hkey
, AllUsersProfileW
, buf
+len
, UNICODE_STRING_MAX_CHARS
-len
))
329 RtlInitUnicodeString(&us_name
, ALLUSERSPROFILEW
);
330 RtlInitUnicodeString(&us_val
, buf
);
331 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
342 len
= sizeof(buf
)/sizeof(WCHAR
);
343 if (GetComputerNameW(buf
, &len
))
345 RtlInitUnicodeString(&us_name
, COMPUTERNAMEW
);
346 RtlInitUnicodeString(&us_val
, buf
);
347 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
350 set_wow64_environment(&env
);
356 len
= strlenW(profiles_dir
);
357 if (len
*sizeof(WCHAR
)+sizeof(DefaultW
) < sizeof(buf
))
359 memcpy(buf
, profiles_dir
, len
*sizeof(WCHAR
));
360 memcpy(buf
+len
, DefaultW
, sizeof(DefaultW
));
361 RtlInitUnicodeString(&us_name
, USERPROFILEW
);
362 RtlInitUnicodeString(&us_val
, buf
);
363 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
368 memcpy(buf
+1, DefaultW
, sizeof(DefaultW
));
372 TOKEN_USER
*token_user
= NULL
;
377 if (GetTokenInformation(hToken
, TokenUser
, NULL
, 0, &len
) ||
378 GetLastError()!=ERROR_INSUFFICIENT_BUFFER
||
379 !(token_user
= HeapAlloc(GetProcessHeap(), 0, len
)) ||
380 !GetTokenInformation(hToken
, TokenUser
, token_user
, len
, &len
) ||
381 !ConvertSidToStringSidW(token_user
->User
.Sid
, &sidW
))
383 HeapFree(GetProcessHeap(), 0, token_user
);
384 RtlDestroyEnvironment(env
);
388 len
= strlenW(profiles_dir
);
389 memcpy(buf
, profiles_dir
, len
*sizeof(WCHAR
));
391 size
= UNICODE_STRING_MAX_CHARS
-len
;
392 if (LookupAccountSidW(NULL
, token_user
->User
.Sid
,
393 buf
+len
, &size
, NULL
, &tmp
, &use
))
395 RtlInitUnicodeString(&us_name
, USERNAMEW
);
396 RtlInitUnicodeString(&us_val
, buf
+len
);
397 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
401 RtlInitUnicodeString(&us_name
, USERPROFILEW
);
402 RtlInitUnicodeString(&us_val
, buf
);
403 RtlSetEnvironmentVariable(&env
, &us_name
, &us_val
);
407 HeapFree(GetProcessHeap(), 0, token_user
);
412 if (RegOpenKeyExW(HKEY_USERS
, buf
, 0, KEY_READ
, &hkey
) == ERROR_SUCCESS
)
414 if (RegOpenKeyExW(hkey
, envW
, 0, KEY_READ
, &hsubkey
) == ERROR_SUCCESS
)
416 set_registry_variables(&env
, hsubkey
, REG_SZ
, !bInherit
);
417 set_registry_variables(&env
, hsubkey
, REG_EXPAND_SZ
, !bInherit
);
418 RegCloseKey(hsubkey
);
421 if (RegOpenKeyExW(hkey
, volatile_envW
, 0, KEY_READ
, &hsubkey
) == ERROR_SUCCESS
)
423 set_registry_variables(&env
, hsubkey
, REG_SZ
, !bInherit
);
424 set_registry_variables(&env
, hsubkey
, REG_EXPAND_SZ
, !bInherit
);
425 RegCloseKey(hsubkey
);
430 *lpEnvironment
= env
;
434 BOOL WINAPI
DestroyEnvironmentBlock(LPVOID lpEnvironment
)
438 TRACE("%p\n", lpEnvironment
);
439 r
= RtlDestroyEnvironment(lpEnvironment
);
440 if (r
== STATUS_SUCCESS
)
445 BOOL WINAPI
ExpandEnvironmentStringsForUserA( HANDLE hToken
, LPCSTR lpSrc
,
446 LPSTR lpDest
, DWORD dwSize
)
450 TRACE("%p %s %p %d\n", hToken
, debugstr_a(lpSrc
), lpDest
, dwSize
);
452 ret
= ExpandEnvironmentStringsA( lpSrc
, lpDest
, dwSize
);
453 TRACE("<- %s\n", debugstr_a(lpDest
));
457 BOOL WINAPI
ExpandEnvironmentStringsForUserW( HANDLE hToken
, LPCWSTR lpSrc
,
458 LPWSTR lpDest
, DWORD dwSize
)
462 TRACE("%p %s %p %d\n", hToken
, debugstr_w(lpSrc
), lpDest
, dwSize
);
464 ret
= ExpandEnvironmentStringsW( lpSrc
, lpDest
, dwSize
);
465 TRACE("<- %s\n", debugstr_w(lpDest
));
469 BOOL WINAPI
GetDefaultUserProfileDirectoryA( LPSTR lpProfileDir
, LPDWORD lpcchSize
)
471 FIXME("%p %p\n", lpProfileDir
, lpcchSize
);
475 BOOL WINAPI
GetDefaultUserProfileDirectoryW( LPWSTR lpProfileDir
, LPDWORD lpcchSize
)
477 FIXME("%p %p\n", lpProfileDir
, lpcchSize
);
481 BOOL WINAPI
GetUserProfileDirectoryA( HANDLE hToken
, LPSTR lpProfileDir
,
487 TRACE( "%p %p %p\n", hToken
, lpProfileDir
, lpcchSize
);
489 if (!lpProfileDir
|| !lpcchSize
)
491 SetLastError( ERROR_INVALID_PARAMETER
);
494 if (!(dirW
= HeapAlloc( GetProcessHeap(), 0, *lpcchSize
* sizeof(WCHAR
) )))
497 if ((ret
= GetUserProfileDirectoryW( hToken
, dirW
, lpcchSize
)))
498 WideCharToMultiByte( CP_ACP
, 0, dirW
, *lpcchSize
, lpProfileDir
, *lpcchSize
, NULL
, NULL
);
500 HeapFree( GetProcessHeap(), 0, dirW
);
504 BOOL WINAPI
GetUserProfileDirectoryW( HANDLE hToken
, LPWSTR lpProfileDir
,
507 static const WCHAR slashW
[] = {'\\',0};
509 WCHAR
*userW
= NULL
, *dirW
= NULL
;
510 DWORD len
, dir_len
, domain_len
;
514 TRACE( "%p %p %p\n", hToken
, lpProfileDir
, lpcchSize
);
518 SetLastError( ERROR_INVALID_PARAMETER
);
523 GetTokenInformation( hToken
, TokenUser
, NULL
, 0, &len
);
524 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER
) return FALSE
;
525 if (!(t
= HeapAlloc( GetProcessHeap(), 0, len
))) return FALSE
;
526 if (!GetTokenInformation( hToken
, TokenUser
, t
, len
, &len
)) goto done
;
528 len
= domain_len
= 0;
529 LookupAccountSidW( NULL
, t
->User
.Sid
, NULL
, &len
, NULL
, &domain_len
, NULL
);
530 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER
) goto done
;
531 if (!(userW
= HeapAlloc( GetProcessHeap(), 0, len
* sizeof(WCHAR
) ))) goto done
;
532 if (!LookupAccountSidW( NULL
, t
->User
.Sid
, userW
, &len
, NULL
, &domain_len
, &use
)) goto done
;
535 GetProfilesDirectoryW( NULL
, &dir_len
);
536 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER
) goto done
;
537 if (!(dirW
= HeapAlloc( GetProcessHeap(), 0, (dir_len
+ 1) * sizeof(WCHAR
) ))) goto done
;
538 if (!GetProfilesDirectoryW( dirW
, &dir_len
)) goto done
;
541 if (*lpcchSize
< len
)
543 SetLastError( ERROR_INSUFFICIENT_BUFFER
);
547 strcpyW( lpProfileDir
, dirW
);
548 strcatW( lpProfileDir
, slashW
);
549 strcatW( lpProfileDir
, userW
);
554 HeapFree( GetProcessHeap(), 0, t
);
555 HeapFree( GetProcessHeap(), 0, userW
);
556 HeapFree( GetProcessHeap(), 0, dirW
);
560 static const char ProfileListA
[] = "Software\\Microsoft\\Windows NT\\CurrentVersion\\ProfileList";
562 BOOL WINAPI
GetProfilesDirectoryA( LPSTR lpProfilesDir
, LPDWORD lpcchSize
)
564 static const char ProfilesDirectory
[] = "ProfilesDirectory";
568 DWORD len
= 0, expanded_len
;
569 LPSTR unexpanded_profiles_dir
= NULL
;
571 TRACE("%p %p\n", lpProfilesDir
, lpcchSize
);
573 if (!lpProfilesDir
|| !lpcchSize
)
575 SetLastError(ERROR_INVALID_PARAMETER
);
579 l
= RegOpenKeyExA(HKEY_LOCAL_MACHINE
, ProfileListA
, 0, KEY_READ
, &key
);
585 l
= RegQueryValueExA(key
, ProfilesDirectory
, NULL
, NULL
, NULL
, &len
);
591 unexpanded_profiles_dir
= HeapAlloc(GetProcessHeap(), 0, len
);
592 if (!unexpanded_profiles_dir
)
594 SetLastError(ERROR_OUTOFMEMORY
);
597 l
= RegQueryValueExA(key
, ProfilesDirectory
, NULL
, NULL
,
598 (BYTE
*)unexpanded_profiles_dir
, &len
);
604 expanded_len
= ExpandEnvironmentStringsA(unexpanded_profiles_dir
, NULL
, 0);
605 /* The returned length doesn't include the NULL terminator. */
606 if (*lpcchSize
< expanded_len
- 1)
608 *lpcchSize
= expanded_len
- 1;
609 SetLastError(ERROR_INSUFFICIENT_BUFFER
);
612 *lpcchSize
= expanded_len
- 1;
613 /* The return value is also the expected length. */
614 ret
= ExpandEnvironmentStringsA(unexpanded_profiles_dir
, lpProfilesDir
,
617 HeapFree(GetProcessHeap(), 0, unexpanded_profiles_dir
);
622 static const WCHAR ProfileListW
[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\','W','i','n','d','o','w','s',' ','N','T','\\','C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\','P','r','o','f','i','l','e','L','i','s','t',0};
624 BOOL WINAPI
GetProfilesDirectoryW( LPWSTR lpProfilesDir
, LPDWORD lpcchSize
)
626 static const WCHAR ProfilesDirectory
[] = {'P','r','o','f','i','l','e','s','D','i','r','e','c','t','o','r','y',0};
630 DWORD len
= 0, expanded_len
;
631 LPWSTR unexpanded_profiles_dir
= NULL
;
633 TRACE("%p %p\n", lpProfilesDir
, lpcchSize
);
637 SetLastError(ERROR_INVALID_PARAMETER
);
641 l
= RegOpenKeyExW(HKEY_LOCAL_MACHINE
, ProfileListW
, 0, KEY_READ
, &key
);
647 l
= RegQueryValueExW(key
, ProfilesDirectory
, NULL
, NULL
, NULL
, &len
);
653 unexpanded_profiles_dir
= HeapAlloc(GetProcessHeap(), 0, len
);
654 if (!unexpanded_profiles_dir
)
656 SetLastError(ERROR_OUTOFMEMORY
);
659 l
= RegQueryValueExW(key
, ProfilesDirectory
, NULL
, NULL
,
660 (BYTE
*)unexpanded_profiles_dir
, &len
);
666 expanded_len
= ExpandEnvironmentStringsW(unexpanded_profiles_dir
, NULL
, 0);
667 /* The returned length doesn't include the NULL terminator. */
668 if (*lpcchSize
< expanded_len
- 1 || !lpProfilesDir
)
670 *lpcchSize
= expanded_len
- 1;
671 SetLastError(ERROR_INSUFFICIENT_BUFFER
);
674 *lpcchSize
= expanded_len
- 1;
675 /* The return value is also the expected length. */
676 ret
= ExpandEnvironmentStringsW(unexpanded_profiles_dir
, lpProfilesDir
,
679 HeapFree(GetProcessHeap(), 0, unexpanded_profiles_dir
);
684 BOOL WINAPI
GetAllUsersProfileDirectoryA( LPSTR lpProfileDir
, LPDWORD lpcchSize
)
686 FIXME("%p %p\n", lpProfileDir
, lpcchSize
);
690 BOOL WINAPI
GetAllUsersProfileDirectoryW( LPWSTR lpProfileDir
, LPDWORD lpcchSize
)
692 FIXME("%p %p\n", lpProfileDir
, lpcchSize
);
696 BOOL WINAPI
GetProfileType( DWORD
*pdwFlags
)
698 FIXME("%p\n", pdwFlags
);
703 BOOL WINAPI
LoadUserProfileA( HANDLE hToken
, LPPROFILEINFOA lpProfileInfo
)
705 FIXME("%p %p\n", hToken
, lpProfileInfo
);
706 lpProfileInfo
->hProfile
= HKEY_CURRENT_USER
;
710 BOOL WINAPI
LoadUserProfileW( HANDLE hToken
, LPPROFILEINFOW lpProfileInfo
)
712 FIXME("%p %p\n", hToken
, lpProfileInfo
);
713 lpProfileInfo
->hProfile
= HKEY_CURRENT_USER
;
717 BOOL WINAPI
RegisterGPNotification( HANDLE event
, BOOL machine
)
719 FIXME("%p %d\n", event
, machine
);
723 BOOL WINAPI
UnregisterGPNotification( HANDLE event
)
725 FIXME("%p\n", event
);
729 BOOL WINAPI
UnloadUserProfile( HANDLE hToken
, HANDLE hProfile
)
731 FIXME("(%p, %p): stub\n", hToken
, hProfile
);
735 HANDLE WINAPI
EnterCriticalPolicySection(BOOL bMachine
)
737 FIXME("(%x)\n", bMachine
);
738 SetLastError(ERROR_ACCESS_DENIED
);
742 BOOL WINAPI
LeaveCriticalPolicySection(HANDLE hSection
)
744 FIXME("(%p)\n", hSection
);
748 DWORD WINAPI
GetAppliedGPOListW(DWORD dwFlags
, LPCWSTR pMachineName
, PSID pSidUser
, GUID
*pGuidExtension
,
749 PGROUP_POLICY_OBJECTW
*ppGPOList
)
751 FIXME("(%x %s %p %s %p)\n", dwFlags
, debugstr_w(pMachineName
), pSidUser
, debugstr_guid(pGuidExtension
), ppGPOList
);
752 return ERROR_ACCESS_DENIED
;
755 /******************************************************************************
761 * int csidl [in] well-known directory location to create link in
762 * LPCSTR lnk_dir [in] directory (relative to directory specified by csidl) to create link in
763 * LPCSTR lnk_filename [in] filename of the link file without .lnk extension
764 * LPCSTR lnk_target [in] file/directory pointed to by link
765 * LPCSTR lnk_iconfile [in] link icon resource filename
766 * DWORD lnk_iconid [in] link icon resource id in file referred by lnk_iconfile
767 * LPCSTR work_directory [in] link target's work directory
768 * WORD hotkey [in] link hotkey (virtual key id)
769 * DWORD win_state [in] initial window size (SW_SHOWMAXIMIZED to start maximized,
770 * SW_SHOWMINNOACTIVE to start minimized, everything else is default state)
771 * LPCSTR comment [in] comment - link's comment
772 * LPCSTR loc_filename_resfile [in] resource file which holds localized filename for this link file
773 * DWORD loc_filename_resid [in] resource id for this link file's localized filename
776 * TRUE: Link file was successfully created
777 * FALSE: Link file was not created
779 BOOL WINAPI
USERENV_138( int csidl
, LPCSTR lnk_dir
, LPCSTR lnk_filename
,
780 LPCSTR lnk_target
, LPCSTR lnk_iconfile
, DWORD lnk_iconid
,
781 LPCSTR work_directory
, WORD hotkey
, DWORD win_state
, LPCSTR comment
,
782 LPCSTR loc_filename_resfile
, DWORD loc_filename_resid
)
784 FIXME("(%d,%s,%s,%s,%s,%d,%s,0x%x,%d,%s,%s,%d) - stub\n", csidl
, debugstr_a(lnk_dir
),
785 debugstr_a(lnk_filename
), debugstr_a(lnk_target
), debugstr_a(lnk_iconfile
),
786 lnk_iconid
, debugstr_a(work_directory
), hotkey
, win_state
,
787 debugstr_a(comment
), debugstr_a(loc_filename_resfile
), loc_filename_resid
);