168 lines
5 KiB
C
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;
|
||
|
}
|