mirror of
https://github.com/reactos/reactos.git
synced 2025-01-02 20:43:18 +00:00
- Complete rewrite of environment variable handling to get rid of memory leaks, heap corruption and spurious pointers.
- Remove some ancient Wine stuff that either wasn't used or didn't work. - Fix realloc to not call HeapReAlloc for allocating new memory. svn path=/trunk/; revision=10703
This commit is contained in:
parent
0f59361d11
commit
aa00673d45
13 changed files with 440 additions and 566 deletions
|
@ -1,4 +1,4 @@
|
||||||
# $Id: Makefile,v 1.46 2004/08/22 20:37:05 hbirr Exp $
|
# $Id: Makefile,v 1.47 2004/08/27 03:08:23 navaraf Exp $
|
||||||
|
|
||||||
PATH_TO_TOP = ../..
|
PATH_TO_TOP = ../..
|
||||||
|
|
||||||
|
@ -454,7 +454,6 @@ TIME_OBJECTS = \
|
||||||
WINE_OBJECTS = \
|
WINE_OBJECTS = \
|
||||||
wine/cpp.o \
|
wine/cpp.o \
|
||||||
wine/cppexcept.o \
|
wine/cppexcept.o \
|
||||||
wine/data.o \
|
|
||||||
wine/heap.o \
|
wine/heap.o \
|
||||||
wine/lock.o \
|
wine/lock.o \
|
||||||
wine/main.o \
|
wine/main.o \
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: dllmain.c,v 1.23 2004/08/15 18:16:37 chorns Exp $
|
/* $Id: dllmain.c,v 1.24 2004/08/27 03:08:23 navaraf Exp $
|
||||||
*
|
*
|
||||||
* dllmain.c
|
* dllmain.c
|
||||||
*
|
*
|
||||||
|
@ -14,9 +14,9 @@
|
||||||
* DISCLAMED. This includes but is not limited to warrenties of
|
* DISCLAMED. This includes but is not limited to warrenties of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
*
|
*
|
||||||
* $Revision: 1.23 $
|
* $Revision: 1.24 $
|
||||||
* $Author: chorns $
|
* $Author: navaraf $
|
||||||
* $Date: 2004/08/15 18:16:37 $
|
* $Date: 2004/08/27 03:08:23 $
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -33,17 +33,22 @@
|
||||||
|
|
||||||
//void __fileno_init(void);
|
//void __fileno_init(void);
|
||||||
extern BOOL __fileno_init(void);
|
extern BOOL __fileno_init(void);
|
||||||
extern int BlockEnvToEnviron(void);
|
extern int BlockEnvToEnvironA(void);
|
||||||
|
extern int BlockEnvToEnvironW(void);
|
||||||
|
extern void FreeEnvironment(char **environment);
|
||||||
|
|
||||||
extern unsigned int _osver;
|
extern unsigned int _osver;
|
||||||
extern unsigned int _winminor;
|
extern unsigned int _winminor;
|
||||||
extern unsigned int _winmajor;
|
extern unsigned int _winmajor;
|
||||||
extern unsigned int _winver;
|
extern unsigned int _winver;
|
||||||
|
|
||||||
extern char* _acmdln; /* pointer to ascii command line */
|
extern char* _acmdln; /* pointer to ascii command line */
|
||||||
|
extern wchar_t* _wcmdln; /* pointer to wide character command line */
|
||||||
#undef _environ
|
#undef _environ
|
||||||
extern char** _environ; /* pointer to environment block */
|
extern char** _environ; /* pointer to environment block */
|
||||||
extern char** __initenv; /* pointer to initial environment block */
|
extern char** __initenv; /* pointer to initial environment block */
|
||||||
|
extern wchar_t** _wenviron; /* pointer to environment block */
|
||||||
|
extern wchar_t** __winitenv; /* pointer to initial environment block */
|
||||||
|
|
||||||
|
|
||||||
/* LIBRARY GLOBAL VARIABLES ***************************************************/
|
/* LIBRARY GLOBAL VARIABLES ***************************************************/
|
||||||
|
@ -77,17 +82,22 @@ DllMain(PVOID hinstDll, ULONG dwReason, PVOID reserved)
|
||||||
if (!CreateThreadData())
|
if (!CreateThreadData())
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
_acmdln = strdup(GetCommandLineA());
|
if (BlockEnvToEnvironA() < 0)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
/* FIXME: This crashes all applications */
|
if (BlockEnvToEnvironW() < 0)
|
||||||
if (BlockEnvToEnviron() < 0)
|
{
|
||||||
return FALSE;
|
FreeEnvironment((char**)_wenviron);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
_acmdln = strdup(GetCommandLineA());
|
||||||
|
_wcmdln = wcsdup(GetCommandLineW());
|
||||||
|
|
||||||
/* FIXME: more initializations... */
|
/* FIXME: more initializations... */
|
||||||
|
|
||||||
/* FIXME: Initialization of the WINE code */
|
/* FIXME: Initialization of the WINE code */
|
||||||
msvcrt_init_mt_locks();
|
msvcrt_init_mt_locks();
|
||||||
msvcrt_init_args();
|
|
||||||
|
|
||||||
DPRINT("Attach done\n");
|
DPRINT("Attach done\n");
|
||||||
break;
|
break;
|
||||||
|
@ -107,16 +117,16 @@ DllMain(PVOID hinstDll, ULONG dwReason, PVOID reserved)
|
||||||
/* destroy tls stuff */
|
/* destroy tls stuff */
|
||||||
DestroyThreadData();
|
DestroyThreadData();
|
||||||
|
|
||||||
|
if (__winitenv && __winitenv != _wenviron)
|
||||||
|
FreeEnvironment((char**)__winitenv);
|
||||||
|
if (_wenviron)
|
||||||
|
FreeEnvironment((char**)_wenviron);
|
||||||
|
|
||||||
if (__initenv && __initenv != _environ)
|
if (__initenv && __initenv != _environ)
|
||||||
{
|
FreeEnvironment(__initenv);
|
||||||
free(__initenv[0]);
|
|
||||||
free(__initenv);
|
|
||||||
}
|
|
||||||
if (_environ)
|
if (_environ)
|
||||||
{
|
FreeEnvironment(_environ);
|
||||||
free(_environ[0]);
|
|
||||||
free(_environ);
|
|
||||||
}
|
|
||||||
/* destroy heap */
|
/* destroy heap */
|
||||||
HeapDestroy(hHeap);
|
HeapDestroy(hHeap);
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: environ.c,v 1.9 2004/08/15 18:16:37 chorns Exp $
|
/* $Id: environ.c,v 1.10 2004/08/27 03:08:23 navaraf Exp $
|
||||||
*
|
*
|
||||||
* dllmain.c
|
* dllmain.c
|
||||||
*
|
*
|
||||||
|
@ -19,20 +19,334 @@ unsigned int _winminor = 0;
|
||||||
unsigned int _winmajor = 0;
|
unsigned int _winmajor = 0;
|
||||||
unsigned int _winver = 0;
|
unsigned int _winver = 0;
|
||||||
|
|
||||||
char *_acmdln = NULL; /* pointer to ascii command line */
|
char *_acmdln = NULL; /* pointer to ascii command line */
|
||||||
unsigned _envvar_count; /* number of environment vars within current environment */
|
wchar_t *_wcmdln = NULL; /* pointer to wide character command line */
|
||||||
#undef _environ
|
#undef _environ
|
||||||
char **_environ = NULL; /* pointer to environment block */
|
#undef _wenviron
|
||||||
char ***_environ_dll = &_environ;/* pointer to environment block */
|
char **_environ = NULL; /* pointer to environment block */
|
||||||
char **__initenv = NULL;
|
wchar_t **_wenviron = NULL; /* pointer to environment block */
|
||||||
|
char **__initenv = NULL; /* pointer to initial environment block */
|
||||||
|
wchar_t **__winitenv = NULL; /* pointer to initial environment block */
|
||||||
#undef _pgmptr
|
#undef _pgmptr
|
||||||
char *_pgmptr = NULL; /* pointer to program name */
|
char *_pgmptr = NULL; /* pointer to program name */
|
||||||
int __app_type = 0; //_UNKNOWN_APP; /* application type */
|
int __app_type = 0; //_UNKNOWN_APP; /* application type */
|
||||||
int __mb_cur_max = 1;
|
int __mb_cur_max = 1;
|
||||||
|
|
||||||
int _commode = _IOCOMMIT;
|
int _commode = _IOCOMMIT;
|
||||||
|
|
||||||
|
|
||||||
|
int BlockEnvToEnvironA(void)
|
||||||
|
{
|
||||||
|
char *ptr, *environment_strings;
|
||||||
|
char **envptr;
|
||||||
|
int count = 1, len;
|
||||||
|
|
||||||
|
DPRINT("BlockEnvToEnvironA()\n");
|
||||||
|
|
||||||
|
environment_strings = GetEnvironmentStringsA();
|
||||||
|
if (environment_strings == NULL) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ptr = environment_strings; *ptr; ptr += len)
|
||||||
|
{
|
||||||
|
len = strlen(ptr) + 1;
|
||||||
|
/* Skip drive letter settings. */
|
||||||
|
if (*ptr != '=')
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
_environ = HeapAlloc(GetProcessHeap(), 0, count * sizeof(char*));
|
||||||
|
if (_environ)
|
||||||
|
{
|
||||||
|
for (ptr = environment_strings, envptr = _environ; count > 1; ptr += len)
|
||||||
|
{
|
||||||
|
len = strlen(ptr) + 1;
|
||||||
|
/* Skip drive letter settings. */
|
||||||
|
if (*ptr != '=')
|
||||||
|
{
|
||||||
|
if ((*envptr = malloc(len)) == NULL)
|
||||||
|
{
|
||||||
|
for (envptr--; envptr >= _environ; envptr--);
|
||||||
|
free(*envptr);
|
||||||
|
FreeEnvironmentStringsA(environment_strings);
|
||||||
|
HeapFree(GetProcessHeap(), 0, _environ);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
memcpy(*envptr++, ptr, len);
|
||||||
|
count--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* Add terminating NULL entry. */
|
||||||
|
*envptr = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
FreeEnvironmentStringsA(environment_strings);
|
||||||
|
return _environ ? 0 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int BlockEnvToEnvironW(void)
|
||||||
|
{
|
||||||
|
wchar_t *ptr, *environment_strings;
|
||||||
|
wchar_t **envptr;
|
||||||
|
int count = 1, len;
|
||||||
|
|
||||||
|
DPRINT("BlockEnvToEnvironW()\n");
|
||||||
|
|
||||||
|
environment_strings = GetEnvironmentStringsW();
|
||||||
|
if (environment_strings == NULL) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ptr = environment_strings; *ptr; ptr += len)
|
||||||
|
{
|
||||||
|
len = wcslen(ptr) + 1;
|
||||||
|
/* Skip drive letter settings. */
|
||||||
|
if (*ptr != '=')
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
_wenviron = HeapAlloc(GetProcessHeap(), 0, count * sizeof(wchar_t*));
|
||||||
|
if (_wenviron)
|
||||||
|
{
|
||||||
|
for (ptr = environment_strings, envptr = _wenviron; count > 1; ptr += len)
|
||||||
|
{
|
||||||
|
len = wcslen(ptr) + 1;
|
||||||
|
/* Skip drive letter settings. */
|
||||||
|
if (*ptr != '=')
|
||||||
|
{
|
||||||
|
if ((*envptr = malloc(len * sizeof(wchar_t))) == NULL)
|
||||||
|
{
|
||||||
|
for (envptr--; envptr >= _wenviron; envptr--);
|
||||||
|
free(*envptr);
|
||||||
|
FreeEnvironmentStringsW(environment_strings);
|
||||||
|
HeapFree(GetProcessHeap(), 0, _wenviron);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
memcpy(*envptr++, ptr, len * sizeof(wchar_t));
|
||||||
|
count--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* Add terminating NULL entry. */
|
||||||
|
*envptr = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
FreeEnvironmentStringsW(environment_strings);
|
||||||
|
return _wenviron ? 0 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal function to duplicate environment block. Although it's
|
||||||
|
* parameter are defined as char**, it's able to work also with
|
||||||
|
* wide character environment block which are of type wchar_t**.
|
||||||
|
*
|
||||||
|
* @param original_environment
|
||||||
|
* Environment to duplicate.
|
||||||
|
* @param wide
|
||||||
|
* Set to zero for multibyte environments, non-zero otherwise.
|
||||||
|
*
|
||||||
|
* @return Original environment in case of failure, otherwise
|
||||||
|
* pointer to new environment block.
|
||||||
|
*/
|
||||||
|
char **DuplicateEnvironment(char **original_environment, int wide)
|
||||||
|
{
|
||||||
|
int count = 1;
|
||||||
|
char **envptr, **newenvptr, **newenv;
|
||||||
|
|
||||||
|
for (envptr = original_environment; *envptr != NULL; envptr++, count++)
|
||||||
|
;
|
||||||
|
|
||||||
|
newenvptr = newenv = HeapAlloc(GetProcessHeap(), 0, count * sizeof(char*));
|
||||||
|
if (newenv == NULL)
|
||||||
|
return original_environment;
|
||||||
|
|
||||||
|
for (envptr = original_environment; count > 1; newenvptr++, count--)
|
||||||
|
{
|
||||||
|
if (wide)
|
||||||
|
*newenvptr = (char*)wcsdup((wchar_t*)*envptr++);
|
||||||
|
else
|
||||||
|
*newenvptr = strdup(*envptr++);
|
||||||
|
if (*newenvptr == NULL)
|
||||||
|
{
|
||||||
|
for (newenvptr--; newenvptr >= newenv; newenvptr--);
|
||||||
|
free(*newenvptr);
|
||||||
|
HeapFree(GetProcessHeap(), 0, newenv);
|
||||||
|
return original_environment;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return newenv;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal function to deallocate environment block. Although it's
|
||||||
|
* parameter are defined as char**, it's able to work also with
|
||||||
|
* wide character environment block which are of type wchar_t**.
|
||||||
|
*
|
||||||
|
* @param environment
|
||||||
|
* Environment to free.
|
||||||
|
*/
|
||||||
|
void FreeEnvironment(char **environment)
|
||||||
|
{
|
||||||
|
char **envptr;
|
||||||
|
for (envptr = environment; *envptr != NULL; envptr++)
|
||||||
|
free(*envptr);
|
||||||
|
HeapFree(GetProcessHeap(), 0, environment);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal version of _wputenv and _putenv. It works duplicates the
|
||||||
|
* original envirnments created during initilization if needed to prevent
|
||||||
|
* having spurious pointers floating around. Then it updates the internal
|
||||||
|
* environment tables (_environ and _wenviron) and at last updates the
|
||||||
|
* OS environemnt.
|
||||||
|
*
|
||||||
|
* Note that there can happen situation when the internal [_w]environ
|
||||||
|
* arrays will be updated, but the OS environment update will fail. In
|
||||||
|
* this case we don't undo the changes to the [_w]environ tables to
|
||||||
|
* comply with the Microsoft behaviour (and it's also much easier :-).
|
||||||
|
*/
|
||||||
|
int SetEnv(const wchar_t *option)
|
||||||
|
{
|
||||||
|
wchar_t *epos, *name;
|
||||||
|
wchar_t **wenvptr;
|
||||||
|
wchar_t *woption;
|
||||||
|
char *mboption;
|
||||||
|
int remove, index, count, size, result = 0, found = 0;
|
||||||
|
|
||||||
|
if (option == NULL || (epos = wcschr(option, L'=')) == NULL)
|
||||||
|
return -1;
|
||||||
|
remove = (epos[1] == 0);
|
||||||
|
|
||||||
|
/* Duplicate environment if needed. */
|
||||||
|
if (_environ == __initenv)
|
||||||
|
{
|
||||||
|
if ((_environ = DuplicateEnvironment(_environ, 0)) == __initenv)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (_wenviron == __winitenv)
|
||||||
|
{
|
||||||
|
if ((_wenviron = (wchar_t**)DuplicateEnvironment((char**)_wenviron, 1)) ==
|
||||||
|
__winitenv)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create a copy of the option name. */
|
||||||
|
name = malloc(epos - option + 1);
|
||||||
|
if (name == NULL)
|
||||||
|
return -1;
|
||||||
|
memcpy(name, option, (epos - option) * sizeof(wchar_t));
|
||||||
|
name[epos - option] = 0;
|
||||||
|
|
||||||
|
/* Find the option we're trying to modify. */
|
||||||
|
for (index = 0, wenvptr = _wenviron; *wenvptr != NULL; wenvptr++, index++)
|
||||||
|
{
|
||||||
|
if (!wcsnicmp(*wenvptr, option, epos - option))
|
||||||
|
{
|
||||||
|
found = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (remove)
|
||||||
|
{
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
free(name);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove the option from wide character environment. */
|
||||||
|
free(*wenvptr);
|
||||||
|
for (count = index; *wenvptr != NULL; wenvptr++, count++)
|
||||||
|
*wenvptr = *(wenvptr + 1);
|
||||||
|
_wenviron = HeapReAlloc(GetProcessHeap(), 0, _wenviron,
|
||||||
|
count * sizeof(wchar_t*));
|
||||||
|
|
||||||
|
/* Remove the option from multibyte environment. We assume
|
||||||
|
* the environments are in sync and the option is at the
|
||||||
|
* same position. */
|
||||||
|
free(_environ[index]);
|
||||||
|
for (; _environ[index] != NULL; index++)
|
||||||
|
_environ[index] = _environ[index + 1];
|
||||||
|
_environ = HeapReAlloc(GetProcessHeap(), 0, _environ,
|
||||||
|
count * sizeof(char*));
|
||||||
|
|
||||||
|
result = SetEnvironmentVariableW(name, NULL) ? 0 : -1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Make a copy of the option that we will store in the environment block. */
|
||||||
|
woption = wcsdup((wchar_t*)option);
|
||||||
|
if (woption == NULL)
|
||||||
|
{
|
||||||
|
free(name);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create a multibyte copy of the option. */
|
||||||
|
size = WideCharToMultiByte(CP_ACP, 0, option, 0, NULL, 0, NULL, NULL);
|
||||||
|
mboption = malloc(size);
|
||||||
|
if (mboption == NULL)
|
||||||
|
{
|
||||||
|
free(name);
|
||||||
|
free(woption);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
WideCharToMultiByte(CP_ACP, 0, option, 0, mboption, size, NULL, NULL);
|
||||||
|
|
||||||
|
if (found)
|
||||||
|
{
|
||||||
|
/* Replace the current entry. */
|
||||||
|
free(*wenvptr);
|
||||||
|
*wenvptr = woption;
|
||||||
|
free(_environ[index]);
|
||||||
|
_environ[index] = mboption;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
wchar_t **wnewenv;
|
||||||
|
char **mbnewenv;
|
||||||
|
|
||||||
|
/* Get the size of the original environment. */
|
||||||
|
for (count = index; *wenvptr != NULL; wenvptr++, count++)
|
||||||
|
;
|
||||||
|
|
||||||
|
/* Create a new entry. */
|
||||||
|
if ((wnewenv = HeapReAlloc(GetProcessHeap(), 0, _wenviron,
|
||||||
|
(count + 2) * sizeof(wchar_t*))) == NULL)
|
||||||
|
{
|
||||||
|
free(name);
|
||||||
|
free(mboption);
|
||||||
|
free(woption);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
_wenviron = wnewenv;
|
||||||
|
if ((mbnewenv = HeapReAlloc(GetProcessHeap(), 0, _environ,
|
||||||
|
(count + 2) * sizeof(char*))) == NULL)
|
||||||
|
{
|
||||||
|
free(name);
|
||||||
|
free(mboption);
|
||||||
|
free(woption);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
_environ = mbnewenv;
|
||||||
|
|
||||||
|
/* Set the last entry to our option. */
|
||||||
|
_wenviron[count] = woption;
|
||||||
|
_environ[count] = mboption;
|
||||||
|
_wenviron[count + 1] = NULL;
|
||||||
|
_environ[count + 1] = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* And finally update the OS environment. */
|
||||||
|
result = SetEnvironmentVariableW(name, epos + 1) ? 0 : -1;
|
||||||
|
}
|
||||||
|
free(name);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
|
@ -41,79 +355,6 @@ int *__p__commode(void) // not exported by NTDLL
|
||||||
return &_commode;
|
return &_commode;
|
||||||
}
|
}
|
||||||
|
|
||||||
int BlockEnvToEnviron(void)
|
|
||||||
{
|
|
||||||
char * ptr, * ptr2;
|
|
||||||
int i, count, len, size;
|
|
||||||
|
|
||||||
DPRINT("BlockEnvToEnviron()\n");
|
|
||||||
|
|
||||||
ptr2 = ptr = (char*)GetEnvironmentStringsA();
|
|
||||||
if (ptr == NULL) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
size = 0;
|
|
||||||
count = 0;
|
|
||||||
while (*ptr2) {
|
|
||||||
len = strlen(ptr2);
|
|
||||||
if (*ptr2 != '=') {
|
|
||||||
count++;
|
|
||||||
size += len + 1;
|
|
||||||
}
|
|
||||||
ptr2 += len + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count != _envvar_count) {
|
|
||||||
if (_environ && _environ != __initenv) {
|
|
||||||
free(_environ[0]);
|
|
||||||
_environ = realloc(_environ, (count + 1) * sizeof(char*));
|
|
||||||
} else {
|
|
||||||
_environ = malloc((count + 1) * sizeof(char*));
|
|
||||||
}
|
|
||||||
if (_environ == NULL) {
|
|
||||||
FreeEnvironmentStringsA(ptr);
|
|
||||||
_envvar_count = 0;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
_environ[0] = NULL;
|
|
||||||
}
|
|
||||||
if (_environ[0] != NULL) {
|
|
||||||
free(_environ[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
_environ[0] = malloc(size);
|
|
||||||
if (_environ[0] == NULL) {
|
|
||||||
FreeEnvironmentStringsA(ptr);
|
|
||||||
free(_environ);
|
|
||||||
_envvar_count = 0;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
ptr2 = ptr;
|
|
||||||
i = 0;
|
|
||||||
while (*ptr2 && i < count) {
|
|
||||||
len = strlen(ptr2);
|
|
||||||
/* skip current directory of the form "=C:=C:\directory\" */
|
|
||||||
if (*ptr2 != '=') {
|
|
||||||
memcpy(_environ[i], ptr2, len + 1);
|
|
||||||
i++;
|
|
||||||
if (i < count) {
|
|
||||||
_environ[i] = _environ[i - 1] + len + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ptr2 += len + 1;
|
|
||||||
}
|
|
||||||
_environ[i] = NULL;
|
|
||||||
_envvar_count = count;
|
|
||||||
if (__initenv == NULL)
|
|
||||||
{
|
|
||||||
__initenv = _environ;
|
|
||||||
}
|
|
||||||
FreeEnvironmentStringsA(ptr);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
|
@ -135,7 +376,15 @@ char **__p__acmdln(void)
|
||||||
*/
|
*/
|
||||||
char ***__p__environ(void)
|
char ***__p__environ(void)
|
||||||
{
|
{
|
||||||
return _environ_dll;
|
return &_environ;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @implemented
|
||||||
|
*/
|
||||||
|
wchar_t ***__p__wenviron(void)
|
||||||
|
{
|
||||||
|
return &_wenviron;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -146,6 +395,14 @@ char ***__p___initenv(void)
|
||||||
return &__initenv;
|
return &__initenv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @implemented
|
||||||
|
*/
|
||||||
|
wchar_t ***__p___winitenv(void)
|
||||||
|
{
|
||||||
|
return &__winitenv;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -12,6 +12,8 @@ extern char**_environ;
|
||||||
#undef __argc
|
#undef __argc
|
||||||
|
|
||||||
char**__argv = NULL;
|
char**__argv = NULL;
|
||||||
|
#undef __wargv
|
||||||
|
wchar_t**__wargv = NULL;
|
||||||
int __argc = 0;
|
int __argc = 0;
|
||||||
|
|
||||||
extern HANDLE hHeap;
|
extern HANDLE hHeap;
|
||||||
|
@ -21,6 +23,7 @@ char* strndup(char* name, int len)
|
||||||
char *s = malloc(len + 1);
|
char *s = malloc(len + 1);
|
||||||
if (s != NULL) {
|
if (s != NULL) {
|
||||||
strncpy(s, name, len);
|
strncpy(s, name, len);
|
||||||
|
name[len] = 0;
|
||||||
}
|
}
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
@ -31,14 +34,12 @@ int add(char* name)
|
||||||
{
|
{
|
||||||
char** _new;
|
char** _new;
|
||||||
if ((__argc % SIZE) == 0) {
|
if ((__argc % SIZE) == 0) {
|
||||||
_new = malloc(sizeof(char*) * (__argc + SIZE));
|
if (__argv == NULL)
|
||||||
if (_new == NULL) {
|
_new = malloc(sizeof(char*) * SIZE);
|
||||||
|
else
|
||||||
|
_new = realloc(__argv, sizeof(char*) * (__argc + SIZE));
|
||||||
|
if (_new == NULL)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
|
||||||
if (__argv) {
|
|
||||||
memcpy(_new, __argv, sizeof(char*) * __argc);
|
|
||||||
free(__argv);
|
|
||||||
}
|
|
||||||
__argv = _new;
|
__argv = _new;
|
||||||
}
|
}
|
||||||
__argv[__argc++] = name;
|
__argv[__argc++] = name;
|
||||||
|
@ -139,6 +140,18 @@ int __getmainargs(int* argc, char*** argv, char*** env, int flag)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @unimplemented
|
||||||
|
*/
|
||||||
|
void __wgetmainargs(int* argc, wchar_t*** wargv, wchar_t*** wenv,
|
||||||
|
int expand_wildcards, int* new_mode)
|
||||||
|
{
|
||||||
|
extern wchar_t **__winitenv;
|
||||||
|
*argc = 0;
|
||||||
|
*wargv = NULL;
|
||||||
|
*wenv = __winitenv;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
; $Id: msvcrt.def,v 1.35 2004/05/15 12:55:47 jfilby Exp $
|
; $Id: msvcrt.def,v 1.36 2004/08/27 03:08:23 navaraf Exp $
|
||||||
;
|
;
|
||||||
; ReactOS MSVCRT Compatibility Library
|
; ReactOS MSVCRT Compatibility Library
|
||||||
;
|
;
|
||||||
|
@ -129,7 +129,7 @@ __p___argv
|
||||||
__p___wargv
|
__p___wargv
|
||||||
__p___initenv
|
__p___initenv
|
||||||
__p___mb_cur_max
|
__p___mb_cur_max
|
||||||
;__p___winitenv
|
__p___winitenv
|
||||||
__p__acmdln
|
__p__acmdln
|
||||||
;__p__amblksiz
|
;__p__amblksiz
|
||||||
__p__commode
|
__p__commode
|
||||||
|
@ -148,7 +148,7 @@ __p__pwctype
|
||||||
;__p__timezone
|
;__p__timezone
|
||||||
;__p__tzname
|
;__p__tzname
|
||||||
;__p__wcmdln
|
;__p__wcmdln
|
||||||
;__p__wenviron
|
__p__wenviron
|
||||||
__p__winmajor
|
__p__winmajor
|
||||||
__p__winminor
|
__p__winminor
|
||||||
__p__winver
|
__p__winver
|
||||||
|
@ -164,9 +164,9 @@ __toascii
|
||||||
__unDName
|
__unDName
|
||||||
;__unDNameEx
|
;__unDNameEx
|
||||||
;__unguarded_readlc_active
|
;__unguarded_readlc_active
|
||||||
;__wargv
|
__wargv
|
||||||
__wgetmainargs
|
__wgetmainargs
|
||||||
;__winitenv
|
__winitenv
|
||||||
;___lc_codepage_func
|
;___lc_codepage_func
|
||||||
;___lc_handle_func
|
;___lc_handle_func
|
||||||
;___mb_cur_max_func
|
;___mb_cur_max_func
|
||||||
|
@ -246,7 +246,6 @@ _dup2
|
||||||
_ecvt
|
_ecvt
|
||||||
_endthread
|
_endthread
|
||||||
_endthreadex
|
_endthreadex
|
||||||
_environ_dll DATA
|
|
||||||
;_environ
|
;_environ
|
||||||
_eof
|
_eof
|
||||||
_errno
|
_errno
|
||||||
|
@ -561,7 +560,7 @@ _waccess
|
||||||
_wasctime
|
_wasctime
|
||||||
_wchdir
|
_wchdir
|
||||||
_wchmod
|
_wchmod
|
||||||
_wcmdln=MSVCRT__wcmdln
|
_wcmdln
|
||||||
_wcreat
|
_wcreat
|
||||||
_wcsdup
|
_wcsdup
|
||||||
;_wcserror
|
;_wcserror
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: _system.c,v 1.10 2004/08/15 18:16:37 chorns Exp $
|
/* $Id: _system.c,v 1.11 2004/08/27 03:08:23 navaraf Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT: ReactOS system libraries
|
* PROJECT: ReactOS system libraries
|
||||||
|
@ -39,10 +39,7 @@ int system(const char *command)
|
||||||
if (szComSpec == NULL)
|
if (szComSpec == NULL)
|
||||||
return 0;
|
return 0;
|
||||||
else
|
else
|
||||||
{
|
|
||||||
free(szComSpec);
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// should return 127 or 0 ( MS ) if the shell is not found
|
// should return 127 or 0 ( MS ) if the shell is not found
|
||||||
|
|
|
@ -85,7 +85,6 @@ const char* find_exec(const char* path, char* rpath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
free(env);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,12 +30,9 @@ FILE *_popen (const char *cm, const char *md) /* program name, pipe mode */
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
szComSpec = getenv("COMSPEC");
|
szComSpec = getenv("COMSPEC");
|
||||||
|
|
||||||
if (szComSpec == NULL)
|
if (szComSpec == NULL)
|
||||||
{
|
{
|
||||||
szComSpec = strdup("cmd.exe");
|
szComSpec = "cmd.exe";
|
||||||
if (szComSpec == NULL)
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
s = max(strrchr(szComSpec, '\\'), strrchr(szComSpec, '/'));
|
s = max(strrchr(szComSpec, '\\'), strrchr(szComSpec, '/'));
|
||||||
|
@ -47,7 +44,6 @@ FILE *_popen (const char *cm, const char *md) /* program name, pipe mode */
|
||||||
szCmdLine = malloc(strlen(s) + 4 + strlen(cm) + 1);
|
szCmdLine = malloc(strlen(s) + 4 + strlen(cm) + 1);
|
||||||
if (szCmdLine == NULL)
|
if (szCmdLine == NULL)
|
||||||
{
|
{
|
||||||
free (szComSpec);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,7 +56,6 @@ FILE *_popen (const char *cm, const char *md) /* program name, pipe mode */
|
||||||
|
|
||||||
if ( !CreatePipe(&hReadPipe,&hWritePipe,&sa,1024))
|
if ( !CreatePipe(&hReadPipe,&hWritePipe,&sa,1024))
|
||||||
{
|
{
|
||||||
free (szComSpec);
|
|
||||||
free (szCmdLine);
|
free (szCmdLine);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -87,7 +82,6 @@ FILE *_popen (const char *cm, const char *md) /* program name, pipe mode */
|
||||||
NULL,
|
NULL,
|
||||||
&StartupInfo,
|
&StartupInfo,
|
||||||
&ProcessInformation);
|
&ProcessInformation);
|
||||||
free (szComSpec);
|
|
||||||
free (szCmdLine);
|
free (szCmdLine);
|
||||||
|
|
||||||
if (result == FALSE)
|
if (result == FALSE)
|
||||||
|
@ -152,9 +146,7 @@ FILE *_wpopen (const wchar_t *cm, const wchar_t *md) /* program name, pipe mode
|
||||||
|
|
||||||
if (szComSpec == NULL)
|
if (szComSpec == NULL)
|
||||||
{
|
{
|
||||||
szComSpec = _wcsdup(L"cmd.exe");
|
szComSpec = L"cmd.exe";
|
||||||
if (szComSpec == NULL)
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
s = max(wcsrchr(szComSpec, L'\\'), wcsrchr(szComSpec, L'/'));
|
s = max(wcsrchr(szComSpec, L'\\'), wcsrchr(szComSpec, L'/'));
|
||||||
|
@ -166,7 +158,6 @@ FILE *_wpopen (const wchar_t *cm, const wchar_t *md) /* program name, pipe mode
|
||||||
szCmdLine = malloc((wcslen(s) + 4 + wcslen(cm) + 1) * sizeof(wchar_t));
|
szCmdLine = malloc((wcslen(s) + 4 + wcslen(cm) + 1) * sizeof(wchar_t));
|
||||||
if (szCmdLine == NULL)
|
if (szCmdLine == NULL)
|
||||||
{
|
{
|
||||||
free (szComSpec);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -179,7 +170,6 @@ FILE *_wpopen (const wchar_t *cm, const wchar_t *md) /* program name, pipe mode
|
||||||
|
|
||||||
if ( !CreatePipe(&hReadPipe,&hWritePipe,&sa,1024))
|
if ( !CreatePipe(&hReadPipe,&hWritePipe,&sa,1024))
|
||||||
{
|
{
|
||||||
free (szComSpec);
|
|
||||||
free (szCmdLine);
|
free (szCmdLine);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -206,7 +196,6 @@ FILE *_wpopen (const wchar_t *cm, const wchar_t *md) /* program name, pipe mode
|
||||||
NULL,
|
NULL,
|
||||||
&StartupInfo,
|
&StartupInfo,
|
||||||
&ProcessInformation);
|
&ProcessInformation);
|
||||||
free (szComSpec);
|
|
||||||
free (szCmdLine);
|
free (szCmdLine);
|
||||||
|
|
||||||
if (result == FALSE)
|
if (result == FALSE)
|
||||||
|
|
|
@ -4,23 +4,24 @@
|
||||||
#define NDEBUG
|
#define NDEBUG
|
||||||
#include <msvcrt/msvcrtdbg.h>
|
#include <msvcrt/msvcrtdbg.h>
|
||||||
|
|
||||||
|
#undef environ
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
char *getenv(const char *name)
|
char *getenv(const char *name)
|
||||||
{
|
{
|
||||||
char *buffer = (char*)0xffffffff;
|
char **environ;
|
||||||
int len = GetEnvironmentVariableA(name,buffer,0) + 1;
|
unsigned int length = strlen(name);
|
||||||
DPRINT("getenv(%s)\n", name);
|
|
||||||
buffer = (char *)malloc(len);
|
for (environ = *__p__environ(); *environ; environ++)
|
||||||
DPRINT("getenv('%s') %d %x\n", name, len, buffer);
|
{
|
||||||
if (buffer == NULL || GetEnvironmentVariableA(name,buffer,len) == 0 )
|
char *str = *environ;
|
||||||
{
|
char *pos = strchr(str,'=');
|
||||||
free(buffer);
|
if (pos && ((pos - str) == length) && !strnicmp(str, name, length))
|
||||||
return NULL;
|
return pos + 1;
|
||||||
}
|
}
|
||||||
return buffer;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -28,14 +29,15 @@ char *getenv(const char *name)
|
||||||
*/
|
*/
|
||||||
wchar_t *_wgetenv(const wchar_t *name)
|
wchar_t *_wgetenv(const wchar_t *name)
|
||||||
{
|
{
|
||||||
wchar_t *buffer = (wchar_t*)0xffffffff;
|
wchar_t **environ;
|
||||||
int len = GetEnvironmentVariableW(name, buffer,0) + 1;
|
unsigned int length = wcslen(name);
|
||||||
DPRINT("_wgetenv(%S)\n", name);
|
|
||||||
buffer = (wchar_t *)malloc(len * sizeof(wchar_t));
|
for (environ = *__p__wenviron(); *environ; environ++)
|
||||||
if (buffer == NULL || GetEnvironmentVariableW(name,buffer,len) == 0)
|
{
|
||||||
{
|
wchar_t *str = *environ;
|
||||||
free(buffer);
|
wchar_t *pos = wcschr(str, L'=');
|
||||||
return NULL;
|
if (pos && ((pos - str) == length) && !wcsnicmp(str, name, length))
|
||||||
}
|
return pos + 1;
|
||||||
return buffer;
|
}
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,6 +56,8 @@ void* calloc(size_t _nmemb, size_t _size)
|
||||||
*/
|
*/
|
||||||
void* realloc(void* _ptr, size_t _size)
|
void* realloc(void* _ptr, size_t _size)
|
||||||
{
|
{
|
||||||
|
if (!_ptr)
|
||||||
|
return HeapAlloc(hHeap, 0, _size);
|
||||||
return HeapReAlloc(hHeap, 0, _ptr, _size);
|
return HeapReAlloc(hHeap, 0, _ptr, _size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,30 +5,23 @@
|
||||||
#define NDEBUG
|
#define NDEBUG
|
||||||
#include <msvcrt/msvcrtdbg.h>
|
#include <msvcrt/msvcrtdbg.h>
|
||||||
|
|
||||||
|
/* misc/environ.c */
|
||||||
extern int BlockEnvToEnviron(); // defined in misc/dllmain.c
|
int SetEnv(const wchar_t *option);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
int _putenv(const char* val)
|
int _putenv(const char* val)
|
||||||
{
|
{
|
||||||
char* buffer;
|
int size, result;
|
||||||
char* epos;
|
wchar_t *woption;
|
||||||
int res;
|
|
||||||
|
size = MultiByteToWideChar(CP_ACP, 0, val, 0, NULL, 0);
|
||||||
DPRINT("_putenv('%s')\n", val);
|
woption = malloc(size);
|
||||||
epos = strchr(val, '=');
|
if (woption == NULL)
|
||||||
if ( epos == NULL )
|
return -1;
|
||||||
return -1;
|
MultiByteToWideChar(CP_ACP, 0, val, 0, woption, size);
|
||||||
buffer = (char*)malloc(epos - val + 1);
|
result = SetEnv(woption);
|
||||||
if (buffer == NULL)
|
free(woption);
|
||||||
return -1;
|
return result;
|
||||||
strncpy(buffer, val, epos - val);
|
|
||||||
buffer[epos - val] = 0;
|
|
||||||
res = SetEnvironmentVariableA(buffer, epos+1);
|
|
||||||
free(buffer);
|
|
||||||
if (BlockEnvToEnviron())
|
|
||||||
return 0;
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,30 +5,13 @@
|
||||||
#define NDEBUG
|
#define NDEBUG
|
||||||
#include <msvcrt/msvcrtdbg.h>
|
#include <msvcrt/msvcrtdbg.h>
|
||||||
|
|
||||||
|
/* misc/environ.c */
|
||||||
extern int BlockEnvToEnviron(); // defined in misc/dllmain.c
|
int SetEnv(const wchar_t *option);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @implemented
|
* @implemented
|
||||||
*/
|
*/
|
||||||
int _wputenv(const wchar_t* val)
|
int _wputenv(const wchar_t* val)
|
||||||
{
|
{
|
||||||
wchar_t* buffer;
|
return SetEnv(val);
|
||||||
wchar_t* epos;
|
|
||||||
int res;
|
|
||||||
|
|
||||||
DPRINT("_wputenv('%S')\n", val);
|
|
||||||
epos = wcsrchr(val, L'=');
|
|
||||||
if (epos == NULL)
|
|
||||||
return -1;
|
|
||||||
buffer = (wchar_t*)malloc((epos - val + 1) * sizeof(wchar_t));
|
|
||||||
if (buffer == NULL)
|
|
||||||
return -1;
|
|
||||||
wcsncpy(buffer, val, epos - val);
|
|
||||||
buffer[epos - val] = 0;
|
|
||||||
res = SetEnvironmentVariableW(buffer, epos+1);
|
|
||||||
free(buffer);
|
|
||||||
if (BlockEnvToEnviron())
|
|
||||||
return 0;
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,369 +0,0 @@
|
||||||
/*
|
|
||||||
* msvcrt.dll dll data items
|
|
||||||
*
|
|
||||||
* Copyright 2000 Jon Griffiths
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "config.h"
|
|
||||||
#include "wine/port.h"
|
|
||||||
|
|
||||||
#include <math.h>
|
|
||||||
#include "msvcrt.h"
|
|
||||||
|
|
||||||
#include "stdlib.h"
|
|
||||||
#include "string.h"
|
|
||||||
|
|
||||||
//#include "wine/library.h"
|
|
||||||
#include "wine/unicode.h"
|
|
||||||
#include "wine/debug.h"
|
|
||||||
|
|
||||||
WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
|
|
||||||
|
|
||||||
unsigned int MSVCRT___argc;
|
|
||||||
unsigned int MSVCRT_basemajor;/* FIXME: */
|
|
||||||
unsigned int MSVCRT_baseminor;/* FIXME: */
|
|
||||||
unsigned int MSVCRT_baseversion; /* FIXME: */
|
|
||||||
unsigned int MSVCRT__commode;
|
|
||||||
unsigned int MSVCRT__fmode;
|
|
||||||
unsigned int MSVCRT_osmajor;/* FIXME: */
|
|
||||||
unsigned int MSVCRT_osminor;/* FIXME: */
|
|
||||||
unsigned int MSVCRT_osmode;/* FIXME: */
|
|
||||||
unsigned int MSVCRT__osver;
|
|
||||||
unsigned int MSVCRT_osversion; /* FIXME: */
|
|
||||||
unsigned int MSVCRT__winmajor;
|
|
||||||
unsigned int MSVCRT__winminor;
|
|
||||||
unsigned int MSVCRT__winver;
|
|
||||||
unsigned int MSVCRT__sys_nerr; /* FIXME: not accessible from Winelib apps */
|
|
||||||
char** MSVCRT__sys_errlist; /* FIXME: not accessible from Winelib apps */
|
|
||||||
unsigned int MSVCRT___setlc_active;
|
|
||||||
unsigned int MSVCRT___unguarded_readlc_active;
|
|
||||||
double MSVCRT__HUGE;
|
|
||||||
char **MSVCRT___argv;
|
|
||||||
MSVCRT_wchar_t **MSVCRT___wargv;
|
|
||||||
char *MSVCRT__acmdln;
|
|
||||||
MSVCRT_wchar_t *MSVCRT__wcmdln;
|
|
||||||
char **MSVCRT__environ = 0;
|
|
||||||
MSVCRT_wchar_t **MSVCRT__wenviron = 0;
|
|
||||||
char **MSVCRT___initenv = 0;
|
|
||||||
MSVCRT_wchar_t **MSVCRT___winitenv = 0;
|
|
||||||
int MSVCRT_timezone;
|
|
||||||
int MSVCRT_app_type;
|
|
||||||
char* MSVCRT__pgmptr = 0;
|
|
||||||
WCHAR* MSVCRT__wpgmptr = 0;
|
|
||||||
|
|
||||||
/* Get a snapshot of the current environment
|
|
||||||
* and construct the __p__environ array
|
|
||||||
*
|
|
||||||
* The pointer returned from GetEnvironmentStrings may get invalid when
|
|
||||||
* some other module cause a reallocation of the env-variable block
|
|
||||||
*
|
|
||||||
* blk is an array of pointers to environment strings, ending with a NULL
|
|
||||||
* and after that the actual copy of the environment strings, ending in a \0
|
|
||||||
*/
|
|
||||||
char ** msvcrt_SnapshotOfEnvironmentA(char **blk)
|
|
||||||
{
|
|
||||||
char* environ_strings = GetEnvironmentStringsA();
|
|
||||||
int count = 1, len = 1, i = 0; /* keep space for the trailing NULLS */
|
|
||||||
char *ptr;
|
|
||||||
|
|
||||||
for (ptr = environ_strings; *ptr; ptr += strlen(ptr) + 1)
|
|
||||||
{
|
|
||||||
count++;
|
|
||||||
len += strlen(ptr) + 1;
|
|
||||||
}
|
|
||||||
if (blk)
|
|
||||||
blk = HeapReAlloc( GetProcessHeap(), 0, blk, count* sizeof(char*) + len );
|
|
||||||
else
|
|
||||||
blk = HeapAlloc(GetProcessHeap(), 0, count* sizeof(char*) + len );
|
|
||||||
|
|
||||||
if (blk)
|
|
||||||
{
|
|
||||||
if (count)
|
|
||||||
{
|
|
||||||
memcpy(&blk[count],environ_strings,len);
|
|
||||||
for (ptr = (char*) &blk[count]; *ptr; ptr += strlen(ptr) + 1)
|
|
||||||
{
|
|
||||||
blk[i++] = ptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
blk[i] = NULL;
|
|
||||||
}
|
|
||||||
FreeEnvironmentStringsA(environ_strings);
|
|
||||||
return blk;
|
|
||||||
}
|
|
||||||
|
|
||||||
MSVCRT_wchar_t ** msvcrt_SnapshotOfEnvironmentW(MSVCRT_wchar_t **wblk)
|
|
||||||
{
|
|
||||||
MSVCRT_wchar_t* wenviron_strings = GetEnvironmentStringsW();
|
|
||||||
int count = 1, len = 1, i = 0; /* keep space for the trailing NULLS */
|
|
||||||
MSVCRT_wchar_t *wptr;
|
|
||||||
|
|
||||||
for (wptr = wenviron_strings; *wptr; wptr += strlenW(wptr) + 1)
|
|
||||||
{
|
|
||||||
count++;
|
|
||||||
len += strlenW(wptr) + 1;
|
|
||||||
}
|
|
||||||
if (wblk)
|
|
||||||
wblk = HeapReAlloc( GetProcessHeap(), 0, wblk, count* sizeof(MSVCRT_wchar_t*) + len * sizeof(MSVCRT_wchar_t));
|
|
||||||
else
|
|
||||||
wblk = HeapAlloc(GetProcessHeap(), 0, count* sizeof(MSVCRT_wchar_t*) + len * sizeof(MSVCRT_wchar_t));
|
|
||||||
if (wblk)
|
|
||||||
{
|
|
||||||
if (count)
|
|
||||||
{
|
|
||||||
memcpy(&wblk[count],wenviron_strings,len * sizeof(MSVCRT_wchar_t));
|
|
||||||
for (wptr = (MSVCRT_wchar_t*)&wblk[count]; *wptr; wptr += strlenW(wptr) + 1)
|
|
||||||
{
|
|
||||||
wblk[i++] = wptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
wblk[i] = NULL;
|
|
||||||
}
|
|
||||||
FreeEnvironmentStringsW(wenviron_strings);
|
|
||||||
return wblk;
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef void (*_INITTERMFUN)(void);
|
|
||||||
#ifndef __REACTOS__
|
|
||||||
/***********************************************************************
|
|
||||||
* __p___argc (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
int* __p___argc(void) { return &MSVCRT___argc; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__commode (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__commode(void) { return &MSVCRT__commode; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__pgmptr (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
char** __p__pgmptr(void) { return &MSVCRT__pgmptr; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__wpgmptr (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
WCHAR** __p__wpgmptr(void) { return &MSVCRT__wpgmptr; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__fmode (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__fmode(void) { return &MSVCRT__fmode; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__osver (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__osver(void) { return &MSVCRT__osver; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__winmajor (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__winmajor(void) { return &MSVCRT__winmajor; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__winminor (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__winminor(void) { return &MSVCRT__winminor; }
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* __p__winver (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int* __p__winver(void) { return &MSVCRT__winver; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p__acmdln (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
char** __p__acmdln(void) { return &MSVCRT__acmdln; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p__wcmdln (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
MSVCRT_wchar_t** __p__wcmdln(void) { return &MSVCRT__wcmdln; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p___argv (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
char*** __p___argv(void) { return &MSVCRT___argv; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p___wargv (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
MSVCRT_wchar_t*** __p___wargv(void) { return &MSVCRT___wargv; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p__environ (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
char*** __p__environ(void)
|
|
||||||
{
|
|
||||||
if (!MSVCRT__environ)
|
|
||||||
MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(NULL);
|
|
||||||
return &MSVCRT__environ;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p__wenviron (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
MSVCRT_wchar_t*** __p__wenviron(void)
|
|
||||||
{
|
|
||||||
if (!MSVCRT__wenviron)
|
|
||||||
MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(NULL);
|
|
||||||
return &MSVCRT__wenviron;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p___initenv (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
char*** __p___initenv(void) { return &MSVCRT___initenv; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p___winitenv (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
MSVCRT_wchar_t*** __p___winitenv(void) { return &MSVCRT___winitenv; }
|
|
||||||
|
|
||||||
/*********************************************************************
|
|
||||||
* __p__timezone (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
int* __p__timezone(void) { return &MSVCRT_timezone; }
|
|
||||||
#endif
|
|
||||||
/* INTERNAL: Create a wide string from an ascii string */
|
|
||||||
static MSVCRT_wchar_t *wstrdupa(const char *str)
|
|
||||||
{
|
|
||||||
const size_t len = strlen(str) + 1 ;
|
|
||||||
MSVCRT_wchar_t *wstr = malloc(len* sizeof (MSVCRT_wchar_t));
|
|
||||||
if (!wstr)
|
|
||||||
return NULL;
|
|
||||||
MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED,str,len,wstr,len);
|
|
||||||
return wstr;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* INTERNAL: Since we can't rely on Winelib startup code calling w/getmainargs,
|
|
||||||
* we initialise data values during DLL loading. When called by a native
|
|
||||||
* program we simply return the data we've already initialised. This also means
|
|
||||||
* you can call multiple times without leaking
|
|
||||||
*/
|
|
||||||
void msvcrt_init_args(void)
|
|
||||||
{
|
|
||||||
DWORD version;
|
|
||||||
|
|
||||||
MSVCRT__acmdln = _strdup( GetCommandLineA() );
|
|
||||||
MSVCRT__wcmdln = wstrdupa(MSVCRT__acmdln);
|
|
||||||
//MSVCRT___argc = __wine_main_argc;
|
|
||||||
//MSVCRT___argv = __wine_main_argv;
|
|
||||||
//MSVCRT___wargv = __wine_main_wargv;
|
|
||||||
|
|
||||||
TRACE("got '%s', wide = %s argc=%d\n", MSVCRT__acmdln,
|
|
||||||
debugstr_w(MSVCRT__wcmdln),MSVCRT___argc);
|
|
||||||
|
|
||||||
version = GetVersion();
|
|
||||||
MSVCRT__osver = version >> 16;
|
|
||||||
MSVCRT__winminor = version & 0xFF;
|
|
||||||
MSVCRT__winmajor = (version>>8) & 0xFF;
|
|
||||||
MSVCRT_baseversion = version >> 16;
|
|
||||||
MSVCRT__winver = ((version >> 8) & 0xFF) + ((version & 0xFF) << 8);
|
|
||||||
MSVCRT_baseminor = (version >> 16) & 0xFF;
|
|
||||||
MSVCRT_basemajor = (version >> 24) & 0xFF;
|
|
||||||
MSVCRT_osversion = version & 0xFFFF;
|
|
||||||
MSVCRT_osminor = version & 0xFF;
|
|
||||||
MSVCRT_osmajor = (version>>8) & 0xFF;
|
|
||||||
MSVCRT__sys_nerr = 43;
|
|
||||||
MSVCRT__HUGE = HUGE_VAL;
|
|
||||||
MSVCRT___setlc_active = 0;
|
|
||||||
MSVCRT___unguarded_readlc_active = 0;
|
|
||||||
MSVCRT_timezone = 0;
|
|
||||||
|
|
||||||
MSVCRT___initenv= msvcrt_SnapshotOfEnvironmentA(NULL);
|
|
||||||
MSVCRT___winitenv= msvcrt_SnapshotOfEnvironmentW(NULL);
|
|
||||||
|
|
||||||
MSVCRT__pgmptr = HeapAlloc(GetProcessHeap(), 0, MAX_PATH);
|
|
||||||
if (MSVCRT__pgmptr)
|
|
||||||
GetModuleFileNameA(0, MSVCRT__pgmptr, MAX_PATH);
|
|
||||||
|
|
||||||
MSVCRT__wpgmptr = HeapAlloc(GetProcessHeap(), 0, MAX_PATH * sizeof(WCHAR));
|
|
||||||
if (MSVCRT__wpgmptr)
|
|
||||||
GetModuleFileNameW(0, MSVCRT__wpgmptr, MAX_PATH);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* INTERNAL: free memory used by args */
|
|
||||||
void msvcrt_free_args(void)
|
|
||||||
{
|
|
||||||
/* FIXME: more things to free */
|
|
||||||
if (MSVCRT___initenv) HeapFree(GetProcessHeap(), 0, MSVCRT___initenv);
|
|
||||||
if (MSVCRT___winitenv) HeapFree(GetProcessHeap(), 0, MSVCRT___winitenv);
|
|
||||||
if (MSVCRT__environ) HeapFree(GetProcessHeap(), 0, MSVCRT__environ);
|
|
||||||
if (MSVCRT__wenviron) HeapFree(GetProcessHeap(), 0, MSVCRT__wenviron);
|
|
||||||
if (MSVCRT__pgmptr) HeapFree(GetProcessHeap(), 0, MSVCRT__pgmptr);
|
|
||||||
if (MSVCRT__wpgmptr) HeapFree(GetProcessHeap(), 0, MSVCRT__wpgmptr);
|
|
||||||
}
|
|
||||||
#ifndef __REACTOS__
|
|
||||||
/*********************************************************************
|
|
||||||
* __getmainargs (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
void __getmainargs(int *argc, char** *argv, char** *envp,
|
|
||||||
int expand_wildcards, int *new_mode)
|
|
||||||
{
|
|
||||||
TRACE("(%p,%p,%p,%d,%p).\n", argc, argv, envp, expand_wildcards, new_mode);
|
|
||||||
*argc = MSVCRT___argc;
|
|
||||||
*argv = MSVCRT___argv;
|
|
||||||
*envp = MSVCRT___initenv;
|
|
||||||
if (new_mode)
|
|
||||||
MSVCRT__set_new_mode( *new_mode );
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
/*********************************************************************
|
|
||||||
* __wgetmainargs (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
void __wgetmainargs(int *argc, MSVCRT_wchar_t** *wargv, MSVCRT_wchar_t** *wenvp,
|
|
||||||
int expand_wildcards, int *new_mode)
|
|
||||||
{
|
|
||||||
TRACE("(%p,%p,%p,%d,%p).\n", argc, wargv, wenvp, expand_wildcards, new_mode);
|
|
||||||
*argc = MSVCRT___argc;
|
|
||||||
*wargv = MSVCRT___wargv;
|
|
||||||
*wenvp = MSVCRT___winitenv;
|
|
||||||
if (new_mode)
|
|
||||||
MSVCRT__set_new_mode( *new_mode );
|
|
||||||
}
|
|
||||||
#ifndef __REACTOS__
|
|
||||||
/*********************************************************************
|
|
||||||
* _initterm (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
unsigned int _initterm(_INITTERMFUN *start,_INITTERMFUN *end)
|
|
||||||
{
|
|
||||||
_INITTERMFUN* current = start;
|
|
||||||
|
|
||||||
TRACE("(%p,%p)\n",start,end);
|
|
||||||
while (current<end)
|
|
||||||
{
|
|
||||||
if (*current)
|
|
||||||
{
|
|
||||||
TRACE("Call init function %p\n",*current);
|
|
||||||
(**current)();
|
|
||||||
TRACE("returned\n");
|
|
||||||
}
|
|
||||||
current++;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
/*********************************************************************
|
|
||||||
* __set_app_type (MSVCRT.@)
|
|
||||||
*/
|
|
||||||
void MSVCRT___set_app_type(int app_type)
|
|
||||||
{
|
|
||||||
TRACE("(%d) %s application\n", app_type, app_type == 2 ? "Gui" : "Console");
|
|
||||||
MSVCRT_app_type = app_type;
|
|
||||||
}
|
|
Loading…
Reference in a new issue