windows-nt/Source/XPSP1/NT/base/crts/crtw32/misc/a_env.c
2020-09-26 16:20:57 +08:00

168 lines
5 KiB
C

/***
*a_env.c - A version of GetEnvironmentStrings.
*
* Copyright (c) 1993-2001, Microsoft Corporation. All rights reserved.
*
*Purpose:
* Use GetEnvironmentStringsW if available, otherwise use A version.
*
*Revision History:
* 03-29-94 CFW Module created.
* 12-27-94 CFW Call direct, all OS's have stubs.
* 01-10-95 CFW Debug CRT allocs.
* 04-07-95 CFW Create __crtGetEnvironmentStringsA.
* 07-03-95 GJF Modified to always malloc a buffer for the
* environment strings, and to free the OS's buffer.
* 06-10-96 GJF Initialize aEnv and wEnv to NULL in
* __crtGetEnvironmentStringsA. Also, detab-ed.
* 05-14-97 GJF Split off W version into another file and renamed this
* one as a_env.c.
* 03-03-98 RKP Supported 64 bits
* 05-17-00 GB Use ERROR_CALL_NOT_IMPLEMENTED for existance of W API
*
*******************************************************************************/
#include <cruntime.h>
#include <internal.h>
#include <stdlib.h>
#include <setlocal.h>
#include <awint.h>
#include <dbgint.h>
#define USE_W 1
#define USE_A 2
/***
*LPVOID __cdecl __crtGetEnvironmentStringsA - Get normal environment block
*
*Purpose:
* Internal support function. Since GetEnvironmentStrings returns in OEM
* and we want ANSI (note that GetEnvironmentVariable returns ANSI!) and
* SetFileApistoAnsi() does not affect it, we have no choice but to
* obtain the block in wide character and convert to ANSI.
*
*Entry:
* VOID
*
*Exit:
* LPVOID - pointer to environment block
*
*Exceptions:
*
*******************************************************************************/
LPVOID __cdecl __crtGetEnvironmentStringsA(
VOID
)
{
static int f_use = 0;
wchar_t *wEnv = NULL;
wchar_t *wTmp;
char *aEnv = NULL;
char *aTmp;
int nSizeW;
int nSizeA;
/*
* Look for 'preferred' flavor. Otherwise use available flavor.
* Must actually call the function to ensure it's not a stub.
*/
if ( 0 == f_use )
{
if ( NULL != (wEnv = GetEnvironmentStringsW()) )
f_use = USE_W;
else if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED)
f_use = USE_A;
}
/* Use "W" version */
if (USE_W == f_use)
{
/* obtain wide environment block */
if ( NULL == wEnv )
if ( NULL == (wEnv = GetEnvironmentStringsW()) )
return NULL;
/* look for double null that indicates end of block */
wTmp = wEnv;
while ( *wTmp != L'\0' ) {
if ( *++wTmp == L'\0' )
wTmp++;
}
/* calculate total size of block, including all nulls */
nSizeW = (int)(wTmp - wEnv + 1);
/* find out how much space needed for multi-byte environment */
nSizeA = WideCharToMultiByte( CP_ACP,
0,
wEnv,
nSizeW,
NULL,
0,
NULL,
NULL );
/* allocate space for multi-byte string */
if ( (nSizeA == 0) ||
((aEnv = (char *)_malloc_crt(nSizeA)) == NULL) )
{
FreeEnvironmentStringsW( wEnv );
return NULL;
}
/* do the conversion */
if ( !WideCharToMultiByte( CP_ACP,
0,
wEnv,
nSizeW,
aEnv,
nSizeA,
NULL,
NULL ) )
{
_free_crt( aEnv );
aEnv = NULL;
}
FreeEnvironmentStringsW( wEnv );
return aEnv;
}
/* Use "A" version */
if (USE_A == f_use || f_use == 0)
{
if ( NULL == aEnv )
if ( NULL == (aEnv = GetEnvironmentStringsA()) )
return NULL;
/* determine how big a buffer is needed */
aTmp = aEnv;
while ( *aTmp != '\0' ) {
if ( *++aTmp == '\0' )
aTmp++;
}
nSizeA = (int)(aTmp - aEnv + 1);
if ( NULL == (aTmp = _malloc_crt( nSizeA )) ) {
FreeEnvironmentStringsA( aEnv );
return NULL;
}
memcpy( aTmp, aEnv, nSizeA );
FreeEnvironmentStringsA( aEnv );
return aTmp;
}
return NULL;
}