| // Copyright 2015 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/child/dwrite_font_proxy/dwrite_font_proxy_win.h" |
| |
| #include <stddef.h> |
| #include <stdint.h> |
| |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/debug/crash_logging.h" |
| #include "base/feature_list.h" |
| #include "base/logging.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/no_destructor.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "base/task/post_task.h" |
| #include "content/child/dwrite_font_proxy/dwrite_localized_strings_win.h" |
| #include "content/public/child/child_thread.h" |
| #include "content/public/common/service_names.mojom.h" |
| #include "services/service_manager/public/cpp/connector.h" |
| |
| namespace mswr = Microsoft::WRL; |
| |
| namespace content { |
| |
| namespace { |
| |
| // This enum is used to define the buckets for an enumerated UMA histogram. |
| // Hence, |
| // (a) existing enumerated constants should never be deleted or reordered, and |
| // (b) new constants should only be appended at the end of the enumeration. |
| enum DirectWriteLoadFamilyResult { |
| LOAD_FAMILY_SUCCESS_SINGLE_FAMILY = 0, |
| LOAD_FAMILY_SUCCESS_MATCHED_FAMILY = 1, |
| LOAD_FAMILY_ERROR_MULTIPLE_FAMILIES = 2, |
| LOAD_FAMILY_ERROR_NO_FAMILIES = 3, |
| LOAD_FAMILY_ERROR_NO_COLLECTION = 4, |
| |
| LOAD_FAMILY_MAX_VALUE |
| }; |
| |
| // This enum is used to define the buckets for an enumerated UMA histogram. |
| // Hence, |
| // (a) existing enumerated constants should never be deleted or reordered, and |
| // (b) new constants should only be appended at the end of the enumeration. |
| enum FontProxyError { |
| FIND_FAMILY_SEND_FAILED = 0, |
| GET_FAMILY_COUNT_SEND_FAILED = 1, |
| COLLECTION_KEY_INVALID = 2, |
| FAMILY_INDEX_OUT_OF_RANGE = 3, |
| GET_FONT_FILES_SEND_FAILED = 4, |
| MAPPED_FILE_FAILED = 5, |
| DUPLICATE_HANDLE_FAILED = 6, |
| |
| FONT_PROXY_ERROR_MAX_VALUE |
| }; |
| |
| void LogLoadFamilyResult(DirectWriteLoadFamilyResult result) { |
| UMA_HISTOGRAM_ENUMERATION("DirectWrite.Fonts.Proxy.LoadFamilyResult", result, |
| LOAD_FAMILY_MAX_VALUE); |
| } |
| |
| void LogFamilyCount(uint32_t count) { |
| UMA_HISTOGRAM_COUNTS_1000("DirectWrite.Fonts.Proxy.FamilyCount", count); |
| } |
| |
| void LogFontProxyError(FontProxyError error) { |
| UMA_HISTOGRAM_ENUMERATION("DirectWrite.Fonts.Proxy.FontProxyError", error, |
| FONT_PROXY_ERROR_MAX_VALUE); |
| } |
| |
| } // namespace |
| |
| HRESULT DWriteFontCollectionProxy::Create( |
| DWriteFontCollectionProxy** proxy_out, |
| IDWriteFactory* dwrite_factory, |
| blink::mojom::DWriteFontProxyPtrInfo proxy) { |
| return Microsoft::WRL::MakeAndInitialize<DWriteFontCollectionProxy>( |
| proxy_out, dwrite_factory, std::move(proxy)); |
| } |
| |
| DWriteFontCollectionProxy::DWriteFontCollectionProxy() = default; |
| |
| DWriteFontCollectionProxy::~DWriteFontCollectionProxy() = default; |
| |
| HRESULT DWriteFontCollectionProxy::FindFamilyName(const WCHAR* family_name, |
| UINT32* index, |
| BOOL* exists) { |
| DCHECK(family_name); |
| DCHECK(index); |
| DCHECK(exists); |
| TRACE_EVENT0("dwrite", "FontProxy::FindFamilyName"); |
| |
| uint32_t family_index = 0; |
| base::string16 name(family_name); |
| |
| auto iter = family_names_.find(name); |
| if (iter != family_names_.end()) { |
| *index = iter->second; |
| *exists = iter->second != UINT_MAX; |
| return S_OK; |
| } |
| |
| if (!GetFontProxy().FindFamily(name, &family_index)) { |
| LogFontProxyError(FIND_FAMILY_SEND_FAILED); |
| // TODO(https://crbug.com/922403): We're inserting an assertion here to |
| // gather crash reports and find more about clients in which this |
| // fails. Investigate crash reports resulting from this failure, then |
| // remove this assertion. |
| CHECK(false); |
| return E_FAIL; |
| } |
| |
| if (family_index != UINT32_MAX) { |
| if (!CreateFamily(family_index)) |
| return E_FAIL; |
| *exists = TRUE; |
| *index = family_index; |
| families_[family_index]->SetName(name); |
| } else { |
| *exists = FALSE; |
| *index = UINT32_MAX; |
| } |
| |
| family_names_[name] = *index; |
| return S_OK; |
| } |
| |
| HRESULT DWriteFontCollectionProxy::GetFontFamily( |
| UINT32 index, |
| IDWriteFontFamily** font_family) { |
| DCHECK(font_family); |
| |
| if (index < families_.size() && families_[index]) { |
| families_[index].CopyTo(font_family); |
| return S_OK; |
| } |
| |
| if (!CreateFamily(index)) |
| return E_FAIL; |
| |
| families_[index].CopyTo(font_family); |
| return S_OK; |
| } |
| |
| UINT32 DWriteFontCollectionProxy::GetFontFamilyCount() { |
| if (family_count_ != UINT_MAX) |
| return family_count_; |
| |
| TRACE_EVENT0("dwrite", "FontProxy::GetFontFamilyCount"); |
| |
| uint32_t family_count = 0; |
| if (!GetFontProxy().GetFamilyCount(&family_count)) { |
| LogFontProxyError(GET_FAMILY_COUNT_SEND_FAILED); |
| return 0; |
| } |
| |
| LogFamilyCount(family_count); |
| family_count_ = family_count; |
| return family_count; |
| } |
| |
| HRESULT DWriteFontCollectionProxy::GetFontFromFontFace( |
| IDWriteFontFace* font_face, |
| IDWriteFont** font) { |
| DCHECK(font_face); |
| DCHECK(font); |
| |
| for (const auto& family : families_) { |
| if (family && family->GetFontFromFontFace(font_face, font)) { |
| return S_OK; |
| } |
| } |
| // If the font came from our collection, at least one family should match |
| DCHECK(false); |
| |
| return E_FAIL; |
| } |
| |
| HRESULT DWriteFontCollectionProxy::CreateEnumeratorFromKey( |
| IDWriteFactory* factory, |
| const void* collection_key, |
| UINT32 collection_key_size, |
| IDWriteFontFileEnumerator** font_file_enumerator) { |
| if (!collection_key || collection_key_size != sizeof(uint32_t)) { |
| LogFontProxyError(COLLECTION_KEY_INVALID); |
| return E_INVALIDARG; |
| } |
| |
| TRACE_EVENT0("dwrite", "FontProxy::LoadingFontFiles"); |
| |
| const uint32_t* family_index = |
| reinterpret_cast<const uint32_t*>(collection_key); |
| |
| if (*family_index >= GetFontFamilyCount()) { |
| LogFontProxyError(FAMILY_INDEX_OUT_OF_RANGE); |
| return E_INVALIDARG; |
| } |
| |
| // If we already loaded the family we should reuse the existing collection. |
| DCHECK(!families_[*family_index]->IsLoaded()); |
| |
| std::vector<base::FilePath> file_names; |
| std::vector<base::File> file_handles; |
| if (!GetFontProxy().GetFontFiles(*family_index, &file_names, &file_handles)) { |
| LogFontProxyError(GET_FONT_FILES_SEND_FAILED); |
| return E_FAIL; |
| } |
| |
| std::vector<HANDLE> handles; |
| handles.reserve(file_names.size() + file_handles.size()); |
| for (const base::FilePath& file_name : file_names) { |
| // This leaks the handles, since they are used as the reference key to |
| // CreateStreamFromKey, and DirectWrite requires the reference keys to |
| // remain valid for the lifetime of the loader. The loader is the font |
| // collection proxy, which remains alive for the lifetime of the renderer. |
| HANDLE handle = |
| CreateFile(file_name.value().c_str(), GENERIC_READ, FILE_SHARE_READ, |
| NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); |
| if (handle != INVALID_HANDLE_VALUE) |
| handles.push_back(handle); |
| } |
| for (auto& file_handle : file_handles) { |
| handles.push_back(file_handle.TakePlatformFile()); |
| } |
| |
| HRESULT hr = mswr::MakeAndInitialize<FontFileEnumerator>( |
| font_file_enumerator, factory, this, &handles); |
| |
| if (!SUCCEEDED(hr)) { |
| DCHECK(false); |
| return E_FAIL; |
| } |
| |
| return S_OK; |
| } |
| |
| HRESULT DWriteFontCollectionProxy::CreateStreamFromKey( |
| const void* font_file_reference_key, |
| UINT32 font_file_reference_key_size, |
| IDWriteFontFileStream** font_file_stream) { |
| if (font_file_reference_key_size != sizeof(HANDLE)) { |
| return E_FAIL; |
| } |
| |
| TRACE_EVENT0("dwrite", "FontFileEnumerator::CreateStreamFromKey"); |
| |
| HANDLE file_handle = |
| *reinterpret_cast<const HANDLE*>(font_file_reference_key); |
| |
| if (file_handle == NULL || file_handle == INVALID_HANDLE_VALUE) { |
| DCHECK(false); |
| return E_FAIL; |
| } |
| |
| mswr::ComPtr<FontFileStream> stream; |
| if (!SUCCEEDED( |
| mswr::MakeAndInitialize<FontFileStream>(&stream, file_handle))) { |
| DCHECK(false); |
| return E_FAIL; |
| } |
| *font_file_stream = stream.Detach(); |
| return S_OK; |
| } |
| |
| HRESULT DWriteFontCollectionProxy::RuntimeClassInitialize( |
| IDWriteFactory* factory, |
| blink::mojom::DWriteFontProxyPtrInfo proxy) { |
| DCHECK(factory); |
| |
| factory_ = factory; |
| if (proxy) |
| SetProxy(std::move(proxy)); |
| else |
| main_task_runner_ = base::ThreadTaskRunnerHandle::Get(); |
| |
| HRESULT hr = factory->RegisterFontCollectionLoader(this); |
| DCHECK(SUCCEEDED(hr)); |
| hr = factory_->RegisterFontFileLoader(this); |
| DCHECK(SUCCEEDED(hr)); |
| return S_OK; |
| } |
| |
| void DWriteFontCollectionProxy::Unregister() { |
| factory_->UnregisterFontCollectionLoader(this); |
| factory_->UnregisterFontFileLoader(this); |
| } |
| |
| bool DWriteFontCollectionProxy::LoadFamily( |
| UINT32 family_index, |
| IDWriteFontCollection** containing_collection) { |
| TRACE_EVENT0("dwrite", "FontProxy::LoadFamily"); |
| |
| uint32_t index = family_index; |
| // CreateCustomFontCollection ends up calling |
| // DWriteFontCollectionProxy::CreateEnumeratorFromKey. |
| HRESULT hr = factory_->CreateCustomFontCollection( |
| this /*collectionLoader*/, reinterpret_cast<const void*>(&index), |
| sizeof(index), containing_collection); |
| |
| return SUCCEEDED(hr); |
| } |
| |
| bool DWriteFontCollectionProxy::GetFontFamily(UINT32 family_index, |
| const base::string16& family_name, |
| IDWriteFontFamily** font_family) { |
| DCHECK(font_family); |
| DCHECK(!family_name.empty()); |
| if (!CreateFamily(family_index)) |
| return false; |
| |
| mswr::ComPtr<DWriteFontFamilyProxy>& family = families_[family_index]; |
| if (!family->IsLoaded() || family->GetName().empty()) |
| family->SetName(family_name); |
| |
| family.CopyTo(font_family); |
| return true; |
| } |
| |
| bool DWriteFontCollectionProxy::LoadFamilyNames( |
| UINT32 family_index, |
| IDWriteLocalizedStrings** localized_strings) { |
| TRACE_EVENT0("dwrite", "FontProxy::LoadFamilyNames"); |
| |
| std::vector<blink::mojom::DWriteStringPairPtr> pairs; |
| if (!GetFontProxy().GetFamilyNames(family_index, &pairs)) { |
| return false; |
| } |
| std::vector<std::pair<base::string16, base::string16>> strings; |
| for (auto& pair : pairs) { |
| strings.emplace_back(std::move(pair->first), std::move(pair->second)); |
| } |
| |
| HRESULT hr = mswr::MakeAndInitialize<DWriteLocalizedStrings>( |
| localized_strings, &strings); |
| |
| return SUCCEEDED(hr); |
| } |
| |
| bool DWriteFontCollectionProxy::CreateFamily(UINT32 family_index) { |
| if (family_index < families_.size() && families_[family_index]) |
| return true; |
| |
| UINT32 family_count = GetFontFamilyCount(); |
| if (family_index >= family_count) { |
| return false; |
| } |
| |
| if (families_.size() < family_count) |
| families_.resize(family_count); |
| |
| mswr::ComPtr<DWriteFontFamilyProxy> family; |
| HRESULT hr = mswr::MakeAndInitialize<DWriteFontFamilyProxy>(&family, this, |
| family_index); |
| DCHECK(SUCCEEDED(hr)); |
| DCHECK_LT(family_index, families_.size()); |
| |
| families_[family_index] = family; |
| return true; |
| } |
| |
| void DWriteFontCollectionProxy::SetProxy( |
| blink::mojom::DWriteFontProxyPtrInfo proxy) { |
| font_proxy_ = blink::mojom::ThreadSafeDWriteFontProxyPtr::Create( |
| std::move(proxy), base::CreateSequencedTaskRunnerWithTraits( |
| {base::WithBaseSyncPrimitives()})); |
| } |
| |
| blink::mojom::DWriteFontProxy& DWriteFontCollectionProxy::GetFontProxy() { |
| if (!font_proxy_) { |
| blink::mojom::DWriteFontProxyPtrInfo dwrite_font_proxy; |
| if (main_task_runner_->RunsTasksInCurrentSequence()) { |
| ChildThread::Get()->GetConnector()->BindInterface( |
| mojom::kBrowserServiceName, mojo::MakeRequest(&dwrite_font_proxy)); |
| } else { |
| main_task_runner_->PostTask( |
| FROM_HERE, base::BindOnce( |
| [](blink::mojom::DWriteFontProxyRequest request) { |
| ChildThread::Get()->GetConnector()->BindInterface( |
| mojom::kBrowserServiceName, std::move(request)); |
| }, |
| mojo::MakeRequest(&dwrite_font_proxy))); |
| } |
| SetProxy(std::move(dwrite_font_proxy)); |
| } |
| return **font_proxy_; |
| } |
| |
| DWriteFontFamilyProxy::DWriteFontFamilyProxy() = default; |
| |
| DWriteFontFamilyProxy::~DWriteFontFamilyProxy() = default; |
| |
| HRESULT DWriteFontFamilyProxy::GetFontCollection( |
| IDWriteFontCollection** font_collection) { |
| DCHECK(font_collection); |
| |
| proxy_collection_.CopyTo(font_collection); |
| return S_OK; |
| } |
| |
| UINT32 DWriteFontFamilyProxy::GetFontCount() { |
| // We could conceivably proxy just the font count. However, calling |
| // GetFontCount is almost certain to be followed by a series of GetFont |
| // calls which will need to load all the fonts anyway, so we might as |
| // well save an IPC here. |
| if (!LoadFamily()) |
| return 0; |
| |
| return family_->GetFontCount(); |
| } |
| |
| HRESULT DWriteFontFamilyProxy::GetFont(UINT32 index, IDWriteFont** font) { |
| DCHECK(font); |
| |
| if (index >= GetFontCount()) { |
| return E_INVALIDARG; |
| } |
| if (!LoadFamily()) |
| return E_FAIL; |
| |
| return family_->GetFont(index, font); |
| } |
| |
| HRESULT DWriteFontFamilyProxy::GetFamilyNames(IDWriteLocalizedStrings** names) { |
| DCHECK(names); |
| |
| // Prefer the real thing, if available. |
| if (family_) { |
| family_names_.Reset(); // Release cached data. |
| return family_->GetFamilyNames(names); |
| } |
| |
| // If already cached, use the cache. |
| if (family_names_) { |
| family_names_.CopyTo(names); |
| return S_OK; |
| } |
| |
| TRACE_EVENT0("dwrite", "FontProxy::GetFamilyNames"); |
| |
| // Otherwise, do the IPC. |
| if (!proxy_collection_->LoadFamilyNames(family_index_, &family_names_)) |
| return E_FAIL; |
| |
| family_names_.CopyTo(names); |
| return S_OK; |
| } |
| |
| HRESULT DWriteFontFamilyProxy::GetFirstMatchingFont( |
| DWRITE_FONT_WEIGHT weight, |
| DWRITE_FONT_STRETCH stretch, |
| DWRITE_FONT_STYLE style, |
| IDWriteFont** matching_font) { |
| DCHECK(matching_font); |
| |
| if (!LoadFamily()) |
| return E_FAIL; |
| |
| return family_->GetFirstMatchingFont(weight, stretch, style, matching_font); |
| } |
| |
| HRESULT DWriteFontFamilyProxy::GetMatchingFonts( |
| DWRITE_FONT_WEIGHT weight, |
| DWRITE_FONT_STRETCH stretch, |
| DWRITE_FONT_STYLE style, |
| IDWriteFontList** matching_fonts) { |
| DCHECK(matching_fonts); |
| |
| if (!LoadFamily()) |
| return E_FAIL; |
| |
| return family_->GetMatchingFonts(weight, stretch, style, matching_fonts); |
| } |
| |
| HRESULT DWriteFontFamilyProxy::RuntimeClassInitialize( |
| DWriteFontCollectionProxy* collection, |
| UINT32 index) { |
| DCHECK(collection); |
| |
| proxy_collection_ = collection; |
| family_index_ = index; |
| return S_OK; |
| } |
| |
| bool DWriteFontFamilyProxy::GetFontFromFontFace(IDWriteFontFace* font_face, |
| IDWriteFont** font) { |
| DCHECK(font_face); |
| DCHECK(font); |
| |
| if (!family_) |
| return false; |
| |
| mswr::ComPtr<IDWriteFontCollection> collection; |
| HRESULT hr = family_->GetFontCollection(&collection); |
| DCHECK(SUCCEEDED(hr)); |
| hr = collection->GetFontFromFontFace(font_face, font); |
| |
| return SUCCEEDED(hr); |
| } |
| |
| void DWriteFontFamilyProxy::SetName(const base::string16& family_name) { |
| family_name_.assign(family_name); |
| } |
| |
| const base::string16& DWriteFontFamilyProxy::GetName() { |
| return family_name_; |
| } |
| |
| bool DWriteFontFamilyProxy::IsLoaded() { |
| return family_ != nullptr; |
| } |
| |
| bool DWriteFontFamilyProxy::LoadFamily() { |
| if (family_) |
| return true; |
| |
| SCOPED_UMA_HISTOGRAM_TIMER("DirectWrite.Fonts.Proxy.LoadFamilyTime"); |
| |
| auto* font_key_name = base::debug::AllocateCrashKeyString( |
| "font_key_name", base::debug::CrashKeySize::Size32); |
| base::debug::ScopedCrashKeyString crash_key(font_key_name, |
| base::WideToUTF8(family_name_)); |
| |
| mswr::ComPtr<IDWriteFontCollection> collection; |
| if (!proxy_collection_->LoadFamily(family_index_, &collection)) { |
| LogLoadFamilyResult(LOAD_FAMILY_ERROR_NO_COLLECTION); |
| return false; |
| } |
| |
| UINT32 family_count = collection->GetFontFamilyCount(); |
| |
| HRESULT hr; |
| if (family_count > 1) { |
| // Some fonts are packaged in a single file containing multiple families. In |
| // such a case we can find the right family by family name. |
| DCHECK(!family_name_.empty()); |
| UINT32 family_index = 0; |
| BOOL found = FALSE; |
| hr = |
| collection->FindFamilyName(family_name_.c_str(), &family_index, &found); |
| if (SUCCEEDED(hr) && found) { |
| hr = collection->GetFontFamily(family_index, &family_); |
| LogLoadFamilyResult(LOAD_FAMILY_SUCCESS_MATCHED_FAMILY); |
| return SUCCEEDED(hr); |
| } |
| } |
| |
| DCHECK_LE(family_count, 1u); |
| |
| if (family_count == 0) { |
| // This is really strange, we successfully loaded no fonts?! |
| LogLoadFamilyResult(LOAD_FAMILY_ERROR_NO_FAMILIES); |
| return false; |
| } |
| |
| LogLoadFamilyResult(family_count == 1 ? LOAD_FAMILY_SUCCESS_SINGLE_FAMILY |
| : LOAD_FAMILY_ERROR_MULTIPLE_FAMILIES); |
| |
| hr = collection->GetFontFamily(0, &family_); |
| |
| return SUCCEEDED(hr); |
| } |
| |
| FontFileEnumerator::FontFileEnumerator() = default; |
| |
| FontFileEnumerator::~FontFileEnumerator() = default; |
| |
| HRESULT FontFileEnumerator::GetCurrentFontFile(IDWriteFontFile** file) { |
| DCHECK(file); |
| if (current_file_ >= files_.size()) { |
| return E_FAIL; |
| } |
| |
| TRACE_EVENT0("dwrite", "FontFileEnumerator::GetCurrentFontFile"); |
| |
| // CreateCustomFontFileReference ends up calling |
| // DWriteFontCollectionProxy::CreateStreamFromKey. |
| HRESULT hr = factory_->CreateCustomFontFileReference( |
| reinterpret_cast<const void*>(&files_[current_file_]), |
| sizeof(files_[current_file_]), loader_.Get() /*IDWriteFontFileLoader*/, |
| file); |
| DCHECK(SUCCEEDED(hr)); |
| return hr; |
| } |
| |
| HRESULT FontFileEnumerator::MoveNext(BOOL* has_current_file) { |
| DCHECK(has_current_file); |
| |
| TRACE_EVENT0("dwrite", "FontFileEnumerator::MoveNext"); |
| if (next_file_ >= files_.size()) { |
| *has_current_file = FALSE; |
| current_file_ = UINT_MAX; |
| return S_OK; |
| } |
| |
| current_file_ = next_file_; |
| ++next_file_; |
| *has_current_file = TRUE; |
| return S_OK; |
| } |
| |
| HRESULT FontFileEnumerator::RuntimeClassInitialize( |
| IDWriteFactory* factory, |
| IDWriteFontFileLoader* loader, |
| std::vector<HANDLE>* files) { |
| factory_ = factory; |
| loader_ = loader; |
| files_.swap(*files); |
| return S_OK; |
| } |
| |
| FontFileStream::FontFileStream() = default; |
| |
| FontFileStream::~FontFileStream() = default; |
| |
| HRESULT FontFileStream::GetFileSize(UINT64* file_size) { |
| *file_size = data_.length(); |
| return S_OK; |
| } |
| |
| HRESULT FontFileStream::GetLastWriteTime(UINT64* last_write_time) { |
| *last_write_time = 0; |
| return S_OK; |
| } |
| |
| HRESULT FontFileStream::ReadFileFragment(const void** fragment_start, |
| UINT64 fragment_offset, |
| UINT64 fragment_size, |
| void** fragment_context) { |
| if (fragment_offset + fragment_size < fragment_offset) |
| return E_FAIL; |
| if (fragment_offset + fragment_size > data_.length()) |
| return E_FAIL; |
| *fragment_start = data_.data() + fragment_offset; |
| *fragment_context = nullptr; |
| return S_OK; |
| } |
| |
| HRESULT FontFileStream::RuntimeClassInitialize(HANDLE handle) { |
| // Duplicate the original handle so we can reopen the file after the memory |
| // mapped section closes it. |
| HANDLE duplicate_handle; |
| if (!DuplicateHandle(GetCurrentProcess(), handle, GetCurrentProcess(), |
| &duplicate_handle, 0 /* dwDesiredAccess */, |
| false /* bInheritHandle */, DUPLICATE_SAME_ACCESS)) { |
| LogFontProxyError(DUPLICATE_HANDLE_FAILED); |
| return E_FAIL; |
| } |
| |
| if (!data_.Initialize(base::File(duplicate_handle))) { |
| LogFontProxyError(MAPPED_FILE_FAILED); |
| return E_FAIL; |
| } |
| return S_OK; |
| } |
| |
| } // namespace content |