/*
|
* Copyright (C) 2014 The Android Open Source Project
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
#include "stdafx.h"
|
#include "utils.h"
|
|
// Set to true to get some extra debug information
|
bool gIsDebug = false;
|
// Set to true to output errors to stderr (for a Console app)
|
// or to false to output using msg box (for a Windows UI app)
|
bool gIsConsole = false;
|
|
// Application name used in error dialog. Defined using initUtils()
|
static CString gAppName("Find Java 2");
|
|
// Called by the application to initialize the app name used in error dialog boxes.
|
void initUtils(const TCHAR *appName) {
|
if (appName != NULL) {
|
gAppName = CString(appName);
|
return;
|
}
|
|
// Try to get the VERSIONINFO.FileDescription and use as app name
|
// Errors are ignored, in which case the default app name is used.
|
|
// First get the module (aka app instance) filename.
|
TCHAR moduleName[MAX_PATH + 1];
|
DWORD sz = ::GetModuleFileName(NULL /*AfxGetInstanceHandle()*/, moduleName, MAX_PATH);
|
if (sz == 0) {
|
// GetModuleFileName failed. Do nothing.
|
return;
|
}
|
moduleName[sz] = '\0'; // make sure string is properly terminated.
|
|
// Get the size of the FileVersionInfo buffer
|
DWORD obsoleteHandle; // see http://blogs.msdn.com/b/oldnewthing/archive/2007/07/31/4138786.aspx
|
DWORD fviSize = ::GetFileVersionInfoSize(moduleName, &obsoleteHandle);
|
if (fviSize == 0) {
|
return; // do nothing on error
|
}
|
|
char *fviBuffer = new char[fviSize];
|
if (::GetFileVersionInfo(moduleName, 0, fviSize, fviBuffer) != 0) {
|
VOID *vBuffer;
|
UINT vLen;
|
|
struct LANGUAGE_CODEPAGE {
|
WORD mLanguage;
|
WORD mCodePage;
|
} *lgcpBuffer;
|
|
UINT lgcpSize;
|
|
// Read the list of languages and code pages (c.f. MSDN for VerQueryValue)
|
if (::VerQueryValue(fviBuffer, _T("\\VarFileInfo\\Translation"), (LPVOID*)&lgcpBuffer, &lgcpSize) != 0 &&
|
lgcpSize >= sizeof(LANGUAGE_CODEPAGE)) {
|
// Use the first available language and code page
|
CString subBlock;
|
subBlock.Format(_T("\\StringFileInfo\\%04x%04x\\FileDescription"),
|
lgcpBuffer[0].mLanguage,
|
lgcpBuffer[0].mCodePage);
|
if (::VerQueryValue(fviBuffer, subBlock, &vBuffer, &vLen) != 0) {
|
gAppName.SetString((LPCTSTR)vBuffer, vLen);
|
}
|
}
|
}
|
delete[] fviBuffer;
|
}
|
|
CString getAppName() {
|
return gAppName;
|
}
|
|
|
// Displays a message in an ok+info dialog box.
|
void msgBox(const TCHAR* text, ...) {
|
CString formatted;
|
va_list ap;
|
va_start(ap, text);
|
formatted.FormatV(text, ap);
|
va_end(ap);
|
|
// TODO global CString to get app name
|
MessageBox(NULL, formatted, gAppName, MB_OK | MB_ICONINFORMATION);
|
}
|
|
// Sets the string to the message matching Win32 GetLastError.
|
// If message is non-null, it is prepended to the last error string.
|
CString getLastWin32Error(const TCHAR* message) {
|
DWORD err = GetLastError();
|
CString result;
|
LPTSTR errStr;
|
if (FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | /* dwFlags */
|
FORMAT_MESSAGE_FROM_SYSTEM,
|
NULL, /* lpSource */
|
err, /* dwMessageId */
|
0, /* dwLanguageId */
|
(LPTSTR) &errStr, /* out lpBuffer */
|
0, /* nSize */
|
NULL) != 0) { /* va_list args */
|
if (message == NULL) {
|
result.Format(_T("[%d] %s"), err, errStr);
|
} else {
|
result.Format(_T("%s[%d] %s"), message, err, errStr);
|
}
|
LocalFree(errStr);
|
}
|
return result;
|
}
|
|
// Displays GetLastError prefixed with a description in an error dialog box
|
void displayLastError(const TCHAR *description, ...) {
|
CString formatted;
|
va_list ap;
|
va_start(ap, description);
|
formatted.FormatV(description, ap);
|
va_end(ap);
|
|
CString error = getLastWin32Error(NULL);
|
formatted.Append(_T("\r\n"));
|
formatted.Append(error);
|
|
if (gIsConsole) {
|
_ftprintf(stderr, _T("%s\n"), (LPCTSTR) formatted);
|
} else {
|
CString name(gAppName);
|
name.Append(_T(" - Error"));
|
MessageBox(NULL, formatted, name, MB_OK | MB_ICONERROR);
|
}
|
}
|
|
// Executes the command line. Does not wait for the program to finish.
|
// The return code is from CreateProcess (0 means failure), not the running app.
|
int execNoWait(const TCHAR *app, const TCHAR *params, const TCHAR *workDir) {
|
STARTUPINFO startup;
|
PROCESS_INFORMATION pinfo;
|
|
ZeroMemory(&pinfo, sizeof(pinfo));
|
|
ZeroMemory(&startup, sizeof(startup));
|
startup.cb = sizeof(startup);
|
startup.dwFlags = STARTF_USESHOWWINDOW;
|
startup.wShowWindow = SW_SHOWDEFAULT;
|
|
int ret = CreateProcess(
|
app, /* program path */
|
(TCHAR *)params, /* command-line */
|
NULL, /* process handle is not inheritable */
|
NULL, /* thread handle is not inheritable */
|
TRUE, /* yes, inherit some handles */
|
0, /* create flags */
|
NULL, /* use parent's environment block */
|
workDir, /* use parent's starting directory */
|
&startup, /* startup info, i.e. std handles */
|
&pinfo);
|
|
if (ret) {
|
CloseHandle(pinfo.hProcess);
|
CloseHandle(pinfo.hThread);
|
}
|
|
return ret;
|
}
|
|
// Executes command, waits for completion and returns exit code.
|
// As indicated in MSDN for CreateProcess, callers should double-quote the program name
|
// e.g. cmd="\"c:\program files\myapp.exe\" arg1 arg2";
|
int execWait(const TCHAR *cmd) {
|
STARTUPINFO startup;
|
PROCESS_INFORMATION pinfo;
|
|
ZeroMemory(&pinfo, sizeof(pinfo));
|
|
ZeroMemory(&startup, sizeof(startup));
|
startup.cb = sizeof(startup);
|
startup.dwFlags = STARTF_USESHOWWINDOW;
|
startup.wShowWindow = SW_HIDE | SW_MINIMIZE;
|
|
int ret = CreateProcess(
|
NULL, /* program path */
|
(LPTSTR)cmd, /* command-line */
|
NULL, /* process handle is not inheritable */
|
NULL, /* thread handle is not inheritable */
|
TRUE, /* yes, inherit some handles */
|
CREATE_NO_WINDOW, /* we don't want a console */
|
NULL, /* use parent's environment block */
|
NULL, /* use parent's starting directory */
|
&startup, /* startup info, i.e. std handles */
|
&pinfo);
|
|
int result = -1;
|
if (ret) {
|
WaitForSingleObject(pinfo.hProcess, INFINITE);
|
|
DWORD exitCode;
|
if (GetExitCodeProcess(pinfo.hProcess, &exitCode)) {
|
// this should not return STILL_ACTIVE (259)
|
result = exitCode;
|
}
|
CloseHandle(pinfo.hProcess);
|
CloseHandle(pinfo.hThread);
|
}
|
|
return result;
|
}
|
|
bool getModuleDir(CPath *outDir) {
|
TCHAR programDir[MAX_PATH];
|
int ret = GetModuleFileName(NULL, programDir, sizeof(programDir) / sizeof(programDir[0]));
|
if (ret != 0) {
|
CPath dir(programDir);
|
dir.RemoveFileSpec();
|
*outDir = dir;
|
return true;
|
}
|
return false;
|
}
|
|
// Disables the FS redirection done by WOW64.
|
// Because this runs as a 32-bit app, Windows automagically remaps some
|
// folder under the hood (e.g. "Programs Files(x86)" is mapped as "Program Files").
|
// This prevents the app from correctly searching for java.exe in these folders.
|
// The registry is also remapped. This method disables this redirection.
|
// Caller should restore the redirection later by using revertWow64FsRedirection().
|
PVOID disableWow64FsRedirection() {
|
|
// The call we want to make is the following:
|
// PVOID oldWow64Value;
|
// Wow64DisableWow64FsRedirection(&oldWow64Value);
|
// However that method may not exist (e.g. on XP non-64 systems) so
|
// we must not call it directly.
|
|
PVOID oldWow64Value = 0;
|
|
HMODULE hmod = LoadLibrary(_T("kernel32.dll"));
|
if (hmod != NULL) {
|
FARPROC proc = GetProcAddress(hmod, "Wow64DisableWow64FsRedirection");
|
if (proc != NULL) {
|
typedef BOOL(WINAPI *disableWow64FuncType)(PVOID *);
|
disableWow64FuncType funcPtr = (disableWow64FuncType)proc;
|
funcPtr(&oldWow64Value);
|
}
|
|
FreeLibrary(hmod);
|
}
|
|
return oldWow64Value;
|
}
|
|
// Reverts the redirection disabled in disableWow64FsRedirection.
|
void revertWow64FsRedirection(PVOID oldWow64Value) {
|
|
// The call we want to make is the following:
|
// Wow64RevertWow64FsRedirection(oldWow64Value);
|
// However that method may not exist (e.g. on XP non-64 systems) so
|
// we must not call it directly.
|
|
HMODULE hmod = LoadLibrary(_T("kernel32.dll"));
|
if (hmod != NULL) {
|
FARPROC proc = GetProcAddress(hmod, "Wow64RevertWow64FsRedirection");
|
if (proc != NULL) {
|
typedef BOOL(WINAPI *revertWow64FuncType)(PVOID);
|
revertWow64FuncType funcPtr = (revertWow64FuncType)proc;
|
funcPtr(oldWow64Value);
|
}
|
|
FreeLibrary(hmod);
|
}
|
}
|