Harald Armin Massa wrote:
> Dave,
>
>> It's coming from direct dependencies on user32.dll (from which we use
>> wsprintf()) and shell32.dll (from which we use SHGetSpecialFolderPath())
>> and is allocated when ResumeThread() is called to kickstart the new
>> backend,
>
> why does every backend need its own heap for user32.dll or
> shell32.dll? Wasn't the point of shared dlls to be shared?
No idea, and I thought so.
It's quite easy to prove using the test program attached. Just monitor
the desktop heap with dheapmon (from Microsoft's website), and run the
program with a single command line argument to get it to spawn a 100
child processes. You can stop it loading various DLLs by commenting out
the dummy calls to functions in them and rebuilding.
Of course, none of this would be an issue if we made the backend
multithreaded. :-)
I'll get my coat...
/D
#include <stdio.h>
#include <Windows.h>
#include <winsock.h>
#define SECURITY_WIN32
#include <Security.h>
#include <shlobj.h>
int main(int argc, char *argv[])
{
// Dummy functions to force linking to specific libs
// user32.lib
IsCharAlpha('a');
// wsock32.lib
WSADATA wsaData;
WSAStartup(MAKEWORD(1, 1), &wsaData);
// secur32.lib
char un[30];
DWORD dwUNLen = 30;
GetUserNameExA(NameUserPrincipal, un, &dwUNLen);
// advapi32.dll
char un2[30];
DWORD dwUN2Len = 30;
GetUserNameA(un2, &dwUN2Len);
// shell32.dll
IsUserAnAdmin();
// Used by child processes
if (argc == 1)
{
while (1)
{
printf("Foo\n");
Sleep(2000);
}
}
else
{
for (int x=0; x<100; x++)
{
STARTUPINFOA si;
PROCESS_INFORMATION pi;
memset(&pi, 0, sizeof(pi));
memset(&si, 0, sizeof(si));
si.cb = sizeof(si);
/*
* Create the subprocess in a suspended state. This will be resumed later,
* once we have written out the parameter file.
*/
printf("Creating process %d...\n", x);
if (!CreateProcessA(NULL, argv[0], NULL, NULL, TRUE, CREATE_SUSPENDED,
NULL, NULL, &si, &pi))
{
printf("CreateProcess call failed: %m (error code %d)", (int) GetLastError());
return -1;
}
printf("Resuming thread %d...\n", x);
if (ResumeThread(pi.hThread) == -1)
{
if (!TerminateProcess(pi.hProcess, 255))
{
printf("could not terminate unstartable process: error code %d", (int) GetLastError());
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
return -1;
}
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
printf("could not resume thread of unstarted process: error code %d", (int) GetLastError());
return -1;
}
}
}
return 0;
}