epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2008 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 8 | #include "SkFontConfigInterface.h" |
djsollen@google.com | 6930b57 | 2013-05-15 20:11:20 +0000 | [diff] [blame] | 9 | #include "SkFontConfigTypeface.h" |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 10 | #include "SkFontDescriptor.h" |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 11 | #include "SkFontHost.h" |
reed@google.com | 0da4861 | 2013-03-19 16:06:52 +0000 | [diff] [blame] | 12 | #include "SkFontHost_FreeType_common.h" |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 13 | #include "SkFontStream.h" |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 14 | #include "SkStream.h" |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 15 | #include "SkTypeface.h" |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 16 | #include "SkTypefaceCache.h" |
reed@google.com | b1c65b6 | 2013-02-26 15:50:51 +0000 | [diff] [blame] | 17 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 18 | SK_DECLARE_STATIC_MUTEX(gFontConfigInterfaceMutex); |
| 19 | static SkFontConfigInterface* gFontConfigInterface; |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 20 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 21 | SkFontConfigInterface* SkFontConfigInterface::RefGlobal() { |
| 22 | SkAutoMutexAcquire ac(gFontConfigInterfaceMutex); |
skia.committer@gmail.com | 5ca3bd0 | 2013-02-26 07:01:22 +0000 | [diff] [blame] | 23 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 24 | return SkSafeRef(gFontConfigInterface); |
| 25 | } |
bungeman@google.com | b13d63c | 2012-11-06 16:55:24 +0000 | [diff] [blame] | 26 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 27 | SkFontConfigInterface* SkFontConfigInterface::SetGlobal(SkFontConfigInterface* fc) { |
| 28 | SkAutoMutexAcquire ac(gFontConfigInterfaceMutex); |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 29 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 30 | SkRefCnt_SafeAssign(gFontConfigInterface, fc); |
| 31 | return fc; |
| 32 | } |
bungeman@google.com | b13d63c | 2012-11-06 16:55:24 +0000 | [diff] [blame] | 33 | |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 34 | /////////////////////////////////////////////////////////////////////////////// |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 35 | /////////////////////////////////////////////////////////////////////////////// |
| 36 | |
| 37 | // convenience function to create the direct interface if none is installed. |
| 38 | extern SkFontConfigInterface* SkCreateDirectFontConfigInterface(); |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 39 | |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 40 | static SkFontConfigInterface* RefFCI() { |
reed@google.com | b1c65b6 | 2013-02-26 15:50:51 +0000 | [diff] [blame] | 41 | for (;;) { |
| 42 | SkFontConfigInterface* fci = SkFontConfigInterface::RefGlobal(); |
| 43 | if (fci) { |
| 44 | return fci; |
| 45 | } |
reed@google.com | d66045e | 2013-03-04 19:07:02 +0000 | [diff] [blame] | 46 | fci = SkFontConfigInterface::GetSingletonDirectInterface(); |
reed@google.com | 750a24b | 2013-04-22 18:45:12 +0000 | [diff] [blame] | 47 | SkFontConfigInterface::SetGlobal(fci); |
reed@google.com | b1c65b6 | 2013-02-26 15:50:51 +0000 | [diff] [blame] | 48 | } |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 49 | } |
| 50 | |
reed@google.com | 381bb43 | 2013-05-13 19:43:59 +0000 | [diff] [blame] | 51 | // export this to SkFontMgr_fontconfig.cpp until this file just goes away. |
| 52 | SkFontConfigInterface* SkFontHost_fontconfig_ref_global(); |
| 53 | SkFontConfigInterface* SkFontHost_fontconfig_ref_global() { |
| 54 | return RefFCI(); |
| 55 | } |
| 56 | |
| 57 | /////////////////////////////////////////////////////////////////////////////// |
| 58 | |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 59 | struct FindRec { |
| 60 | FindRec(const char* name, SkTypeface::Style style) |
| 61 | : fFamilyName(name) // don't need to make a deep copy |
| 62 | , fStyle(style) {} |
| 63 | |
| 64 | const char* fFamilyName; |
| 65 | SkTypeface::Style fStyle; |
| 66 | }; |
| 67 | |
| 68 | static bool find_proc(SkTypeface* face, SkTypeface::Style style, void* ctx) { |
| 69 | FontConfigTypeface* fci = (FontConfigTypeface*)face; |
| 70 | const FindRec* rec = (const FindRec*)ctx; |
| 71 | |
| 72 | return rec->fStyle == style && fci->isFamilyName(rec->fFamilyName); |
| 73 | } |
| 74 | |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 75 | SkTypeface* SkFontHost::CreateTypeface(const SkTypeface* familyFace, |
| 76 | const char familyName[], |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 77 | SkTypeface::Style style) { |
| 78 | SkAutoTUnref<SkFontConfigInterface> fci(RefFCI()); |
| 79 | if (NULL == fci.get()) { |
| 80 | return NULL; |
| 81 | } |
| 82 | |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 83 | if (familyFace) { |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 84 | FontConfigTypeface* fct = (FontConfigTypeface*)familyFace; |
| 85 | familyName = fct->getFamilyName(); |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 86 | } |
| 87 | |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 88 | FindRec rec(familyName, style); |
| 89 | SkTypeface* face = SkTypefaceCache::FindByProcAndRef(find_proc, &rec); |
| 90 | if (face) { |
reed@google.com | 0b8a2fe | 2013-03-13 14:58:41 +0000 | [diff] [blame] | 91 | // SkDebugf("found cached face <%s> <%s> %p [%d]\n", familyName, ((FontConfigTypeface*)face)->getFamilyName(), face, face->getRefCnt()); |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 92 | return face; |
| 93 | } |
| 94 | |
| 95 | SkFontConfigInterface::FontIdentity indentity; |
| 96 | SkString outFamilyName; |
| 97 | SkTypeface::Style outStyle; |
| 98 | |
| 99 | if (!fci->matchFamilyName(familyName, style, |
| 100 | &indentity, &outFamilyName, &outStyle)) { |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 101 | return NULL; |
| 102 | } |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 103 | |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 104 | face = SkNEW_ARGS(FontConfigTypeface, (outStyle, indentity, outFamilyName)); |
| 105 | SkTypefaceCache::Add(face, style); |
reed@google.com | 0b8a2fe | 2013-03-13 14:58:41 +0000 | [diff] [blame] | 106 | // SkDebugf("add face <%s> <%s> %p [%d]\n", familyName, outFamilyName.c_str(), face, face->getRefCnt()); |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 107 | return face; |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 108 | } |
| 109 | |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 110 | SkTypeface* SkFontHost::CreateTypefaceFromStream(SkStream* stream) { |
| 111 | if (!stream) { |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 112 | return NULL; |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 113 | } |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 114 | const size_t length = stream->getLength(); |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 115 | if (!length) { |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 116 | return NULL; |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 117 | } |
| 118 | if (length >= 1024 * 1024 * 1024) { |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 119 | return NULL; // don't accept too large fonts (>= 1GB) for safety. |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 120 | } |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 121 | |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 122 | // TODO should the caller give us the style? |
| 123 | SkTypeface::Style style = SkTypeface::kNormal; |
| 124 | SkTypeface* face = SkNEW_ARGS(FontConfigTypeface, (style, stream)); |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 125 | return face; |
reed@android.com | ac98154 | 2009-07-31 16:17:01 +0000 | [diff] [blame] | 126 | } |
| 127 | |
reed@google.com | 80f5465 | 2013-02-25 22:19:20 +0000 | [diff] [blame] | 128 | SkTypeface* SkFontHost::CreateTypefaceFromFile(const char path[]) { |
mike@reedtribe.org | f381162 | 2013-03-19 02:18:33 +0000 | [diff] [blame] | 129 | SkAutoTUnref<SkStream> stream(SkStream::NewFromFile(path)); |
| 130 | return stream.get() ? CreateTypefaceFromStream(stream) : NULL; |
agl@chromium.org | df8ecfb | 2009-04-28 17:30:01 +0000 | [diff] [blame] | 131 | } |
| 132 | |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 133 | /////////////////////////////////////////////////////////////////////////////// |
| 134 | |
reed@google.com | 2cdc671 | 2013-03-21 18:22:00 +0000 | [diff] [blame] | 135 | SkStream* FontConfigTypeface::onOpenStream(int* ttcIndex) const { |
| 136 | SkStream* stream = this->getLocalStream(); |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 137 | if (stream) { |
reed@google.com | 5bfc839 | 2013-03-05 22:12:11 +0000 | [diff] [blame] | 138 | // should have been provided by CreateFromStream() |
| 139 | *ttcIndex = 0; |
bungeman@google.com | 6cab1a4 | 2013-05-29 13:43:31 +0000 | [diff] [blame] | 140 | |
| 141 | SkAutoTUnref<SkStream> dupStream(stream->duplicate()); |
| 142 | if (dupStream) { |
| 143 | return dupStream.detach(); |
| 144 | } |
| 145 | |
| 146 | // TODO: update interface use, remove the following code in this block. |
| 147 | size_t length = stream->getLength(); |
| 148 | |
| 149 | const void* memory = stream->getMemoryBase(); |
| 150 | if (NULL != memory) { |
| 151 | return new SkMemoryStream(memory, length, true); |
| 152 | } |
| 153 | |
| 154 | SkAutoTMalloc<uint8_t> allocMemory(length); |
| 155 | stream->rewind(); |
| 156 | if (length == stream->read(allocMemory.get(), length)) { |
bungeman@google.com | 88605bb | 2013-06-06 17:25:42 +0000 | [diff] [blame] | 157 | SkAutoTUnref<SkMemoryStream> copyStream(new SkMemoryStream()); |
| 158 | copyStream->setMemoryOwned(allocMemory.detach(), length); |
| 159 | return copyStream.detach(); |
bungeman@google.com | 6cab1a4 | 2013-05-29 13:43:31 +0000 | [diff] [blame] | 160 | } |
| 161 | |
| 162 | stream->rewind(); |
| 163 | stream->ref(); |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 164 | } else { |
| 165 | SkAutoTUnref<SkFontConfigInterface> fci(RefFCI()); |
| 166 | if (NULL == fci.get()) { |
| 167 | return NULL; |
| 168 | } |
reed@google.com | 2cdc671 | 2013-03-21 18:22:00 +0000 | [diff] [blame] | 169 | stream = fci->openStream(this->getIdentity()); |
| 170 | *ttcIndex = this->getIdentity().fTTCIndex; |
reed@google.com | f71a233 | 2013-02-27 19:06:30 +0000 | [diff] [blame] | 171 | } |
| 172 | return stream; |
reed@google.com | d71fe99 | 2013-02-25 20:38:07 +0000 | [diff] [blame] | 173 | } |
| 174 | |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 175 | int FontConfigTypeface::onGetTableTags(SkFontTableTag tags[]) const { |
reed@google.com | 5bfc839 | 2013-03-05 22:12:11 +0000 | [diff] [blame] | 176 | int ttcIndex; |
reed@google.com | 2cdc671 | 2013-03-21 18:22:00 +0000 | [diff] [blame] | 177 | SkAutoTUnref<SkStream> stream(this->openStream(&ttcIndex)); |
reed@google.com | 5bfc839 | 2013-03-05 22:12:11 +0000 | [diff] [blame] | 178 | return stream.get() |
| 179 | ? SkFontStream::GetTableTags(stream, ttcIndex, tags) |
| 180 | : 0; |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 181 | } |
| 182 | |
| 183 | size_t FontConfigTypeface::onGetTableData(SkFontTableTag tag, size_t offset, |
| 184 | size_t length, void* data) const { |
reed@google.com | 5bfc839 | 2013-03-05 22:12:11 +0000 | [diff] [blame] | 185 | int ttcIndex; |
reed@google.com | 2cdc671 | 2013-03-21 18:22:00 +0000 | [diff] [blame] | 186 | SkAutoTUnref<SkStream> stream(this->openStream(&ttcIndex)); |
reed@google.com | 5bfc839 | 2013-03-05 22:12:11 +0000 | [diff] [blame] | 187 | return stream.get() |
| 188 | ? SkFontStream::GetTableData(stream, ttcIndex, |
| 189 | tag, offset, length, data) |
| 190 | : 0; |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 191 | } |
| 192 | |
reed@google.com | 5526ede | 2013-03-25 13:03:37 +0000 | [diff] [blame] | 193 | void FontConfigTypeface::onGetFontDescriptor(SkFontDescriptor* desc, |
| 194 | bool* isLocalStream) const { |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 195 | desc->setFamilyName(this->getFamilyName()); |
reed@google.com | 5526ede | 2013-03-25 13:03:37 +0000 | [diff] [blame] | 196 | *isLocalStream = SkToBool(this->getLocalStream()); |
reed@google.com | 822bde7 | 2013-03-04 16:28:33 +0000 | [diff] [blame] | 197 | } |
reed@google.com | 070da5e | 2013-03-27 20:01:49 +0000 | [diff] [blame] | 198 | |
| 199 | /////////////////////////////////////////////////////////////////////////////// |