| /* |
| * Copyright (C) 2007 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "resource.h" |
| |
| #include <shlwapi.h> |
| #include <stdio.h> |
| #include <tchar.h> |
| #include <windows.h> |
| |
| #define LOG(header, ...) \ |
| do { \ |
| _ftprintf(stderr, header); \ |
| _ftprintf(stderr, __VA_ARGS__); \ |
| } while (0) |
| #define LOG_WARNING(...) LOG(TEXT("WARNING: "), __VA_ARGS__) |
| #define LOG_ERROR(...) LOG(TEXT("ERROR: "), __VA_ARGS__) |
| |
| static TCHAR* getStringValue(HKEY key, LPCTSTR valueName) |
| { |
| DWORD type = 0; |
| DWORD bufferSize = 0; |
| if (RegQueryValueEx(key, valueName, 0, &type, 0, &bufferSize) != ERROR_SUCCESS || type != REG_SZ) |
| return 0; |
| |
| TCHAR* buffer = (TCHAR*)malloc(bufferSize); |
| if (RegQueryValueEx(key, 0, 0, &type, reinterpret_cast<LPBYTE>(buffer), &bufferSize) != ERROR_SUCCESS) { |
| free(buffer); |
| return 0; |
| } |
| |
| return buffer; |
| } |
| |
| static LPOLESTR getWebViewCLSID() |
| { |
| LPCTSTR webViewProgID = TEXT("WebKit.WebView"); |
| |
| CLSID clsid = CLSID_NULL; |
| HRESULT hr = CLSIDFromProgID(webViewProgID, &clsid); |
| if (FAILED(hr)) { |
| LOG_WARNING(TEXT("Failed to get CLSID for %s\n"), webViewProgID); |
| return 0; |
| } |
| |
| LPOLESTR clsidString = 0; |
| if (FAILED(StringFromCLSID(clsid, &clsidString))) { |
| LOG_WARNING(TEXT("Failed to get string representation of CLSID for WebView\n")); |
| return 0; |
| } |
| |
| return clsidString; |
| } |
| |
| static TCHAR* getInstalledWebKitDirectory() |
| { |
| LPCTSTR keyPrefix = TEXT("SOFTWARE\\Classes\\CLSID\\"); |
| LPCTSTR keySuffix = TEXT("\\InprocServer32"); |
| |
| LPOLESTR clsid = getWebViewCLSID(); |
| if (!clsid) |
| return 0; |
| |
| size_t keyBufferLength = _tcslen(keyPrefix) + _tcslen(clsid) + _tcslen(keySuffix) + 1; |
| TCHAR* keyString = (TCHAR*)malloc(keyBufferLength * sizeof(TCHAR)); |
| |
| int ret = _sntprintf_s(keyString, keyBufferLength, keyBufferLength - 1, TEXT("%s%s%s"), keyPrefix, clsid, keySuffix); |
| CoTaskMemFree(clsid); |
| if (ret == -1) { |
| LOG_WARNING(TEXT("Failed to construct InprocServer32 key\n")); |
| return 0; |
| } |
| |
| HKEY serverKey = 0; |
| LONG error = RegOpenKeyEx(HKEY_LOCAL_MACHINE, keyString, 0, KEY_READ, &serverKey); |
| free(keyString); |
| if (error != ERROR_SUCCESS) { |
| LOG_WARNING(TEXT("Failed to open registry key %s\n"), keyString); |
| return 0; |
| } |
| |
| TCHAR* webKitPath = getStringValue(serverKey, 0); |
| RegCloseKey(serverKey); |
| if (!webKitPath) { |
| LOG_WARNING(TEXT("Couldn't retrieve value for registry key %s\n"), keyString); |
| return 0; |
| } |
| |
| TCHAR* startOfFileName = PathFindFileName(webKitPath); |
| if (startOfFileName == webKitPath) { |
| LOG_WARNING(TEXT("Couldn't find filename from path %s\n"), webKitPath); |
| free(webKitPath); |
| return 0; |
| } |
| |
| *startOfFileName = '\0'; |
| return webKitPath; |
| } |
| |
| static char* copyManifest(HMODULE module, LPCTSTR id) |
| { |
| HRSRC resHandle = FindResource(module, id, MAKEINTRESOURCE(RT_MANIFEST)); |
| if (!resHandle) |
| return 0; |
| DWORD manifestSize = SizeofResource(module, resHandle); |
| if (!manifestSize) |
| return 0; |
| HGLOBAL resData = LoadResource(module, resHandle); |
| if (!resData) |
| return 0; |
| void* data = LockResource(resData); |
| if (!data) |
| return 0; |
| |
| char* dataCopy = static_cast<char*>(malloc(manifestSize + 1)); |
| if (!dataCopy) |
| return 0; |
| memcpy(dataCopy, data, manifestSize); |
| dataCopy[manifestSize] = 0; |
| return dataCopy; |
| } |
| |
| static void replaceManifest() |
| { |
| TCHAR safariPath[MAX_PATH]; |
| ::ExpandEnvironmentStrings(TEXT("%TMP%\\WebKitNightly\\Safari.exe"), safariPath, ARRAYSIZE(safariPath)); |
| |
| // get the existing manifest out of Safari.exe |
| HMODULE safariModule = LoadLibraryEx(safariPath, 0, LOAD_LIBRARY_AS_DATAFILE); |
| if (!safariModule) |
| return; |
| char* safariManifest = copyManifest(safariModule, MAKEINTRESOURCE(1)); |
| FreeLibrary(safariModule); |
| if (!safariManifest) |
| return; |
| |
| // see if the existing Safari manifest contains registry free COM info |
| // (we only need to update if it is not registry free COM-aware) |
| bool needsUpdate = !strstr(safariManifest, "<comClass"); |
| free(safariManifest); |
| if (!needsUpdate) |
| return; |
| |
| // replace the manifest with the extra manifest stashed in FindSafar.exe (ID 100) |
| char* replacementManifest = copyManifest(0, MAKEINTRESOURCE(IDR_SAFARI_MANIFEST)); |
| if (!replacementManifest) |
| return; |
| if (HANDLE h = BeginUpdateResource(safariPath, FALSE)) { |
| UpdateResource(h, MAKEINTRESOURCE(RT_MANIFEST), MAKEINTRESOURCE(1), MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), replacementManifest, strlen(replacementManifest)); |
| EndUpdateResource(h, FALSE); |
| } |
| |
| free(replacementManifest); |
| } |
| |
| int _tmain(int argc, TCHAR* argv[]) |
| { |
| TCHAR* path = getInstalledWebKitDirectory(); |
| if (!path) { |
| LOG_ERROR(TEXT("Couldn't determine installed Safari path\n")); |
| return 1; |
| } |
| |
| bool printLauncher = false; |
| bool printEnvironment = false; |
| bool debugger = false; |
| bool updateManifest = false; |
| |
| for (int i = 1; i < argc; ++i) { |
| if (!_tcscmp(argv[i], TEXT("/printSafariLauncher"))) { |
| printLauncher = true; |
| continue; |
| } |
| if (!_tcscmp(argv[i], TEXT("/printSafariEnvironment"))) { |
| printEnvironment = true; |
| continue; |
| } |
| if (!_tcscmp(argv[i], TEXT("/debugger"))) { |
| debugger = true; |
| continue; |
| } |
| |
| if (!_tcscmp(argv[i], TEXT("/updateManifest"))) { |
| updateManifest = true; |
| continue; |
| } |
| } |
| |
| if (updateManifest) { |
| replaceManifest(); |
| return 0; |
| } |
| |
| // printLauncher is inclusive of printEnvironment, so do not |
| // leave both enabled: |
| if (printLauncher && printEnvironment) |
| printEnvironment = false; |
| |
| if (!printLauncher && !printEnvironment) { |
| _tprintf(TEXT("%s\n"), path); |
| free(path); |
| return 0; |
| } |
| |
| LPCTSTR lines[] = { |
| TEXT("@echo off"), |
| TEXT("del /s /q \"%%TMP%%\\WebKitNightly\""), |
| TEXT("mkdir 2>NUL \"%%TMP%%\\WebKitNightly\\Safari.resources\""), |
| TEXT("mkdir 2>NUL \"%%TMP%%\\WebKitNightly\\WebKit.resources\""), |
| TEXT("xcopy /y /i /d \"%sSafari.exe\" \"%%TMP%%\\WebKitNightly\""), |
| TEXT("xcopy /y /i /d /e \"%sSafari.resources\" \"%%TMP%%\\WebKitNightly\\Safari.resources\""), |
| TEXT("xcopy /y /i /d /e \"%splugins\" \"%%TMP%%\\WebKitNightly\\plugins\""), |
| TEXT("xcopy /y /i /d WebKit.dll \"%%TMP%%\\WebKitNightly\""), |
| TEXT("xcopy /y /i /d WebKit.pdb \"%%TMP%%\\WebKitNightly\""), |
| TEXT("xcopy /y /i /d /e WebKit.resources \"%%TMP%%\\WebKitNightly\\WebKit.resources\""), |
| TEXT("FindSafari.exe /updateManifest"), |
| TEXT("set PATH=%%CD%%;%s;%%PATH%%"), |
| }; |
| |
| LPCTSTR command = TEXT("\"%TMP%\\WebKitNightly\\Safari.exe\""); |
| |
| LPCTSTR launchLines[] = { |
| TEXT("%s"), |
| }; |
| |
| LPCTSTR debuggerLines[] = { |
| TEXT("if exist \"%%DevEnvDir%%\\VCExpress.exe\" ("), |
| TEXT("\"%%DevEnvDir%%\\VCExpress.exe\" /debugExe %s"), |
| TEXT(") else ("), |
| TEXT("\"%%DevEnvDir%%\\devenv.exe\" /debugExe %s"), |
| TEXT(")"), |
| }; |
| |
| for (int i = 0; i < ARRAYSIZE(lines); ++i) { |
| _tprintf(lines[i], path); |
| _tprintf(TEXT("\n")); |
| } |
| |
| LPCTSTR* endLines = debugger ? debuggerLines : launchLines; |
| |
| // Don't print launch command if we just want the environment set up... |
| if (!printEnvironment) { |
| for (unsigned i = 0; i < (debugger ? ARRAYSIZE(debuggerLines) : ARRAYSIZE(launchLines)); ++i) { |
| _tprintf(endLines[i], command); |
| _tprintf(TEXT("\n")); |
| } |
| } |
| |
| free(path); |
| return 0; |
| } |