1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifdef _WIN32 18 19 #include "utils.h" 20 21 #define _CRT_SECURE_NO_WARNINGS 1 22 23 // Set to true to get some extra debug information 24 bool gIsDebug = false; 25 // Set to true to output errors to stderr (for a Console app) 26 // or to false to output using msg box (for a Windows UI app) 27 bool gIsConsole = false; 28 29 // Displays a message in an ok+info dialog box. msgBox(const char * text,...)30 void msgBox(const char* text, ...) { 31 CString formatted; 32 va_list ap; 33 va_start(ap, text); 34 formatted.setv(text, ap); 35 va_end(ap); 36 37 MessageBoxA(NULL, formatted.cstr(), "Android SDK Manager", MB_OK | MB_ICONINFORMATION); 38 } 39 40 // Displays GetLastError prefixed with a description in an error dialog box displayLastError(const char * description,...)41 void displayLastError(const char *description, ...) { 42 CString formatted; 43 va_list ap; 44 va_start(ap, description); 45 formatted.setv(description, ap); 46 va_end(ap); 47 48 CString error; 49 error.setLastWin32Error(NULL); 50 formatted.add("\r\n"); 51 formatted.add(error.cstr()); 52 53 if (gIsConsole) { 54 fprintf(stderr, "%s\n", formatted.cstr()); 55 } else { 56 MessageBox(NULL, formatted.cstr(), "Android SDK Manager - Error", MB_OK | MB_ICONERROR); 57 } 58 } 59 60 // Executes the command line. Does not wait for the program to finish. 61 // The return code is from CreateProcess (0 means failure), not the running app. execNoWait(const char * app,const char * params,const char * workDir)62 int execNoWait(const char *app, const char *params, const char *workDir) { 63 STARTUPINFO startup; 64 PROCESS_INFORMATION pinfo; 65 66 ZeroMemory(&pinfo, sizeof(pinfo)); 67 68 ZeroMemory(&startup, sizeof(startup)); 69 startup.cb = sizeof(startup); 70 startup.dwFlags = STARTF_USESHOWWINDOW; 71 startup.wShowWindow = SW_SHOWDEFAULT; 72 73 int ret = CreateProcessA( 74 (LPSTR) app, /* program path */ 75 (LPSTR) params, /* command-line */ 76 NULL, /* process handle is not inheritable */ 77 NULL, /* thread handle is not inheritable */ 78 TRUE, /* yes, inherit some handles */ 79 0, /* create flags */ 80 NULL, /* use parent's environment block */ 81 workDir, /* use parent's starting directory */ 82 &startup, /* startup info, i.e. std handles */ 83 &pinfo); 84 85 if (ret) { 86 CloseHandle(pinfo.hProcess); 87 CloseHandle(pinfo.hThread); 88 } 89 90 return ret; 91 } 92 93 // Executes command, waits for completion and returns exit code. 94 // As indicated in MSDN for CreateProcess, callers should double-quote the program name 95 // e.g. cmd="\"c:\program files\myapp.exe\" arg1 arg2"; execWait(const char * cmd)96 int execWait(const char *cmd) { 97 STARTUPINFO startup; 98 PROCESS_INFORMATION pinfo; 99 100 ZeroMemory(&pinfo, sizeof(pinfo)); 101 102 ZeroMemory(&startup, sizeof(startup)); 103 startup.cb = sizeof(startup); 104 startup.dwFlags = STARTF_USESHOWWINDOW; 105 startup.wShowWindow = SW_HIDE|SW_MINIMIZE; 106 107 int ret = CreateProcessA( 108 NULL, /* program path */ 109 (LPSTR) cmd, /* command-line */ 110 NULL, /* process handle is not inheritable */ 111 NULL, /* thread handle is not inheritable */ 112 TRUE, /* yes, inherit some handles */ 113 CREATE_NO_WINDOW, /* we don't want a console */ 114 NULL, /* use parent's environment block */ 115 NULL, /* use parent's starting directory */ 116 &startup, /* startup info, i.e. std handles */ 117 &pinfo); 118 119 int result = -1; 120 if (ret) { 121 WaitForSingleObject(pinfo.hProcess, INFINITE); 122 123 DWORD exitCode; 124 if (GetExitCodeProcess(pinfo.hProcess, &exitCode)) { 125 // this should not return STILL_ACTIVE (259) 126 result = exitCode; 127 } 128 CloseHandle(pinfo.hProcess); 129 CloseHandle(pinfo.hThread); 130 } 131 132 return result; 133 } 134 getModuleDir(CPath * outDir)135 bool getModuleDir(CPath *outDir) { 136 CHAR programDir[MAX_PATH]; 137 int ret = GetModuleFileName(NULL, programDir, sizeof(programDir)); 138 if (ret != 0) { 139 // Remove the last segment to keep only the directory. 140 int pos = ret - 1; 141 while (pos > 0 && programDir[pos] != '\\') { 142 --pos; 143 } 144 outDir->set(programDir, pos); 145 return true; 146 } 147 return false; 148 } 149 150 // Disables the FS redirection done by WOW64. 151 // Because this runs as a 32-bit app, Windows automagically remaps some 152 // folder under the hood (e.g. "Programs Files(x86)" is mapped as "Program Files"). 153 // This prevents the app from correctly searching for java.exe in these folders. 154 // The registry is also remapped. This method disables this redirection. 155 // Caller should restore the redirection later by using revertWow64FsRedirection(). disableWow64FsRedirection()156 PVOID disableWow64FsRedirection() { 157 158 // The call we want to make is the following: 159 // PVOID oldWow64Value; 160 // Wow64DisableWow64FsRedirection(&oldWow64Value); 161 // However that method may not exist (e.g. on XP non-64 systems) so 162 // we must not call it directly. 163 164 PVOID oldWow64Value = 0; 165 166 HMODULE hmod = LoadLibrary("kernel32.dll"); 167 if (hmod != NULL) { 168 FARPROC proc = GetProcAddress(hmod, "Wow64DisableWow64FsRedirection"); 169 if (proc != NULL) { 170 typedef BOOL (WINAPI *disableWow64FuncType)(PVOID *); 171 disableWow64FuncType funcPtr = (disableWow64FuncType)proc; 172 funcPtr(&oldWow64Value); 173 } 174 175 FreeLibrary(hmod); 176 } 177 178 return oldWow64Value; 179 } 180 181 // Reverts the redirection disabled in disableWow64FsRedirection. revertWow64FsRedirection(PVOID oldWow64Value)182 void revertWow64FsRedirection(PVOID oldWow64Value) { 183 184 // The call we want to make is the following: 185 // Wow64RevertWow64FsRedirection(oldWow64Value); 186 // However that method may not exist (e.g. on XP non-64 systems) so 187 // we must not call it directly. 188 189 HMODULE hmod = LoadLibrary("kernel32.dll"); 190 if (hmod != NULL) { 191 FARPROC proc = GetProcAddress(hmod, "Wow64RevertWow64FsRedirection"); 192 if (proc != NULL) { 193 typedef BOOL (WINAPI *revertWow64FuncType)(PVOID); 194 revertWow64FuncType funcPtr = (revertWow64FuncType)proc; 195 funcPtr(oldWow64Value); 196 } 197 198 FreeLibrary(hmod); 199 } 200 } 201 202 #endif /* _WIN32 */ 203