mirror of
https://github.com/reactos/reactos.git
synced 2025-08-11 13:55:36 +00:00
632 lines
13 KiB
C
632 lines
13 KiB
C
/* $Id: env.c,v 1.23 2004/01/12 11:05:38 gvg Exp $
|
|
*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS system libraries
|
|
* FILE: lib/kernel32/misc/env.c
|
|
* PURPOSE: Environment functions
|
|
* PROGRAMMER: Ariadne ( ariadne@xs4all.nl)
|
|
* UPDATE HISTORY:
|
|
* Created 01/11/98
|
|
*/
|
|
|
|
#include <k32.h>
|
|
|
|
#define NDEBUG
|
|
#include <kernel32/kernel32.h>
|
|
|
|
|
|
/* FUNCTIONS ******************************************************************/
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
DWORD
|
|
STDCALL
|
|
GetEnvironmentVariableA (
|
|
LPCSTR lpName,
|
|
LPSTR lpBuffer,
|
|
DWORD nSize
|
|
)
|
|
{
|
|
ANSI_STRING VarName;
|
|
ANSI_STRING VarValue;
|
|
UNICODE_STRING VarNameU;
|
|
UNICODE_STRING VarValueU;
|
|
NTSTATUS Status;
|
|
|
|
/* initialize unicode variable name string */
|
|
RtlInitAnsiString (&VarName,
|
|
(LPSTR)lpName);
|
|
RtlAnsiStringToUnicodeString (&VarNameU,
|
|
&VarName,
|
|
TRUE);
|
|
|
|
/* initialize ansi variable value string */
|
|
VarValue.Length = 0;
|
|
VarValue.MaximumLength = nSize;
|
|
VarValue.Buffer = lpBuffer;
|
|
|
|
/* initialize unicode variable value string and allocate buffer */
|
|
VarValueU.Length = 0;
|
|
VarValueU.MaximumLength = nSize * sizeof(WCHAR);
|
|
VarValueU.Buffer = RtlAllocateHeap (RtlGetProcessHeap (),
|
|
0,
|
|
VarValueU.MaximumLength);
|
|
|
|
/* get unicode environment variable */
|
|
Status = RtlQueryEnvironmentVariable_U (NULL,
|
|
&VarNameU,
|
|
&VarValueU);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
/* free unicode buffer */
|
|
RtlFreeHeap (RtlGetProcessHeap (),
|
|
0,
|
|
VarValueU.Buffer);
|
|
|
|
/* free unicode variable name string */
|
|
RtlFreeUnicodeString (&VarNameU);
|
|
|
|
SetLastErrorByStatus (Status);
|
|
if (Status == STATUS_BUFFER_TOO_SMALL)
|
|
{
|
|
return VarValueU.Length / sizeof(WCHAR) + 1;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* convert unicode value string to ansi */
|
|
RtlUnicodeStringToAnsiString (&VarValue,
|
|
&VarValueU,
|
|
FALSE);
|
|
|
|
/* free unicode buffer */
|
|
RtlFreeHeap (RtlGetProcessHeap (),
|
|
0,
|
|
VarValueU.Buffer);
|
|
|
|
/* free unicode variable name string */
|
|
RtlFreeUnicodeString (&VarNameU);
|
|
|
|
return (VarValueU.Length / sizeof(WCHAR));
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
DWORD
|
|
STDCALL
|
|
GetEnvironmentVariableW (
|
|
LPCWSTR lpName,
|
|
LPWSTR lpBuffer,
|
|
DWORD nSize
|
|
)
|
|
{
|
|
UNICODE_STRING VarName;
|
|
UNICODE_STRING VarValue;
|
|
NTSTATUS Status;
|
|
|
|
RtlInitUnicodeString (&VarName,
|
|
lpName);
|
|
|
|
VarValue.Length = 0;
|
|
VarValue.MaximumLength = nSize * sizeof(WCHAR);
|
|
VarValue.Buffer = lpBuffer;
|
|
|
|
Status = RtlQueryEnvironmentVariable_U (NULL,
|
|
&VarName,
|
|
&VarValue);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
SetLastErrorByStatus (Status);
|
|
if (Status == STATUS_BUFFER_TOO_SMALL)
|
|
{
|
|
return (VarValue.Length / sizeof(WCHAR)) + 1;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return (VarValue.Length / sizeof(WCHAR));
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
SetEnvironmentVariableA (
|
|
LPCSTR lpName,
|
|
LPCSTR lpValue
|
|
)
|
|
{
|
|
ANSI_STRING VarName;
|
|
ANSI_STRING VarValue;
|
|
UNICODE_STRING VarNameU;
|
|
UNICODE_STRING VarValueU;
|
|
NTSTATUS Status;
|
|
|
|
DPRINT("SetEnvironmentVariableA(Name '%s', Value '%s')\n", lpName, lpValue);
|
|
|
|
RtlInitAnsiString (&VarName,
|
|
(LPSTR)lpName);
|
|
RtlAnsiStringToUnicodeString (&VarNameU,
|
|
&VarName,
|
|
TRUE);
|
|
|
|
RtlInitAnsiString (&VarValue,
|
|
(LPSTR)lpValue);
|
|
RtlAnsiStringToUnicodeString (&VarValueU,
|
|
&VarValue,
|
|
TRUE);
|
|
|
|
Status = RtlSetEnvironmentVariable (NULL,
|
|
&VarNameU,
|
|
&VarValueU);
|
|
|
|
RtlFreeUnicodeString (&VarNameU);
|
|
RtlFreeUnicodeString (&VarValueU);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
SetLastErrorByStatus (Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
SetEnvironmentVariableW (
|
|
LPCWSTR lpName,
|
|
LPCWSTR lpValue
|
|
)
|
|
{
|
|
UNICODE_STRING VarName;
|
|
UNICODE_STRING VarValue;
|
|
NTSTATUS Status;
|
|
|
|
DPRINT("SetEnvironmentVariableW(Name '%S', Value '%S')\n", lpName, lpValue);
|
|
|
|
RtlInitUnicodeString (&VarName,
|
|
lpName);
|
|
|
|
RtlInitUnicodeString (&VarValue,
|
|
lpValue);
|
|
|
|
Status = RtlSetEnvironmentVariable (NULL,
|
|
&VarName,
|
|
&VarValue);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
SetLastErrorByStatus (Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
DWORD
|
|
STDCALL
|
|
GetVersion(VOID)
|
|
{
|
|
PPEB pPeb = NtCurrentPeb();
|
|
DWORD nVersion;
|
|
|
|
nVersion = MAKEWORD(pPeb->OSMajorVersion, pPeb->OSMinorVersion);
|
|
|
|
/* behave consistently when posing as another operating system */
|
|
/* build number */
|
|
if(pPeb->OSPlatformId != VER_PLATFORM_WIN32_WINDOWS)
|
|
nVersion |= ((DWORD)(pPeb->OSBuildNumber)) << 16;
|
|
|
|
/* non-NT platform flag */
|
|
if(pPeb->OSPlatformId != VER_PLATFORM_WIN32_NT)
|
|
nVersion |= 0x80000000;
|
|
|
|
return nVersion;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
GetVersionExW(
|
|
LPOSVERSIONINFOW lpVersionInformation
|
|
)
|
|
{
|
|
PPEB pPeb = NtCurrentPeb();
|
|
|
|
/* TODO: move this into RtlGetVersion */
|
|
switch(lpVersionInformation->dwOSVersionInfoSize)
|
|
{
|
|
case sizeof(OSVERSIONINFOEXW):
|
|
{
|
|
LPOSVERSIONINFOEXW lpVersionInformationEx =
|
|
(LPOSVERSIONINFOEXW)lpVersionInformation;
|
|
|
|
lpVersionInformationEx->wServicePackMajor = pPeb->SPMajorVersion;
|
|
lpVersionInformationEx->wServicePackMinor = pPeb->SPMinorVersion;
|
|
/* TODO: read from the KUSER_SHARED_DATA */
|
|
lpVersionInformationEx->wSuiteMask = 0;
|
|
/* TODO: call RtlGetNtProductType */
|
|
lpVersionInformationEx->wProductType = 0;
|
|
/* ??? */
|
|
lpVersionInformationEx->wReserved = 0;
|
|
/* fall through */
|
|
}
|
|
|
|
case sizeof(OSVERSIONINFOW):
|
|
{
|
|
lpVersionInformation->dwMajorVersion = pPeb->OSMajorVersion;
|
|
lpVersionInformation->dwMinorVersion = pPeb->OSMinorVersion;
|
|
lpVersionInformation->dwBuildNumber = pPeb->OSBuildNumber;
|
|
lpVersionInformation->dwPlatformId = pPeb->OSPlatformId;
|
|
|
|
/* version string is "ReactOS x.y.z" */
|
|
wcsncpy
|
|
(
|
|
lpVersionInformation->szCSDVersion,
|
|
L"ReactOS " KERNEL_VERSION_STR L" (Build " KERNEL_VERSION_BUILD_STR L")",
|
|
sizeof(lpVersionInformation->szCSDVersion) / sizeof(WCHAR)
|
|
);
|
|
|
|
/* null-terminate, just in case */
|
|
lpVersionInformation->szCSDVersion
|
|
[
|
|
sizeof(lpVersionInformation->szCSDVersion) / sizeof(WCHAR) - 1
|
|
] = 0;
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
/* unknown version information revision */
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
GetVersionExA(
|
|
LPOSVERSIONINFOA lpVersionInformation
|
|
)
|
|
{
|
|
NTSTATUS nErrCode;
|
|
OSVERSIONINFOEXW oviVerInfo;
|
|
LPOSVERSIONINFOEXA lpVersionInformationEx;
|
|
|
|
/* UNICODE_STRING descriptor of the Unicode version string */
|
|
UNICODE_STRING wstrVerStr =
|
|
{
|
|
/*
|
|
gives extra work to RtlUnicodeStringToAnsiString, but spares us an
|
|
RtlInitUnicodeString round
|
|
*/
|
|
0,
|
|
sizeof(((LPOSVERSIONINFOW)NULL)->szCSDVersion),
|
|
oviVerInfo.szCSDVersion
|
|
};
|
|
|
|
/* ANSI_STRING descriptor of the ANSI version string buffer */
|
|
ANSI_STRING strVerStr =
|
|
{
|
|
0,
|
|
sizeof(((LPOSVERSIONINFOA)NULL)->szCSDVersion),
|
|
lpVersionInformation->szCSDVersion
|
|
};
|
|
|
|
switch(lpVersionInformation->dwOSVersionInfoSize)
|
|
{
|
|
case sizeof(OSVERSIONINFOEXA):
|
|
{
|
|
oviVerInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEXW);
|
|
break;
|
|
}
|
|
|
|
case sizeof(OSVERSIONINFOA):
|
|
{
|
|
oviVerInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFOW);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
/* unknown version information revision */
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
if(!GetVersionExW((LPOSVERSIONINFOW)&oviVerInfo))
|
|
return FALSE;
|
|
|
|
/* null-terminate, just in case */
|
|
oviVerInfo.szCSDVersion
|
|
[
|
|
sizeof(((LPOSVERSIONINFOW)NULL)->szCSDVersion) /
|
|
sizeof(((LPOSVERSIONINFOW)NULL)->szCSDVersion[0]) -
|
|
1
|
|
] = 0;
|
|
wstrVerStr.Length = wcslen(wstrVerStr.Buffer) * sizeof(WCHAR);
|
|
|
|
/* convert the version string */
|
|
nErrCode = RtlUnicodeStringToAnsiString(&strVerStr, &wstrVerStr, FALSE);
|
|
|
|
if(!NT_SUCCESS(nErrCode))
|
|
{
|
|
/* failure */
|
|
SetLastErrorByStatus(nErrCode);
|
|
return FALSE;
|
|
}
|
|
|
|
/* copy the fields */
|
|
lpVersionInformation->dwMajorVersion = oviVerInfo.dwMajorVersion;
|
|
lpVersionInformation->dwMinorVersion = oviVerInfo.dwMinorVersion;
|
|
lpVersionInformation->dwBuildNumber = oviVerInfo.dwBuildNumber;
|
|
lpVersionInformation->dwPlatformId = oviVerInfo.dwPlatformId;
|
|
|
|
if(lpVersionInformation->dwOSVersionInfoSize < sizeof(OSVERSIONINFOEXA))
|
|
/* success */
|
|
return TRUE;
|
|
|
|
/* copy the extended fields */
|
|
lpVersionInformationEx = (LPOSVERSIONINFOEXA)lpVersionInformation;
|
|
lpVersionInformationEx->wServicePackMajor = oviVerInfo.wServicePackMajor;
|
|
lpVersionInformationEx->wServicePackMinor = oviVerInfo.wServicePackMinor;
|
|
lpVersionInformationEx->wSuiteMask = oviVerInfo.wSuiteMask;
|
|
lpVersionInformationEx->wProductType = oviVerInfo.wProductType;
|
|
lpVersionInformationEx->wReserved = oviVerInfo.wReserved;
|
|
|
|
/* success */
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
LPSTR
|
|
STDCALL
|
|
GetEnvironmentStringsA (
|
|
VOID
|
|
)
|
|
{
|
|
UNICODE_STRING UnicodeString;
|
|
ANSI_STRING AnsiString;
|
|
PWCHAR EnvU;
|
|
PWCHAR PtrU;
|
|
ULONG Length;
|
|
PCHAR EnvPtr = NULL;
|
|
|
|
EnvU = (PWCHAR)(NtCurrentPeb ()->ProcessParameters->Environment);
|
|
|
|
if (EnvU == NULL)
|
|
return NULL;
|
|
|
|
if (*EnvU == 0)
|
|
return NULL;
|
|
|
|
/* get environment size */
|
|
PtrU = EnvU;
|
|
while (*PtrU)
|
|
{
|
|
while (*PtrU)
|
|
PtrU++;
|
|
PtrU++;
|
|
}
|
|
Length = (ULONG)(PtrU - EnvU);
|
|
DPRINT("Length %lu\n", Length);
|
|
|
|
/* allocate environment buffer */
|
|
EnvPtr = RtlAllocateHeap (RtlGetProcessHeap (),
|
|
0,
|
|
Length + 1);
|
|
DPRINT("EnvPtr %p\n", EnvPtr);
|
|
|
|
/* convert unicode environment to ansi */
|
|
UnicodeString.MaximumLength = Length * sizeof(WCHAR) + sizeof(WCHAR);
|
|
UnicodeString.Buffer = EnvU;
|
|
|
|
AnsiString.MaximumLength = Length + 1;
|
|
AnsiString.Length = 0;
|
|
AnsiString.Buffer = EnvPtr;
|
|
|
|
DPRINT ("UnicodeString.Buffer \'%S\'\n", UnicodeString.Buffer);
|
|
|
|
while (*(UnicodeString.Buffer))
|
|
{
|
|
UnicodeString.Length = wcslen (UnicodeString.Buffer) * sizeof(WCHAR);
|
|
UnicodeString.MaximumLength = UnicodeString.Length + sizeof(WCHAR);
|
|
if (UnicodeString.Length > 0)
|
|
{
|
|
AnsiString.Length = 0;
|
|
AnsiString.MaximumLength = Length + 1 - (AnsiString.Buffer - EnvPtr);
|
|
|
|
RtlUnicodeStringToAnsiString (&AnsiString,
|
|
&UnicodeString,
|
|
FALSE);
|
|
|
|
AnsiString.Buffer += (AnsiString.Length + 1);
|
|
UnicodeString.Buffer += ((UnicodeString.Length / sizeof(WCHAR)) + 1);
|
|
}
|
|
}
|
|
*(AnsiString.Buffer) = 0;
|
|
|
|
return EnvPtr;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
LPWSTR
|
|
STDCALL
|
|
GetEnvironmentStringsW (
|
|
VOID
|
|
)
|
|
{
|
|
return (LPWSTR)(NtCurrentPeb ()->ProcessParameters->Environment);
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
FreeEnvironmentStringsA (
|
|
LPSTR EnvironmentStrings
|
|
)
|
|
{
|
|
if (EnvironmentStrings == NULL)
|
|
return FALSE;
|
|
|
|
RtlFreeHeap (RtlGetProcessHeap (),
|
|
0,
|
|
EnvironmentStrings);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
WINBOOL
|
|
STDCALL
|
|
FreeEnvironmentStringsW (
|
|
LPWSTR EnvironmentStrings
|
|
)
|
|
{
|
|
(void)EnvironmentStrings;
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
DWORD
|
|
STDCALL
|
|
ExpandEnvironmentStringsA (
|
|
LPCSTR lpSrc,
|
|
LPSTR lpDst,
|
|
DWORD nSize
|
|
)
|
|
{
|
|
ANSI_STRING Source;
|
|
ANSI_STRING Destination;
|
|
UNICODE_STRING SourceU;
|
|
UNICODE_STRING DestinationU;
|
|
NTSTATUS Status;
|
|
ULONG Length = 0;
|
|
|
|
RtlInitAnsiString (&Source,
|
|
(LPSTR)lpSrc);
|
|
RtlAnsiStringToUnicodeString (&SourceU,
|
|
&Source,
|
|
TRUE);
|
|
|
|
Destination.Length = 0;
|
|
Destination.MaximumLength = nSize;
|
|
Destination.Buffer = lpDst,
|
|
|
|
DestinationU.Length = 0;
|
|
DestinationU.MaximumLength = nSize * sizeof(WCHAR);
|
|
DestinationU.Buffer = RtlAllocateHeap (RtlGetProcessHeap (),
|
|
0,
|
|
DestinationU.MaximumLength);
|
|
|
|
Status = RtlExpandEnvironmentStrings_U (NULL,
|
|
&SourceU,
|
|
&DestinationU,
|
|
&Length);
|
|
|
|
RtlFreeUnicodeString (&SourceU);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
RtlFreeHeap (RtlGetProcessHeap (),
|
|
0,
|
|
DestinationU.Buffer);
|
|
SetLastErrorByStatus (Status);
|
|
return 0;
|
|
}
|
|
|
|
RtlUnicodeStringToAnsiString (&Destination,
|
|
&DestinationU,
|
|
FALSE);
|
|
|
|
RtlFreeHeap (RtlGetProcessHeap (),
|
|
0,
|
|
DestinationU.Buffer);
|
|
|
|
return (Length / sizeof(WCHAR));
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
DWORD
|
|
STDCALL
|
|
ExpandEnvironmentStringsW (
|
|
LPCWSTR lpSrc,
|
|
LPWSTR lpDst,
|
|
DWORD nSize
|
|
)
|
|
{
|
|
UNICODE_STRING Source;
|
|
UNICODE_STRING Destination;
|
|
NTSTATUS Status;
|
|
ULONG Length = 0;
|
|
|
|
RtlInitUnicodeString (&Source,
|
|
(LPWSTR)lpSrc);
|
|
|
|
Destination.Length = 0;
|
|
Destination.MaximumLength = nSize * sizeof(WCHAR);
|
|
Destination.Buffer = lpDst;
|
|
|
|
Status = RtlExpandEnvironmentStrings_U (NULL,
|
|
&Source,
|
|
&Destination,
|
|
&Length);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
SetLastErrorByStatus (Status);
|
|
return 0;
|
|
}
|
|
|
|
return (Length / sizeof(WCHAR));
|
|
}
|
|
|
|
/* EOF */
|