/* * Copyright (C) 2009 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. */ /* * The "SDK Manager" is for Windows only. * This simple .exe will sit at the root of the Windows SDK * and currently simply executes tools\android.bat. * Eventually it should simply replace the batch file. * * TODO: * - create temp dir, always copy *.jar there, exec android.jar * - get jars to copy from some file * - use a version number to copy jars only if needed (tools.revision?) */ #ifdef _WIN32 #include <stdio.h> #include <stdarg.h> #include <string.h> #include <windows.h> int _enable_dprintf = 0; void dprintf(char *msg, ...) { va_list ap; va_start(ap, msg); if (_enable_dprintf) { vfprintf(stderr, msg, ap); } va_end(ap); } void display_error(LPSTR description) { DWORD err = GetLastError(); LPSTR s, s2; fprintf(stderr, "%s, error %ld\n", description, err); if (FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | /* dwFlags */ FORMAT_MESSAGE_FROM_SYSTEM, NULL, /* lpSource */ err, /* dwMessageId */ 0, /* dwLanguageId */ (LPSTR)&s, /* lpBuffer */ 0, /* nSize */ NULL) != 0) { /* va_list args */ fprintf(stderr, "%s", s); s2 = (LPSTR) malloc(strlen(description) + strlen(s) + 5); sprintf(s2, "%s\r\n%s", description, s); MessageBox(NULL, s2, "Android SDK Manager - Error", MB_OK); free(s2); LocalFree(s); } } HANDLE create_temp_file(LPSTR temp_filename) { HANDLE file_handle = INVALID_HANDLE_VALUE; LPSTR temp_path = (LPSTR) malloc(MAX_PATH); /* Get the temp directory path using GetTempPath. GetTempFilename indicates that the temp path dir should not be larger than MAX_PATH-14. */ int ret = GetTempPath(MAX_PATH - 14, temp_path); if (ret > MAX_PATH || ret == 0) { display_error("GetTempPath failed"); free(temp_path); return INVALID_HANDLE_VALUE; } /* Now get a temp filename in the temp directory. */ if (!GetTempFileName(temp_path, "txt", 0, temp_filename)) { display_error("GetTempFileName failed"); } else { SECURITY_ATTRIBUTES sattr; ZeroMemory(&sattr, sizeof(sattr)); sattr.nLength = sizeof(SECURITY_ATTRIBUTES); sattr.bInheritHandle = TRUE; file_handle = CreateFile(temp_filename, // filename GENERIC_WRITE, // access: write FILE_SHARE_READ, // share mode: read OK &sattr, // security attributes CREATE_ALWAYS, // create even if exists FILE_ATTRIBUTE_NORMAL, // flags and attributes NULL); // template if (file_handle == INVALID_HANDLE_VALUE) { display_error("Create temp file failed"); } } free(temp_path); return file_handle; } void read_temp_file(LPSTR temp_filename) { HANDLE handle; handle = CreateFile(temp_filename, // filename GENERIC_READ, // access: read FILE_SHARE_READ, // share mode: read OK NULL, // security attributes OPEN_EXISTING, // only open existing file FILE_ATTRIBUTE_NORMAL, // flags and attributes NULL); // template if (handle == INVALID_HANDLE_VALUE) { display_error("Open temp file failed"); return; } /* Cap the size we're reading. 4K is good enough to display in a message box. */ DWORD size = 4096; LPSTR buffer = (LPSTR) malloc(size + 1); LPSTR p = buffer; DWORD num_left = size; DWORD num_read; do { if (!ReadFile(handle, p, num_left, &num_read, NULL)) { display_error("Read Output failed"); break; } num_left -= num_read; p += num_read; } while (num_read > 0); if (p != buffer) { *p = 0; /* Only output the buffer if it contains special keywords WARNING or ERROR. */ char* s1 = strstr(buffer, "WARNING"); char* s2 = strstr(buffer, "ERROR"); if (s2 != NULL && s2 < s1) { s1 = s2; } if (s1 != NULL) { /* We end the message at the first occurence of [INFO]. */ s2 = strstr(s1, "[INFO]"); if (s2 != NULL) { *s2 = 0; } MessageBox(NULL, s1, "Android SDK Manager - Output", MB_OK); } } free(buffer); if (!CloseHandle(handle)) { display_error("CloseHandle read temp file failed"); } } int sdk_launcher() { int result = 0; STARTUPINFO startup; PROCESS_INFORMATION pinfo; CHAR program_dir[MAX_PATH]; int ret, pos; CHAR temp_filename[MAX_PATH]; HANDLE temp_handle; ZeroMemory(&pinfo, sizeof(pinfo)); temp_handle = create_temp_file(temp_filename); if (temp_handle == INVALID_HANDLE_VALUE) { return 1; } ZeroMemory(&startup, sizeof(startup)); startup.cb = sizeof(startup); startup.dwFlags = STARTF_USESTDHANDLES; startup.hStdInput = GetStdHandle(STD_INPUT_HANDLE); startup.hStdOutput = temp_handle; startup.hStdError = temp_handle; /* get path of current program, to switch dirs here when executing the command. */ ret = GetModuleFileName(NULL, program_dir, sizeof(program_dir)); if (ret == 0) { display_error("Failed to get program's filename:"); result = 1; } else { /* Remove the last segment to keep only the directory. */ pos = ret - 1; while (pos > 0 && program_dir[pos] != '\\') { --pos; } program_dir[pos] = 0; } if (!result) { dprintf("Program dir: %s\n", program_dir); ret = CreateProcess( NULL, /* program path */ "tools\\android.bat update sdk", /* 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 */ program_dir, /* use parent's starting directory */ &startup, /* startup info, i.e. std handles */ &pinfo); dprintf("CreateProcess returned %d\n", ret); if (!ret) { display_error("Failed to execute tools\\android.bat:"); result = 1; } else { dprintf("Wait for process to finish.\n"); WaitForSingleObject(pinfo.hProcess, INFINITE); CloseHandle(pinfo.hProcess); CloseHandle(pinfo.hThread); } } dprintf("Cleanup.\n"); if (!CloseHandle(temp_handle)) { display_error("CloseHandle temp file failed"); } if (!result) { read_temp_file(temp_filename); } if (!DeleteFile(temp_filename)) { display_error("Delete temp file failed"); } return result; } int main(int argc, char **argv) { _enable_dprintf = argc > 1 && strcmp(argv[1], "-v") == 0; dprintf("Verbose debug mode.\n"); return sdk_launcher(); } #endif /* _WIN32 */