Index: content/common/font_warmup_win.cc |
diff --git a/content/common/font_warmup_win.cc b/content/common/font_warmup_win.cc |
deleted file mode 100644 |
index 2968218c231a88281bdca074aa94eafc8bb30dbe..0000000000000000000000000000000000000000 |
--- a/content/common/font_warmup_win.cc |
+++ /dev/null |
@@ -1,529 +0,0 @@ |
-// Copyright 2014 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/common/font_warmup_win.h" |
- |
-#include <dwrite.h> |
-#include <stdint.h> |
-#include <map> |
- |
-#include "base/debug/alias.h" |
-#include "base/files/file_path.h" |
-#include "base/lazy_instance.h" |
-#include "base/logging.h" |
-#include "base/macros.h" |
-#include "base/memory/ref_counted.h" |
-#include "base/numerics/safe_conversions.h" |
-#include "base/numerics/safe_math.h" |
-#include "base/strings/utf_string_conversions.h" |
-#include "base/synchronization/lock.h" |
-#include "base/sys_byteorder.h" |
-#include "base/trace_event/trace_event.h" |
-#include "base/win/iat_patch_function.h" |
-#include "base/win/windows_version.h" |
-#include "content/public/common/dwrite_font_platform_win.h" |
-#include "ppapi/shared_impl/proxy_lock.h" |
-#include "skia/ext/fontmgr_default_win.h" |
-#include "skia/ext/refptr.h" |
-#include "third_party/WebKit/public/web/win/WebFontRendering.h" |
-#include "third_party/skia/include/core/SkPaint.h" |
-#include "third_party/skia/include/ports/SkFontMgr.h" |
-#include "third_party/skia/include/ports/SkTypeface_win.h" |
-#include "ui/gfx/hud_font.h" |
- |
-namespace content { |
- |
-namespace { |
- |
-// The Skia font manager, used for the life of the process (leaked at the end). |
-SkFontMgr* g_warmup_fontmgr = nullptr; |
- |
-base::win::IATPatchFunction g_iat_patch_open_sc_manager; |
-base::win::IATPatchFunction g_iat_patch_close_service_handle; |
-base::win::IATPatchFunction g_iat_patch_open_service; |
-base::win::IATPatchFunction g_iat_patch_start_service; |
-base::win::IATPatchFunction g_iat_patch_nt_connect_port; |
- |
-// These are from ntddk.h |
-#if !defined(STATUS_ACCESS_DENIED) |
-#define STATUS_ACCESS_DENIED ((NTSTATUS)0xC0000022L) |
-#endif |
- |
-typedef LONG NTSTATUS; |
- |
-const uintptr_t kFakeSCMHandle = 0xdead0001; |
-const uintptr_t kFakeServiceHandle = 0xdead0002; |
- |
-SC_HANDLE WINAPI OpenSCManagerWPatch(const wchar_t* machine_name, |
- const wchar_t* database_name, |
- DWORD access_mask) { |
- ::SetLastError(0); |
- return reinterpret_cast<SC_HANDLE>(kFakeSCMHandle); |
-} |
- |
-SC_HANDLE WINAPI OpenServiceWPatch(SC_HANDLE sc_manager, |
- const wchar_t* service_name, |
- DWORD access_mask) { |
- ::SetLastError(0); |
- return reinterpret_cast<SC_HANDLE>(kFakeServiceHandle); |
-} |
- |
-BOOL WINAPI CloseServiceHandlePatch(SC_HANDLE service_handle) { |
- if (service_handle != reinterpret_cast<SC_HANDLE>(kFakeServiceHandle) && |
- service_handle != reinterpret_cast<SC_HANDLE>(kFakeSCMHandle)) |
- CHECK(false); |
- ::SetLastError(0); |
- return TRUE; |
-} |
- |
-BOOL WINAPI StartServiceWPatch(SC_HANDLE service, |
- DWORD args, |
- const wchar_t** arg_vectors) { |
- if (service != reinterpret_cast<SC_HANDLE>(kFakeServiceHandle)) |
- CHECK(false); |
- ::SetLastError(ERROR_ACCESS_DENIED); |
- return FALSE; |
-} |
- |
-NTSTATUS WINAPI NtALpcConnectPortPatch(HANDLE* port_handle, |
- void* port_name, |
- void* object_attribs, |
- void* port_attribs, |
- DWORD flags, |
- void* server_sid, |
- void* message, |
- DWORD* buffer_length, |
- void* out_message_attributes, |
- void* in_message_attributes, |
- void* time_out) { |
- return STATUS_ACCESS_DENIED; |
-} |
- |
-// Windows-only DirectWrite support. These warm up the DirectWrite paths |
-// before sandbox lock down to allow Skia access to the Font Manager service. |
-void CreateDirectWriteFactory(IDWriteFactory** factory) { |
- typedef decltype(DWriteCreateFactory)* DWriteCreateFactoryProc; |
- HMODULE dwrite_dll = LoadLibraryW(L"dwrite.dll"); |
- // TODO(scottmg): Temporary code to track crash in http://crbug.com/387867. |
- if (!dwrite_dll) { |
- DWORD load_library_get_last_error = GetLastError(); |
- base::debug::Alias(&dwrite_dll); |
- base::debug::Alias(&load_library_get_last_error); |
- CHECK(false); |
- } |
- |
- PatchServiceManagerCalls(); |
- |
- DWriteCreateFactoryProc dwrite_create_factory_proc = |
- reinterpret_cast<DWriteCreateFactoryProc>( |
- GetProcAddress(dwrite_dll, "DWriteCreateFactory")); |
- // TODO(scottmg): Temporary code to track crash in http://crbug.com/387867. |
- if (!dwrite_create_factory_proc) { |
- DWORD get_proc_address_get_last_error = GetLastError(); |
- base::debug::Alias(&dwrite_create_factory_proc); |
- base::debug::Alias(&get_proc_address_get_last_error); |
- CHECK(false); |
- } |
- CHECK(SUCCEEDED(dwrite_create_factory_proc( |
- DWRITE_FACTORY_TYPE_ISOLATED, __uuidof(IDWriteFactory), |
- reinterpret_cast<IUnknown**>(factory)))); |
-} |
- |
-HRESULT STDMETHODCALLTYPE StubFontCollection(IDWriteFactory* factory, |
- IDWriteFontCollection** col, |
- BOOL checkUpdates) { |
- // We always return pre-created font collection from here. |
- IDWriteFontCollection* custom_collection = GetCustomFontCollection(factory); |
- DCHECK(custom_collection != nullptr); |
- *col = custom_collection; |
- return S_OK; |
-} |
- |
-void PatchDWriteFactory(IDWriteFactory* factory) { |
- const unsigned int kGetSystemFontCollectionVTableIndex = 3; |
- |
- PROC* vtable = *reinterpret_cast<PROC**>(factory); |
- PROC* function_ptr = &vtable[kGetSystemFontCollectionVTableIndex]; |
- void* stub_function = &StubFontCollection; |
- base::win::ModifyCode(function_ptr, &stub_function, sizeof(PROC)); |
-} |
- |
-// Class to fake out a DC or a Font object. Maintains a reference to a |
-// SkTypeFace to emulate the simple operation of a DC and Font. |
-class FakeGdiObject : public base::RefCountedThreadSafe<FakeGdiObject> { |
- public: |
- FakeGdiObject(uint32_t magic, void* handle) |
- : handle_(handle), magic_(magic) {} |
- |
- void set_typeface(const skia::RefPtr<SkTypeface>& typeface) { |
- typeface_ = typeface; |
- } |
- |
- skia::RefPtr<SkTypeface> typeface() { return typeface_; } |
- void* handle() { return handle_; } |
- uint32_t magic() { return magic_; } |
- |
- private: |
- friend class base::RefCountedThreadSafe<FakeGdiObject>; |
- ~FakeGdiObject() {} |
- |
- void* handle_; |
- uint32_t magic_; |
- skia::RefPtr<SkTypeface> typeface_; |
- |
- DISALLOW_COPY_AND_ASSIGN(FakeGdiObject); |
-}; |
- |
-// This class acts as a factory for creating new fake GDI objects. It also maps |
-// the new instances of the FakeGdiObject class to an incrementing handle value |
-// which is passed to the caller of the emulated GDI function for later |
-// reference. We can't be sure that this won't be used in a multi-threaded |
-// environment so we need to ensure a lock is taken before accessing the map of |
-// issued objects. |
-class FakeGdiObjectFactory { |
- public: |
- FakeGdiObjectFactory() : curr_handle_(0) {} |
- |
- // Find a corresponding fake GDI object and verify its magic value. |
- // The returned value is either nullptr or the validated object. |
- scoped_refptr<FakeGdiObject> Validate(void* obj, uint32_t magic) { |
- if (obj) { |
- base::AutoLock scoped_lock(objects_lock_); |
- auto handle_entry = objects_.find(obj); |
- if (handle_entry != objects_.end() && |
- handle_entry->second->magic() == magic) { |
- return handle_entry->second; |
- } |
- } |
- return nullptr; |
- } |
- |
- scoped_refptr<FakeGdiObject> Create(uint32_t magic) { |
- base::AutoLock scoped_lock(objects_lock_); |
- curr_handle_++; |
- // We don't support wrapping the fake handle value. |
- void* handle = reinterpret_cast<void*>(curr_handle_.ValueOrDie()); |
- scoped_refptr<FakeGdiObject> object(new FakeGdiObject(magic, handle)); |
- objects_[handle] = object; |
- return object; |
- } |
- |
- bool DeleteObject(void* obj, uint32_t magic) { |
- base::AutoLock scoped_lock(objects_lock_); |
- auto handle_entry = objects_.find(obj); |
- if (handle_entry != objects_.end() && |
- handle_entry->second->magic() == magic) { |
- objects_.erase(handle_entry); |
- return true; |
- } |
- return false; |
- } |
- |
- size_t GetObjectCount() { |
- base::AutoLock scoped_lock(objects_lock_); |
- return objects_.size(); |
- } |
- |
- void ResetObjectHandles() { |
- base::AutoLock scoped_lock(objects_lock_); |
- curr_handle_ = 0; |
- objects_.clear(); |
- } |
- |
- private: |
- base::CheckedNumeric<uintptr_t> curr_handle_; |
- std::map<void*, scoped_refptr<FakeGdiObject>> objects_; |
- base::Lock objects_lock_; |
- |
- DISALLOW_COPY_AND_ASSIGN(FakeGdiObjectFactory); |
-}; |
- |
-base::LazyInstance<FakeGdiObjectFactory>::Leaky g_fake_gdi_object_factory = |
- LAZY_INSTANCE_INITIALIZER; |
- |
-// Magic values for the fake GDI objects. |
-const uint32_t kFakeDCMagic = 'fkdc'; |
-const uint32_t kFakeFontMagic = 'fkft'; |
- |
-skia::RefPtr<SkTypeface> GetTypefaceFromLOGFONT(const LOGFONTW* log_font) { |
- CHECK(g_warmup_fontmgr); |
- int weight = log_font->lfWeight; |
- if (weight == FW_DONTCARE) |
- weight = SkFontStyle::kNormal_Weight; |
- |
- SkFontStyle style(weight, log_font->lfWidth, |
- log_font->lfItalic ? SkFontStyle::kItalic_Slant |
- : SkFontStyle::kUpright_Slant); |
- |
- std::string family_name = base::WideToUTF8(log_font->lfFaceName); |
- ppapi::ProxyAutoLock lock; // Needed for DirectWrite font proxy. |
- return skia::AdoptRef( |
- g_warmup_fontmgr->matchFamilyStyle(family_name.c_str(), style)); |
-} |
- |
-HDC WINAPI CreateCompatibleDCPatch(HDC dc_handle) { |
- scoped_refptr<FakeGdiObject> ret = |
- g_fake_gdi_object_factory.Get().Create(kFakeDCMagic); |
- return static_cast<HDC>(ret->handle()); |
-} |
- |
-HFONT WINAPI CreateFontIndirectWPatch(const LOGFONTW* log_font) { |
- if (!log_font) |
- return nullptr; |
- |
- skia::RefPtr<SkTypeface> typeface = GetTypefaceFromLOGFONT(log_font); |
- if (!typeface) |
- return nullptr; |
- |
- scoped_refptr<FakeGdiObject> ret = |
- g_fake_gdi_object_factory.Get().Create(kFakeFontMagic); |
- ret->set_typeface(typeface); |
- |
- return static_cast<HFONT>(ret->handle()); |
-} |
- |
-BOOL WINAPI DeleteDCPatch(HDC dc_handle) { |
- return g_fake_gdi_object_factory.Get().DeleteObject(dc_handle, kFakeDCMagic); |
-} |
- |
-BOOL WINAPI DeleteObjectPatch(HGDIOBJ object_handle) { |
- return g_fake_gdi_object_factory.Get().DeleteObject(object_handle, |
- kFakeFontMagic); |
-} |
- |
-int WINAPI EnumFontFamiliesExWPatch(HDC dc_handle, |
- LPLOGFONTW log_font, |
- FONTENUMPROCW enum_callback, |
- LPARAM callback_param, |
- DWORD flags) { |
- scoped_refptr<FakeGdiObject> dc_obj = |
- g_fake_gdi_object_factory.Get().Validate(dc_handle, kFakeDCMagic); |
- if (!dc_obj) |
- return 1; |
- |
- if (!log_font || !enum_callback) |
- return 1; |
- |
- skia::RefPtr<SkTypeface> typeface = GetTypefaceFromLOGFONT(log_font); |
- if (!typeface) |
- return 1; |
- |
- ENUMLOGFONTEXDVW enum_log_font = {}; |
- enum_log_font.elfEnumLogfontEx.elfLogFont = *log_font; |
- // TODO: Fill in the rest of the text metric structure. Not yet needed for |
- // Flash support but might be in the future. |
- NEWTEXTMETRICEXW text_metric = {}; |
- text_metric.ntmTm.ntmFlags = NTM_PS_OPENTYPE; |
- |
- return enum_callback(&enum_log_font.elfEnumLogfontEx.elfLogFont, |
- reinterpret_cast<TEXTMETRIC*>(&text_metric), |
- TRUETYPE_FONTTYPE, callback_param); |
-} |
- |
-DWORD WINAPI GetFontDataPatch(HDC dc_handle, |
- DWORD table_tag, |
- DWORD table_offset, |
- LPVOID buffer, |
- DWORD buffer_length) { |
- scoped_refptr<FakeGdiObject> dc_obj = |
- g_fake_gdi_object_factory.Get().Validate(dc_handle, kFakeDCMagic); |
- if (!dc_obj) |
- return GDI_ERROR; |
- |
- skia::RefPtr<SkTypeface> typeface = dc_obj->typeface(); |
- if (!typeface) |
- return GDI_ERROR; |
- |
- // |getTableData| handles |buffer| being nullptr. However if it is nullptr |
- // then set the size to INT32_MAX otherwise |getTableData| will return the |
- // minimum value between the table entry size and the size passed in. The |
- // common Windows idiom is to pass 0 as |buffer_length| when passing nullptr, |
- // which would in this case result in |getTableData| returning 0 which isn't |
- // the correct answer for emulating GDI. |table_tag| must also have its |
- // byte order swapped to counter the swap which occurs in the called method. |
- size_t length = typeface->getTableData( |
- base::ByteSwap(base::strict_cast<uint32_t>(table_tag)), table_offset, |
- buffer ? buffer_length : INT32_MAX, buffer); |
- // We can't distinguish between an empty table and an error. |
- if (length == 0) |
- return GDI_ERROR; |
- |
- return base::checked_cast<DWORD>(length); |
-} |
- |
-HGDIOBJ WINAPI SelectObjectPatch(HDC dc_handle, HGDIOBJ object_handle) { |
- scoped_refptr<FakeGdiObject> dc_obj = |
- g_fake_gdi_object_factory.Get().Validate(dc_handle, kFakeDCMagic); |
- if (!dc_obj) |
- return nullptr; |
- |
- scoped_refptr<FakeGdiObject> font_obj = |
- g_fake_gdi_object_factory.Get().Validate(object_handle, kFakeFontMagic); |
- if (!font_obj) |
- return nullptr; |
- |
- // Construct a new fake font object to handle the old font if there's one. |
- scoped_refptr<FakeGdiObject> new_font_obj; |
- skia::RefPtr<SkTypeface> old_typeface = dc_obj->typeface(); |
- if (old_typeface) { |
- new_font_obj = g_fake_gdi_object_factory.Get().Create(kFakeFontMagic); |
- new_font_obj->set_typeface(old_typeface); |
- } |
- dc_obj->set_typeface(font_obj->typeface()); |
- |
- if (new_font_obj) |
- return static_cast<HGDIOBJ>(new_font_obj->handle()); |
- return nullptr; |
-} |
- |
-void DoSingleGdiPatch(base::win::IATPatchFunction& patch, |
- const base::FilePath& path, |
- const char* function_name, |
- void* new_function) { |
- patch.Patch(path.value().c_str(), "gdi32.dll", function_name, new_function); |
-} |
- |
-class GdiFontPatchDataImpl : public content::GdiFontPatchData { |
- public: |
- GdiFontPatchDataImpl(const base::FilePath& path); |
- |
- private: |
- base::win::IATPatchFunction create_compatible_dc_patch_; |
- base::win::IATPatchFunction create_font_indirect_patch_; |
- base::win::IATPatchFunction create_delete_dc_patch_; |
- base::win::IATPatchFunction create_delete_object_patch_; |
- base::win::IATPatchFunction create_enum_font_families_patch_; |
- base::win::IATPatchFunction create_get_font_data_patch_; |
- base::win::IATPatchFunction create_select_object_patch_; |
-}; |
- |
-GdiFontPatchDataImpl::GdiFontPatchDataImpl(const base::FilePath& path) { |
- DoSingleGdiPatch(create_compatible_dc_patch_, path, "CreateCompatibleDC", |
- CreateCompatibleDCPatch); |
- DoSingleGdiPatch(create_font_indirect_patch_, path, "CreateFontIndirectW", |
- CreateFontIndirectWPatch); |
- DoSingleGdiPatch(create_delete_dc_patch_, path, "DeleteDC", DeleteDCPatch); |
- DoSingleGdiPatch(create_delete_object_patch_, path, "DeleteObject", |
- DeleteObjectPatch); |
- DoSingleGdiPatch(create_enum_font_families_patch_, path, |
- "EnumFontFamiliesExW", EnumFontFamiliesExWPatch); |
- DoSingleGdiPatch(create_get_font_data_patch_, path, "GetFontData", |
- GetFontDataPatch); |
- DoSingleGdiPatch(create_select_object_patch_, path, "SelectObject", |
- SelectObjectPatch); |
-} |
- |
-} // namespace |
- |
-// Directwrite connects to the font cache service to retrieve information about |
-// fonts installed on the system etc. This works well outside the sandbox and |
-// within the sandbox as long as the lpc connection maintained by the current |
-// process with the font cache service remains valid. It appears that there |
-// are cases when this connection is dropped after which directwrite is unable |
-// to connect to the font cache service which causes problems with characters |
-// disappearing. |
-// Directwrite has fallback code to enumerate fonts if it is unable to connect |
-// to the font cache service. We need to intercept the following APIs to |
-// ensure that it does not connect to the font cache service. |
-// NtALpcConnectPort |
-// OpenSCManagerW |
-// OpenServiceW |
-// StartServiceW |
-// CloseServiceHandle. |
-// These are all IAT patched. |
-void PatchServiceManagerCalls() { |
- static bool is_patched = false; |
- if (is_patched) |
- return; |
- const char* service_provider_dll = |
- (base::win::GetVersion() >= base::win::VERSION_WIN8 |
- ? "api-ms-win-service-management-l1-1-0.dll" |
- : "advapi32.dll"); |
- |
- is_patched = true; |
- |
- DWORD patched = |
- g_iat_patch_open_sc_manager.Patch(L"dwrite.dll", service_provider_dll, |
- "OpenSCManagerW", OpenSCManagerWPatch); |
- DCHECK(patched == 0); |
- |
- patched = g_iat_patch_close_service_handle.Patch( |
- L"dwrite.dll", service_provider_dll, "CloseServiceHandle", |
- CloseServiceHandlePatch); |
- DCHECK(patched == 0); |
- |
- patched = g_iat_patch_open_service.Patch(L"dwrite.dll", service_provider_dll, |
- "OpenServiceW", OpenServiceWPatch); |
- DCHECK(patched == 0); |
- |
- patched = g_iat_patch_start_service.Patch( |
- L"dwrite.dll", service_provider_dll, "StartServiceW", StartServiceWPatch); |
- DCHECK(patched == 0); |
- |
- patched = g_iat_patch_nt_connect_port.Patch( |
- L"dwrite.dll", "ntdll.dll", "NtAlpcConnectPort", NtALpcConnectPortPatch); |
- DCHECK(patched == 0); |
-} |
- |
-void DoPreSandboxWarmupForTypeface(SkTypeface* typeface) { |
- SkPaint paint_warmup; |
- paint_warmup.setTypeface(typeface); |
- wchar_t glyph = L'S'; |
- paint_warmup.measureText(&glyph, 2); |
-} |
- |
-SkFontMgr* GetPreSandboxWarmupFontMgr() { |
- if (!g_warmup_fontmgr) { |
- IDWriteFactory* factory; |
- CreateDirectWriteFactory(&factory); |
- |
- GetCustomFontCollection(factory); |
- |
- PatchDWriteFactory(factory); |
- |
- blink::WebFontRendering::setDirectWriteFactory(factory); |
- g_warmup_fontmgr = SkFontMgr_New_DirectWrite(factory); |
- } |
- return g_warmup_fontmgr; |
-} |
- |
-GdiFontPatchData* PatchGdiFontEnumeration(const base::FilePath& path) { |
- if (ShouldUseDirectWriteFontProxyFieldTrial() && !g_warmup_fontmgr) |
- g_warmup_fontmgr = SkFontMgr_New_DirectWrite(); |
- // If not using the font proxy, we assume |g_warmup_fontmgr| is already |
- // initialized before this function is called. |
- DCHECK(g_warmup_fontmgr); |
- return new GdiFontPatchDataImpl(path); |
-} |
- |
-size_t GetEmulatedGdiHandleCountForTesting() { |
- return g_fake_gdi_object_factory.Get().GetObjectCount(); |
-} |
- |
-void ResetEmulatedGdiHandlesForTesting() { |
- g_fake_gdi_object_factory.Get().ResetObjectHandles(); |
-} |
- |
-void SetPreSandboxWarmupFontMgrForTesting(SkFontMgr* fontmgr) { |
- g_warmup_fontmgr = fontmgr; |
-} |
- |
-void WarmupDirectWrite() { |
- TRACE_EVENT0("startup", "content::WarmupDirectWrite"); |
- |
- // The objects used here are intentionally not freed as we want the Skia |
- // code to use these objects after warmup. |
- SetDefaultSkiaFactory(GetPreSandboxWarmupFontMgr()); |
- |
- // We need to warm up *some* font for DirectWrite. We also need to pass one |
- // down for the CC HUD code, so use the same one here. Note that we don't use |
- // a monospace as would be nice in an attempt to avoid a small startup time |
- // regression, see http://crbug.com/463613. |
- skia::RefPtr<SkTypeface> hud_typeface = skia::AdoptRef( |
- GetPreSandboxWarmupFontMgr()->legacyCreateTypeface("Times New Roman", 0)); |
- DoPreSandboxWarmupForTypeface(hud_typeface.get()); |
- gfx::SetHudTypeface(hud_typeface); |
-} |
- |
-} // namespace content |