b192684c0f7a6f64da1364b06f7025629777ef71
[WebKit-https.git] / Source / WebCore / platform / graphics / mac / FontCustomPlatformData.cpp
1 /*
2  * Copyright (C) 2007, 2008, 2010 Apple Inc. All rights reserved.
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public License
15  * along with this library; see the file COPYING.LIB.  If not, write to
16  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  *
19  */
20
21 #include "config.h"
22 #include "FontCustomPlatformData.h"
23
24 #include "FontPlatformData.h"
25 #include "OpenTypeSanitizer.h"
26 #include "SharedBuffer.h"
27 #include "WOFFFileFormat.h"
28 #include <ApplicationServices/ApplicationServices.h>
29
30 #if USE(SKIA_ON_MAC_CHROMIUM)
31 #include "SkStream.h"
32 #include "SkTypeface.h"
33 #endif
34
35 namespace WebCore {
36
37 #if USE(SKIA_ON_MAC_CHROMIUM)
38 class RemoteFontStream : public SkStream {
39 public:
40     explicit RemoteFontStream(PassRefPtr<SharedBuffer> buffer)
41         : m_buffer(buffer)
42         , m_offset(0)
43     {
44     }
45
46     virtual ~RemoteFontStream()
47     {
48     }
49
50     virtual bool rewind()
51     {
52         m_offset = 0;
53         return true;
54     }
55
56     virtual size_t read(void* buffer, size_t size)
57     {
58         if (!buffer && !size) {
59             // This is request for the length of the stream.
60             return m_buffer->size();
61         }
62         // This is a request to read bytes or skip bytes (when buffer is 0).
63         if (!m_buffer->data() || !m_buffer->size())
64             return 0;
65         size_t left = m_buffer->size() - m_offset;
66         size_t bytesToConsume = std::min(left, size);
67         if (buffer)
68             std::memcpy(buffer, m_buffer->data() + m_offset, bytesToConsume);
69         m_offset += bytesToConsume;
70         return bytesToConsume;
71     }
72
73 private:
74     RefPtr<SharedBuffer> m_buffer;
75     size_t m_offset;
76 };
77 #endif
78
79 FontCustomPlatformData::~FontCustomPlatformData()
80 {
81 #if __MAC_OS_X_VERSION_MIN_REQUIRED == 1050
82     if (m_atsContainer)
83         ATSFontDeactivate(m_atsContainer, NULL, kATSOptionFlagsDefault);
84 #endif
85 #if USE(SKIA_ON_MAC_CHROMIUM)
86     SkSafeUnref(m_typeface);
87 #endif
88     CGFontRelease(m_cgFont);
89 }
90
91 FontPlatformData FontCustomPlatformData::fontPlatformData(int size, bool bold, bool italic, FontOrientation orientation, TextOrientation textOrientation, FontWidthVariant widthVariant, FontRenderingMode)
92 {
93     return FontPlatformData(m_cgFont, size, bold, italic, orientation, textOrientation, widthVariant);
94 }
95
96 FontCustomPlatformData* createFontCustomPlatformData(SharedBuffer* buffer)
97 {
98     ASSERT_ARG(buffer, buffer);
99
100 #if USE(OPENTYPE_SANITIZER)
101     OpenTypeSanitizer sanitizer(buffer);
102     RefPtr<SharedBuffer> transcodeBuffer = sanitizer.sanitize();
103     if (!transcodeBuffer)
104         return 0; // validation failed.
105     buffer = transcodeBuffer.get();
106 #else
107     RefPtr<SharedBuffer> sfntBuffer;
108     if (isWOFF(buffer)) {
109         Vector<char> sfnt;
110         if (!convertWOFFToSfnt(buffer, sfnt))
111             return 0;
112
113         sfntBuffer = SharedBuffer::adoptVector(sfnt);
114         buffer = sfntBuffer.get();
115     }
116 #endif
117
118     ATSFontContainerRef containerRef = 0;
119
120     RetainPtr<CGFontRef> cgFontRef;
121
122 #if __MAC_OS_X_VERSION_MIN_REQUIRED >= 1060
123     RetainPtr<CFDataRef> bufferData(AdoptCF, buffer->createCFData());
124     RetainPtr<CGDataProviderRef> dataProvider(AdoptCF, CGDataProviderCreateWithCFData(bufferData.get()));
125
126     cgFontRef.adoptCF(CGFontCreateWithDataProvider(dataProvider.get()));
127     if (!cgFontRef)
128         return 0;
129 #else
130     // Use ATS to activate the font.
131
132     // The value "3" means that the font is private and can't be seen by anyone else.
133     ATSFontActivateFromMemory((void*)buffer->data(), buffer->size(), 3, kATSFontFormatUnspecified, NULL, kATSOptionFlagsDefault, &containerRef);
134     if (!containerRef)
135         return 0;
136     ItemCount fontCount;
137     ATSFontFindFromContainer(containerRef, kATSOptionFlagsDefault, 0, NULL, &fontCount);
138     
139     // We just support the first font in the list.
140     if (fontCount == 0) {
141         ATSFontDeactivate(containerRef, NULL, kATSOptionFlagsDefault);
142         return 0;
143     }
144     
145     ATSFontRef fontRef = 0;
146     ATSFontFindFromContainer(containerRef, kATSOptionFlagsDefault, 1, &fontRef, NULL);
147     if (!fontRef) {
148         ATSFontDeactivate(containerRef, NULL, kATSOptionFlagsDefault);
149         return 0;
150     }
151     
152     cgFontRef.adoptCF(CGFontCreateWithPlatformFont(&fontRef));
153     // Workaround for <rdar://problem/5675504>.
154     if (cgFontRef && !CGFontGetNumberOfGlyphs(cgFontRef.get()))
155         cgFontRef = 0;
156     if (!cgFontRef) {
157         ATSFontDeactivate(containerRef, NULL, kATSOptionFlagsDefault);
158         return 0;
159     }
160 #endif // __MAC_OS_X_VERSION_MIN_REQUIRED >= 1060
161
162     FontCustomPlatformData* fontCustomPlatformData = new FontCustomPlatformData(containerRef, cgFontRef.leakRef());
163 #if USE(SKIA_ON_MAC_CHROMIUM)
164     RemoteFontStream* stream = new RemoteFontStream(buffer);
165     fontCustomPlatformData->m_typeface = SkTypeface::CreateFromStream(stream);
166     stream->unref();
167 #endif
168     return fontCustomPlatformData;
169 }
170
171 bool FontCustomPlatformData::supportsFormat(const String& format)
172 {
173     return equalIgnoringCase(format, "truetype") || equalIgnoringCase(format, "opentype") || equalIgnoringCase(format, "woff");
174 }
175
176 }