mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-01 06:35:42 +00:00
181 lines
4.3 KiB
C++
181 lines
4.3 KiB
C++
// This file is a .cpp file meant to be included in nsBrowserApp.cpp and other
|
|
// similar bootstrap code. It converts wide-character windows wmain into UTF-8
|
|
// narrow-character strings.
|
|
|
|
#ifndef XP_WIN
|
|
#error This file only makes sense on Windows.
|
|
#endif
|
|
|
|
#include "nsUTF8Utils.h"
|
|
|
|
#ifndef XRE_DONT_PROTECT_DLL_LOAD
|
|
#include "nsSetDllDirectory.h"
|
|
#endif
|
|
|
|
#if defined(_MSC_VER) && (defined(_M_IX86) || defined(_M_X64)) && defined(XRE_WANT_DLL_BLOCKLIST)
|
|
#include "nsWindowsDllBlocklist.cpp"
|
|
#else
|
|
#undef XRE_WANT_DLL_BLOCKLIST
|
|
#endif
|
|
|
|
|
|
#ifdef __MINGW32__
|
|
|
|
/* MingW currently does not implement a wide version of the
|
|
startup routines. Workaround is to implement something like
|
|
it ourselves. See bug 411826 */
|
|
|
|
#include <shellapi.h>
|
|
|
|
int wmain(int argc, WCHAR **argv);
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
LPWSTR commandLine = GetCommandLineW();
|
|
int argcw = 0;
|
|
LPWSTR *argvw = CommandLineToArgvW(commandLine, &argcw);
|
|
if (!argvw)
|
|
return 127;
|
|
|
|
int result = wmain(argcw, argvw);
|
|
LocalFree(argvw);
|
|
return result;
|
|
}
|
|
#endif /* __MINGW32__ */
|
|
|
|
#define main NS_internal_main
|
|
|
|
int main(int argc, char **argv);
|
|
|
|
static char*
|
|
AllocConvertUTF16toUTF8(const WCHAR *arg)
|
|
{
|
|
// be generous... UTF16 units can expand up to 3 UTF8 units
|
|
int len = wcslen(arg);
|
|
char *s = new char[len * 3 + 1];
|
|
if (!s)
|
|
return NULL;
|
|
|
|
ConvertUTF16toUTF8 convert(s);
|
|
convert.write(arg, len);
|
|
convert.write_terminator();
|
|
return s;
|
|
}
|
|
|
|
static void
|
|
FreeAllocStrings(int argc, char **argv)
|
|
{
|
|
while (argc) {
|
|
--argc;
|
|
delete [] argv[argc];
|
|
}
|
|
|
|
delete [] argv;
|
|
}
|
|
|
|
#ifdef XRE_PRELOAD_XUL
|
|
static void preload(LPCWSTR dll)
|
|
{
|
|
HANDLE fd = CreateFileW(dll, GENERIC_READ, FILE_SHARE_READ,
|
|
NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL);
|
|
char buf[64 * 1024];
|
|
|
|
if (fd == INVALID_HANDLE_VALUE)
|
|
return;
|
|
|
|
DWORD dwBytesRead;
|
|
// Do dummy reads to trigger kernel-side readhead via FILE_FLAG_SEQUENTIAL_SCAN.
|
|
// Abort when underfilling because during testing the buffers are read fully
|
|
// A buffer that's not keeping up would imply that readahead isn't working right
|
|
while (ReadFile(fd, buf, sizeof(buf), &dwBytesRead, NULL) && dwBytesRead == sizeof(buf))
|
|
/* Nothing */;
|
|
|
|
CloseHandle(fd);
|
|
}
|
|
|
|
/** Populate the windows page cache */
|
|
static void preload_libs(LPCWSTR appPath)
|
|
{
|
|
wchar_t buf[MAX_PATH];
|
|
size_t pathlen = wcslen(appPath);
|
|
|
|
// rewind until the last \
|
|
for(;pathlen && appPath[pathlen] != L'\\'; pathlen--);
|
|
|
|
// if there is a directory name present, keep the last \
|
|
if (pathlen)
|
|
pathlen++;
|
|
|
|
if (pathlen + 10 > MAX_PATH) {
|
|
return;
|
|
}
|
|
|
|
wcsncpy(buf, appPath, pathlen);
|
|
size_t remaining = sizeof(buf)/sizeof(buf[0]) - pathlen - 1;
|
|
// For now only preload only the most expensive libs.
|
|
// The rest cost a few magnitudes less.
|
|
LPCWSTR files[] = {L"xul.dll",
|
|
#ifndef MOZ_STATIC_JS
|
|
L"mozjs.dll",
|
|
#endif
|
|
0};
|
|
for(int i = 0; files[i]; i++) {
|
|
wcsncpy(buf + pathlen, files[i], remaining);
|
|
preload(buf);
|
|
}
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
int wmain(int argc, WCHAR **argv)
|
|
{
|
|
#ifdef XRE_PRELOAD_XUL
|
|
// GetProcessIoCounters().ReadOperationCount seems to have little to
|
|
// do with actual read operations. It reports 0 or 1 at this stage
|
|
// in the program. Luckily 1 coincides with when prefetch is
|
|
// enabled. If Windows prefetch didn't happen we can do our own
|
|
// faster dll preloading.
|
|
IO_COUNTERS ioCounters;
|
|
if (GetProcessIoCounters(GetCurrentProcess(), &ioCounters)
|
|
&& !ioCounters.ReadOperationCount)
|
|
{
|
|
preload_libs(argv[0]);
|
|
}
|
|
#endif
|
|
|
|
#ifndef XRE_DONT_PROTECT_DLL_LOAD
|
|
mozilla::NS_SetDllDirectory(L"");
|
|
#endif
|
|
|
|
#ifdef XRE_WANT_DLL_BLOCKLIST
|
|
SetupDllBlocklist();
|
|
#endif
|
|
|
|
char **argvConverted = new char*[argc + 1];
|
|
if (!argvConverted)
|
|
return 127;
|
|
|
|
for (int i = 0; i < argc; ++i) {
|
|
argvConverted[i] = AllocConvertUTF16toUTF8(argv[i]);
|
|
if (!argvConverted[i]) {
|
|
return 127;
|
|
}
|
|
}
|
|
argvConverted[argc] = NULL;
|
|
|
|
// need to save argvConverted copy for later deletion.
|
|
char **deleteUs = new char*[argc+1];
|
|
if (!deleteUs) {
|
|
FreeAllocStrings(argc, argvConverted);
|
|
return 127;
|
|
}
|
|
for (int i = 0; i < argc; i++)
|
|
deleteUs[i] = argvConverted[i];
|
|
int result = main(argc, argvConverted);
|
|
|
|
delete[] argvConverted;
|
|
FreeAllocStrings(argc, deleteUs);
|
|
|
|
return result;
|
|
}
|