blob: 24db173bf1ee0a6ac5fef9f5bda5fa528fe257b8 [file] [log] [blame]
/*
* Copyright (C) 2007, 2008, 2009 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
*/
#include "config.h"
#include "FontCustomPlatformData.h"
#include "Base64.h"
#include "FontPlatformData.h"
#include "OpenTypeUtilities.h"
#include "SharedBuffer.h"
#include "SoftLinking.h"
#include <ApplicationServices/ApplicationServices.h>
#include <WebKitSystemInterface/WebKitSystemInterface.h>
#include <wtf/RetainPtr.h>
// From t2embapi.h, which is missing from the Microsoft Platform SDK.
typedef unsigned long(WINAPIV *READEMBEDPROC) (void*, void*, unsigned long);
struct TTLOADINFO;
#define TTLOAD_PRIVATE 0x00000001
#define LICENSE_PREVIEWPRINT 0x0004
#define E_NONE 0x0000L
namespace WebCore {
using namespace std;
SOFT_LINK_LIBRARY(T2embed);
SOFT_LINK(T2embed, TTLoadEmbeddedFont, LONG, __stdcall, (HANDLE* phFontReference, ULONG ulFlags, ULONG* pulPrivStatus, ULONG ulPrivs, ULONG* pulStatus, READEMBEDPROC lpfnReadFromStream, LPVOID lpvReadStream, LPWSTR szWinFamilyName, LPSTR szMacFamilyName, TTLOADINFO* pTTLoadInfo), (phFontReference, ulFlags,pulPrivStatus, ulPrivs, pulStatus, lpfnReadFromStream, lpvReadStream, szWinFamilyName, szMacFamilyName, pTTLoadInfo));
SOFT_LINK(T2embed, TTGetNewFontName, LONG, __stdcall, (HANDLE* phFontReference, LPWSTR szWinFamilyName, long cchMaxWinName, LPSTR szMacFamilyName, long cchMaxMacName), (phFontReference, szWinFamilyName, cchMaxWinName, szMacFamilyName, cchMaxMacName));
SOFT_LINK(T2embed, TTDeleteEmbeddedFont, LONG, __stdcall, (HANDLE hFontReference, ULONG ulFlags, ULONG* pulStatus), (hFontReference, ulFlags, pulStatus));
FontCustomPlatformData::~FontCustomPlatformData()
{
CGFontRelease(m_cgFont);
if (m_fontReference) {
if (m_name.isNull()) {
ASSERT(T2embedLibrary());
ULONG status;
TTDeleteEmbeddedFont(m_fontReference, 0, &status);
} else
RemoveFontMemResourceEx(m_fontReference);
}
}
FontPlatformData FontCustomPlatformData::fontPlatformData(int size, bool bold, bool italic, FontRenderingMode renderingMode)
{
ASSERT(m_cgFont);
ASSERT(m_fontReference);
ASSERT(T2embedLibrary());
LOGFONT& logFont = *static_cast<LOGFONT*>(malloc(sizeof(LOGFONT)));
if (m_name.isNull())
TTGetNewFontName(&m_fontReference, logFont.lfFaceName, LF_FACESIZE, 0, 0);
else
memcpy(logFont.lfFaceName, m_name.charactersWithNullTermination(), sizeof(logFont.lfFaceName[0]) * min(static_cast<size_t>(LF_FACESIZE), 1 + m_name.length()));
logFont.lfHeight = -size;
if (renderingMode == NormalRenderingMode)
logFont.lfHeight *= 32;
logFont.lfWidth = 0;
logFont.lfEscapement = 0;
logFont.lfOrientation = 0;
logFont.lfUnderline = false;
logFont.lfStrikeOut = false;
logFont.lfCharSet = DEFAULT_CHARSET;
logFont.lfOutPrecision = OUT_TT_ONLY_PRECIS;
logFont.lfQuality = CLEARTYPE_QUALITY;
logFont.lfPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE;
logFont.lfItalic = italic;
logFont.lfWeight = bold ? 700 : 400;
HFONT hfont = CreateFontIndirect(&logFont);
wkSetFontPlatformInfo(m_cgFont, &logFont, free);
return FontPlatformData(hfont, m_cgFont, size, bold, italic, renderingMode == AlternateRenderingMode);
}
const void* getData(void* info)
{
SharedBuffer* buffer = static_cast<SharedBuffer*>(info);
buffer->ref();
return (void*)buffer->data();
}
void releaseData(void* info, const void* data)
{
static_cast<SharedBuffer*>(info)->deref();
}
size_t getBytesWithOffset(void *info, void* buffer, size_t offset, size_t count)
{
SharedBuffer* sharedBuffer = static_cast<SharedBuffer*>(info);
size_t availBytes = count;
if (offset + count > sharedBuffer->size())
availBytes -= (offset + count) - sharedBuffer->size();
memcpy(buffer, sharedBuffer->data() + offset, availBytes);
return availBytes;
}
// Streams the concatenation of a header and font data.
class EOTStream {
public:
EOTStream(const EOTHeader& eotHeader, const SharedBuffer* fontData, size_t overlayDst, size_t overlaySrc, size_t overlayLength)
: m_eotHeader(eotHeader)
, m_fontData(fontData)
, m_overlayDst(overlayDst)
, m_overlaySrc(overlaySrc)
, m_overlayLength(overlayLength)
, m_offset(0)
, m_inHeader(true)
{
}
size_t read(void* buffer, size_t count);
private:
const EOTHeader& m_eotHeader;
const SharedBuffer* m_fontData;
size_t m_overlayDst;
size_t m_overlaySrc;
size_t m_overlayLength;
size_t m_offset;
bool m_inHeader;
};
size_t EOTStream::read(void* buffer, size_t count)
{
size_t bytesToRead = count;
if (m_inHeader) {
size_t bytesFromHeader = min(m_eotHeader.size() - m_offset, count);
memcpy(buffer, m_eotHeader.data() + m_offset, bytesFromHeader);
m_offset += bytesFromHeader;
bytesToRead -= bytesFromHeader;
if (m_offset == m_eotHeader.size()) {
m_inHeader = false;
m_offset = 0;
}
}
if (bytesToRead && !m_inHeader) {
size_t bytesFromData = min(m_fontData->size() - m_offset, bytesToRead);
memcpy(buffer, m_fontData->data() + m_offset, bytesFromData);
if (m_offset < m_overlayDst + m_overlayLength && m_offset + bytesFromData >= m_overlayDst) {
size_t dstOffset = max<int>(m_overlayDst - m_offset, 0);
size_t srcOffset = max<int>(0, m_offset - m_overlayDst);
size_t bytesToCopy = min(bytesFromData - dstOffset, m_overlayLength - srcOffset);
memcpy(reinterpret_cast<char*>(buffer) + dstOffset, m_fontData->data() + m_overlaySrc + srcOffset, bytesToCopy);
}
m_offset += bytesFromData;
bytesToRead -= bytesFromData;
}
return count - bytesToRead;
}
static unsigned long WINAPIV readEmbedProc(void* stream, void* buffer, unsigned long length)
{
return static_cast<EOTStream*>(stream)->read(buffer, length);
}
// Creates a unique and unpredictable font name, in order to avoid collisions and to
// not allow access from CSS.
static String createUniqueFontName()
{
Vector<char> fontUuid(sizeof(GUID));
CoCreateGuid(reinterpret_cast<GUID*>(fontUuid.data()));
Vector<char> fontNameVector;
base64Encode(fontUuid, fontNameVector);
ASSERT(fontNameVector.size() < LF_FACESIZE);
return String(fontNameVector.data(), fontNameVector.size());
}
FontCustomPlatformData* createFontCustomPlatformData(SharedBuffer* buffer)
{
ASSERT_ARG(buffer, buffer);
ASSERT(T2embedLibrary());
// Get CG to create the font.
CGDataProviderDirectAccessCallbacks callbacks = { &getData, &releaseData, &getBytesWithOffset, NULL };
RetainPtr<CGDataProviderRef> dataProvider(AdoptCF, CGDataProviderCreateDirectAccess(buffer, buffer->size(), &callbacks));
CGFontRef cgFont = CGFontCreateWithDataProvider(dataProvider.get());
if (!cgFont)
return 0;
// Introduce the font to GDI. AddFontMemResourceEx cannot be used, because it will pollute the process's
// font namespace (Windows has no API for creating an HFONT from data without exposing the font to the
// entire process first). TTLoadEmbeddedFont lets us override the font family name, so using a unique name
// we avoid namespace collisions.
String fontName = createUniqueFontName();
// TTLoadEmbeddedFont works only with Embedded OpenType (.eot) data, so we need to create an EOT header
// and prepend it to the font data.
EOTHeader eotHeader;
size_t overlayDst;
size_t overlaySrc;
size_t overlayLength;
if (!getEOTHeader(buffer, eotHeader, overlayDst, overlaySrc, overlayLength)) {
CGFontRelease(cgFont);
return 0;
}
HANDLE fontReference;
ULONG privStatus;
ULONG status;
EOTStream eotStream(eotHeader, buffer, overlayDst, overlaySrc, overlayLength);
LONG loadEmbeddedFontResult = TTLoadEmbeddedFont(&fontReference, TTLOAD_PRIVATE, &privStatus, LICENSE_PREVIEWPRINT, &status, readEmbedProc, &eotStream, const_cast<LPWSTR>(fontName.charactersWithNullTermination()), 0, 0);
if (loadEmbeddedFontResult == E_NONE)
fontName = String();
else {
fontReference = renameAndActivateFont(buffer, fontName);
if (!fontReference) {
CGFontRelease(cgFont);
return 0;
}
}
return new FontCustomPlatformData(cgFont, fontReference, fontName);
}
}