/* * Copyright (C) 2010 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. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. 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 INC. 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 "config.h" #include "InjectedBundle.h" #include "WKBundleAPICast.h" #include "WKBundleInitialize.h" #include "WebCertificateInfo.h" #include <WebCore/ResourceHandle.h> #include <WebCore/SimpleFontData.h> #include <wtf/text/CString.h> #include <windows.h> #include <winbase.h> #include <shlobj.h> #include <shlwapi.h> #if USE(CFNETWORK) #include <WebCore/CertificateCFWin.h> #endif using namespace WebCore; namespace WebKit { // FIXME: This should try and use <WebCore/FileSystem.h>. static String pathGetFileName(const String& path) { return String(::PathFindFileName(String(path).charactersWithNullTermination())); } static String directoryName(const String& path) { String fileName = pathGetFileName(path); String dirName = String(path); dirName.truncate(dirName.length() - pathGetFileName(path).length()); return dirName; } bool InjectedBundle::load(APIObject* initializationUserData) { WCHAR currentPath[MAX_PATH]; if (!::GetCurrentDirectoryW(MAX_PATH, currentPath)) return false; String directorBundleResidesIn = directoryName(m_path); if (!::SetCurrentDirectoryW(directorBundleResidesIn.charactersWithNullTermination())) return false; m_platformBundle = ::LoadLibraryExW(m_path.charactersWithNullTermination(), 0, LOAD_WITH_ALTERED_SEARCH_PATH); if (!m_platformBundle) return false; // Reset the current directory. if (!::SetCurrentDirectoryW(currentPath)) { return false; } WKBundleInitializeFunctionPtr initializeFunction = reinterpret_cast<WKBundleInitializeFunctionPtr>(::GetProcAddress(m_platformBundle, "WKBundleInitialize")); if (!initializeFunction) return false; initializeFunction(toAPI(this), toAPI(initializationUserData)); return true; } void InjectedBundle::activateMacFontAscentHack() { SimpleFontData::setShouldApplyMacAscentHack(true); } void InjectedBundle::setHostAllowsAnyHTTPSCertificate(const String& host) { #if USE(CFNETWORK) ResourceHandle::setHostAllowsAnyHTTPSCertificate(host); #endif } void InjectedBundle::setClientCertificate(const String& host, const String& certificateSystemStoreName, const WebCertificateInfo* certificateInfo) { #if USE(CFNETWORK) ASSERT_ARG(certificateInfo, certificateInfo); if (!certificateInfo) return; const Vector<PCCERT_CONTEXT> certificateChain = certificateInfo->platformCertificateInfo().certificateChain(); ASSERT(certificateChain.size() == 1); if (certificateChain.size() != 1) return; ASSERT_ARG(certificateSystemStoreName, !certificateSystemStoreName.isEmpty()); if (certificateSystemStoreName.isEmpty()) return; // The PCCERT_CONTEXT in the WebCertificateInfo we created using the message from the UI process doesn't contain enough information // to actually use it in a request, we need to get the real certificate from the certificate store (which is typically the "MY" store). String mutableCertificateSystemStoreName = certificateSystemStoreName; HCERTSTORE certStore = ::CertOpenSystemStore(0, mutableCertificateSystemStoreName.charactersWithNullTermination()); if (!certStore) { LOG_ERROR("Could not open system certificate store %s", certificateSystemStoreName.ascii().data()); return; } PCCERT_CONTEXT realCert = ::CertFindCertificateInStore(certStore, X509_ASN_ENCODING | PKCS_7_ASN_ENCODING, 0, CERT_FIND_EXISTING, certificateChain.first(), 0); if (!realCert) { LOG_ERROR("Could not find certificate in system certificate store"); return; } ResourceHandle::setClientCertificate(host, WebCore::copyCertificateToData(realCert).get()); CertFreeCertificateContext(realCert); // We can't close certStore here, since the certificate is still in use. #endif } } // namespace WebKit