mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Make pgwin32_putenv() follow DLL loading and unloading.
Until now, the first putenv() call of a given postgres.exe process would cache the set of loaded CRTs. If a CRT unloaded after that call, the next putenv() would crash. That risk was largely theoretical, because the first putenv() precedes all PostgreSQL-initiated module loading. However, this might explain bad interactions with antivirus and other software that injects threads asynchronously. If an additional CRT loaded after the first putenv(), pgwin32_putenv() would not discover it. That CRT would have all environment changes predating its load, but it would not receive later PostgreSQL-initiated changes. An additional CRT loading concurrently with the first putenv() might miss that change in addition to missing later changes. Fix all those problems. This removes the cache mechanism from pgwin32_putenv(); the cost, less than 100 μs per backend startup, is negligible. No resulting misbehavior was known to be user-visible given the core distribution alone, but one can readily construct an affected extension module. No back-patch given the lack of complaints and the potential for behavior changes in non-PostgreSQL code running in the backend. Christian Ullrich, reviewed by Michael Paquier.
This commit is contained in:
@@ -21,126 +21,37 @@ pgwin32_putenv(const char *envval)
|
|||||||
{
|
{
|
||||||
char *envcpy;
|
char *envcpy;
|
||||||
char *cp;
|
char *cp;
|
||||||
|
|
||||||
/*
|
|
||||||
* Each CRT has its own _putenv() symbol and copy of the environment.
|
|
||||||
* Update the environment in each CRT module currently loaded, so every
|
|
||||||
* third-party library sees this change regardless of the CRT it links
|
|
||||||
* against.
|
|
||||||
*/
|
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
typedef int (_cdecl * PUTENVPROC) (const char *);
|
typedef int (_cdecl * PUTENVPROC) (const char *);
|
||||||
static struct
|
static const char *const modulenames[] = {
|
||||||
{
|
"msvcrt", /* Visual Studio 6.0 / MinGW */
|
||||||
char *modulename;
|
"msvcrtd",
|
||||||
HMODULE hmodule;
|
"msvcr70", /* Visual Studio 2002 */
|
||||||
PUTENVPROC putenvFunc;
|
"msvcr70d",
|
||||||
} rtmodules[] =
|
"msvcr71", /* Visual Studio 2003 */
|
||||||
{
|
"msvcr71d",
|
||||||
{
|
"msvcr80", /* Visual Studio 2005 */
|
||||||
"msvcrt", NULL, NULL
|
"msvcr80d",
|
||||||
}, /* Visual Studio 6.0 / MinGW */
|
"msvcr90", /* Visual Studio 2008 */
|
||||||
{
|
"msvcr90d",
|
||||||
"msvcrtd", NULL, NULL
|
"msvcr100", /* Visual Studio 2010 */
|
||||||
},
|
"msvcr100d",
|
||||||
{
|
"msvcr110", /* Visual Studio 2012 */
|
||||||
"msvcr70", NULL, NULL
|
"msvcr110d",
|
||||||
}, /* Visual Studio 2002 */
|
"msvcr120", /* Visual Studio 2013 */
|
||||||
{
|
"msvcr120d",
|
||||||
"msvcr70d", NULL, NULL
|
"ucrtbase", /* Visual Studio 2015 and later */
|
||||||
},
|
"ucrtbased",
|
||||||
{
|
NULL
|
||||||
"msvcr71", NULL, NULL
|
|
||||||
}, /* Visual Studio 2003 */
|
|
||||||
{
|
|
||||||
"msvcr71d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"msvcr80", NULL, NULL
|
|
||||||
}, /* Visual Studio 2005 */
|
|
||||||
{
|
|
||||||
"msvcr80d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"msvcr90", NULL, NULL
|
|
||||||
}, /* Visual Studio 2008 */
|
|
||||||
{
|
|
||||||
"msvcr90d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"msvcr100", NULL, NULL
|
|
||||||
}, /* Visual Studio 2010 */
|
|
||||||
{
|
|
||||||
"msvcr100d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"msvcr110", NULL, NULL
|
|
||||||
}, /* Visual Studio 2012 */
|
|
||||||
{
|
|
||||||
"msvcr110d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"msvcr120", NULL, NULL
|
|
||||||
}, /* Visual Studio 2013 */
|
|
||||||
{
|
|
||||||
"msvcr120d", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"ucrtbase", NULL, NULL
|
|
||||||
}, /* Visual Studio 2015 and later */
|
|
||||||
{
|
|
||||||
"ucrtbased", NULL, NULL
|
|
||||||
},
|
|
||||||
{
|
|
||||||
NULL, NULL, NULL
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; rtmodules[i].modulename; i++)
|
|
||||||
{
|
|
||||||
if (rtmodules[i].putenvFunc == NULL)
|
|
||||||
{
|
|
||||||
if (rtmodules[i].hmodule == NULL)
|
|
||||||
{
|
|
||||||
/* Not attempted before, so try to find this DLL */
|
|
||||||
rtmodules[i].hmodule = GetModuleHandle(rtmodules[i].modulename);
|
|
||||||
if (rtmodules[i].hmodule == NULL)
|
|
||||||
{
|
|
||||||
/*
|
|
||||||
* Set to INVALID_HANDLE_VALUE so we know we have tried
|
|
||||||
* this one before, and won't try again.
|
|
||||||
*/
|
|
||||||
rtmodules[i].hmodule = INVALID_HANDLE_VALUE;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rtmodules[i].putenvFunc = (PUTENVPROC) GetProcAddress(rtmodules[i].hmodule, "_putenv");
|
|
||||||
if (rtmodules[i].putenvFunc == NULL)
|
|
||||||
{
|
|
||||||
rtmodules[i].hmodule = INVALID_HANDLE_VALUE;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/*
|
|
||||||
* Module loaded, but we did not find the function last time.
|
|
||||||
* We're not going to find it this time either...
|
|
||||||
*/
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/* At this point, putenvFunc is set or we have exited the loop */
|
|
||||||
rtmodules[i].putenvFunc(envval);
|
|
||||||
}
|
|
||||||
#endif /* _MSC_VER */
|
#endif /* _MSC_VER */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Update process environment, making this change visible to child
|
* Update process environment, making this change visible to child
|
||||||
* processes and to CRTs initializing in the future.
|
* processes and to CRTs initializing in the future. Do this before the
|
||||||
|
* _putenv() loop, for the benefit of any CRT that initializes during this
|
||||||
|
* pgwin32_putenv() execution, after the loop checks that CRT.
|
||||||
*
|
*
|
||||||
* Need a copy of the string so we can modify it.
|
* Need a copy of the string so we can modify it.
|
||||||
*/
|
*/
|
||||||
@@ -170,6 +81,31 @@ pgwin32_putenv(const char *envval)
|
|||||||
}
|
}
|
||||||
free(envcpy);
|
free(envcpy);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Each CRT has its own _putenv() symbol and copy of the environment.
|
||||||
|
* Update the environment in each CRT module currently loaded, so every
|
||||||
|
* third-party library sees this change regardless of the CRT it links
|
||||||
|
* against. Addresses within these modules may become invalid the moment
|
||||||
|
* we call FreeLibrary(), so don't cache them.
|
||||||
|
*/
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
for (i = 0; modulenames[i]; i++)
|
||||||
|
{
|
||||||
|
HMODULE hmodule = NULL;
|
||||||
|
BOOL res = GetModuleHandleEx(0, modulenames[i], &hmodule);
|
||||||
|
|
||||||
|
if (res != 0 && hmodule != NULL)
|
||||||
|
{
|
||||||
|
PUTENVPROC putenvFunc;
|
||||||
|
|
||||||
|
putenvFunc = (PUTENVPROC) GetProcAddress(hmodule, "_putenv");
|
||||||
|
if (putenvFunc)
|
||||||
|
putenvFunc(envval);
|
||||||
|
FreeLibrary(hmodule);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* _MSC_VER */
|
||||||
|
|
||||||
/* Finally, update our "own" cache */
|
/* Finally, update our "own" cache */
|
||||||
return _putenv(envval);
|
return _putenv(envval);
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user