InjectedBundleWin.cpp   [plain text]


/*
 * 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