mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-23 12:51:06 +00:00
40cd36e64d
We check that we can cc and kill the global object on shutdown. At this point it doesn't really matter that the FontFace is still loading, the promise will never resolve. To fix this, implement the "keep alive" by using a target shutdown task. Differential Revision: https://phabricator.services.mozilla.com/D214741
321 lines
8.9 KiB
C++
321 lines
8.9 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/FontFace.h"
|
|
|
|
#include <algorithm>
|
|
#include "gfxFontUtils.h"
|
|
#include "mozilla/dom/CSSFontFaceRule.h"
|
|
#include "mozilla/dom/FontFaceBinding.h"
|
|
#include "mozilla/dom/FontFaceImpl.h"
|
|
#include "mozilla/dom/FontFaceSet.h"
|
|
#include "mozilla/dom/Promise.h"
|
|
#include "mozilla/dom/TypedArray.h"
|
|
#include "mozilla/dom/UnionTypes.h"
|
|
#include "mozilla/CycleCollectedJSContext.h"
|
|
#include "mozilla/ServoBindings.h"
|
|
#include "mozilla/ServoCSSParser.h"
|
|
#include "mozilla/ServoStyleSet.h"
|
|
#include "mozilla/ServoUtils.h"
|
|
#include "mozilla/StaticPrefs_layout.h"
|
|
#include "nsStyleUtil.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
// -- Utility functions ------------------------------------------------------
|
|
|
|
template <typename T>
|
|
static void GetDataFrom(const T& aObject, uint8_t*& aBuffer,
|
|
uint32_t& aLength) {
|
|
MOZ_ASSERT(!aBuffer);
|
|
// We need to use malloc here because the gfxUserFontEntry will be calling
|
|
// free on it, so the Vector's default AllocPolicy (MallocAllocPolicy) is
|
|
// fine.
|
|
Maybe<Vector<uint8_t>> buffer =
|
|
aObject.template CreateFromData<Vector<uint8_t>>();
|
|
if (buffer.isNothing()) {
|
|
return;
|
|
}
|
|
aLength = buffer->length();
|
|
aBuffer = buffer->extractOrCopyRawBuffer();
|
|
}
|
|
|
|
// -- FontFace ---------------------------------------------------------------
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(FontFace)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(FontFace)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mLoaded)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(FontFace)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mLoaded)
|
|
tmp->Destroy();
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(FontFace)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(FontFace)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(FontFace)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(FontFace)
|
|
|
|
FontFace::FontFace(nsIGlobalObject* aParent) : mLoadedRejection(NS_OK) {
|
|
BindToOwner(aParent);
|
|
}
|
|
|
|
FontFace::~FontFace() {
|
|
// Assert that we don't drop any FontFace objects during a Servo traversal,
|
|
// since PostTraversalTask objects can hold raw pointers to FontFaces.
|
|
MOZ_ASSERT(!gfxFontUtils::IsInServoTraversal());
|
|
Destroy();
|
|
}
|
|
|
|
void FontFace::Destroy() { mImpl->Destroy(); }
|
|
|
|
void FontFace::DisconnectFromOwner() {
|
|
GlobalTeardownObserver::DisconnectFromOwner();
|
|
if (mImpl) {
|
|
mImpl->StopKeepingOwnerAlive();
|
|
}
|
|
}
|
|
|
|
JSObject* FontFace::WrapObject(JSContext* aCx,
|
|
JS::Handle<JSObject*> aGivenProto) {
|
|
return FontFace_Binding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
already_AddRefed<FontFace> FontFace::CreateForRule(
|
|
nsIGlobalObject* aGlobal, FontFaceSet* aFontFaceSet,
|
|
StyleLockedFontFaceRule* aRule) {
|
|
FontFaceSetImpl* setImpl = aFontFaceSet->GetImpl();
|
|
MOZ_ASSERT(setImpl);
|
|
|
|
RefPtr<FontFace> obj = new FontFace(aGlobal);
|
|
obj->mImpl = FontFaceImpl::CreateForRule(obj, setImpl, aRule);
|
|
return obj.forget();
|
|
}
|
|
|
|
already_AddRefed<FontFace> FontFace::Constructor(
|
|
const GlobalObject& aGlobal, const nsACString& aFamily,
|
|
const UTF8StringOrArrayBufferOrArrayBufferView& aSource,
|
|
const FontFaceDescriptors& aDescriptors, ErrorResult& aRv) {
|
|
nsCOMPtr<nsIGlobalObject> global = do_QueryInterface(aGlobal.GetAsSupports());
|
|
|
|
FontFaceSet* set = global->GetFonts();
|
|
if (NS_WARN_IF(!set)) {
|
|
aRv.Throw(NS_ERROR_FAILURE);
|
|
return nullptr;
|
|
}
|
|
|
|
FontFaceSetImpl* setImpl = set->GetImpl();
|
|
if (NS_WARN_IF(!setImpl)) {
|
|
aRv.Throw(NS_ERROR_FAILURE);
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<FontFace> obj = new FontFace(global);
|
|
obj->mImpl = new FontFaceImpl(obj, setImpl);
|
|
if (!obj->mImpl->SetDescriptors(aFamily, aDescriptors)) {
|
|
return obj.forget();
|
|
}
|
|
|
|
if (aSource.IsUTF8String()) {
|
|
obj->mImpl->InitializeSourceURL(aSource.GetAsUTF8String());
|
|
} else {
|
|
uint8_t* buffer = nullptr;
|
|
uint32_t length = 0;
|
|
if (aSource.IsArrayBuffer()) {
|
|
GetDataFrom(aSource.GetAsArrayBuffer(), buffer, length);
|
|
} else if (aSource.IsArrayBufferView()) {
|
|
GetDataFrom(aSource.GetAsArrayBufferView(), buffer, length);
|
|
} else {
|
|
MOZ_ASSERT_UNREACHABLE("Unhandled source type!");
|
|
return nullptr;
|
|
}
|
|
|
|
obj->mImpl->InitializeSourceBuffer(buffer, length);
|
|
}
|
|
|
|
return obj.forget();
|
|
}
|
|
|
|
void FontFace::GetFamily(nsACString& aResult) { mImpl->GetFamily(aResult); }
|
|
|
|
void FontFace::SetFamily(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetFamily(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetStyle(nsACString& aResult) { mImpl->GetStyle(aResult); }
|
|
|
|
void FontFace::SetStyle(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetStyle(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetWeight(nsACString& aResult) { mImpl->GetWeight(aResult); }
|
|
|
|
void FontFace::SetWeight(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetWeight(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetStretch(nsACString& aResult) { mImpl->GetStretch(aResult); }
|
|
|
|
void FontFace::SetStretch(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetStretch(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetUnicodeRange(nsACString& aResult) {
|
|
mImpl->GetUnicodeRange(aResult);
|
|
}
|
|
|
|
void FontFace::SetUnicodeRange(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetUnicodeRange(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetVariant(nsACString& aResult) { mImpl->GetVariant(aResult); }
|
|
|
|
void FontFace::SetVariant(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetVariant(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetFeatureSettings(nsACString& aResult) {
|
|
mImpl->GetFeatureSettings(aResult);
|
|
}
|
|
|
|
void FontFace::SetFeatureSettings(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetFeatureSettings(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetVariationSettings(nsACString& aResult) {
|
|
mImpl->GetVariationSettings(aResult);
|
|
}
|
|
|
|
void FontFace::SetVariationSettings(const nsACString& aValue,
|
|
ErrorResult& aRv) {
|
|
mImpl->SetVariationSettings(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetDisplay(nsACString& aResult) { mImpl->GetDisplay(aResult); }
|
|
|
|
void FontFace::SetDisplay(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetDisplay(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetAscentOverride(nsACString& aResult) {
|
|
mImpl->GetAscentOverride(aResult);
|
|
}
|
|
|
|
void FontFace::SetAscentOverride(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetAscentOverride(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetDescentOverride(nsACString& aResult) {
|
|
mImpl->GetDescentOverride(aResult);
|
|
}
|
|
|
|
void FontFace::SetDescentOverride(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetDescentOverride(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetLineGapOverride(nsACString& aResult) {
|
|
mImpl->GetLineGapOverride(aResult);
|
|
}
|
|
|
|
void FontFace::SetLineGapOverride(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetLineGapOverride(aValue, aRv);
|
|
}
|
|
|
|
void FontFace::GetSizeAdjust(nsACString& aResult) {
|
|
mImpl->GetSizeAdjust(aResult);
|
|
}
|
|
|
|
void FontFace::SetSizeAdjust(const nsACString& aValue, ErrorResult& aRv) {
|
|
mImpl->SetSizeAdjust(aValue, aRv);
|
|
}
|
|
|
|
FontFaceLoadStatus FontFace::Status() { return mImpl->Status(); }
|
|
|
|
Promise* FontFace::Load(ErrorResult& aRv) {
|
|
EnsurePromise();
|
|
|
|
if (!mLoaded) {
|
|
aRv.Throw(NS_ERROR_FAILURE);
|
|
return nullptr;
|
|
}
|
|
|
|
mImpl->Load(aRv);
|
|
return mLoaded;
|
|
}
|
|
|
|
Promise* FontFace::GetLoaded(ErrorResult& aRv) {
|
|
EnsurePromise();
|
|
|
|
if (!mLoaded) {
|
|
aRv.Throw(NS_ERROR_FAILURE);
|
|
return nullptr;
|
|
}
|
|
|
|
return mLoaded;
|
|
}
|
|
|
|
void FontFace::MaybeResolve() {
|
|
gfxFontUtils::AssertSafeThreadOrServoFontMetricsLocked();
|
|
|
|
if (!mLoaded) {
|
|
return;
|
|
}
|
|
|
|
if (ServoStyleSet* ss = gfxFontUtils::CurrentServoStyleSet()) {
|
|
// See comments in Gecko_GetFontMetrics.
|
|
ss->AppendTask(PostTraversalTask::ResolveFontFaceLoadedPromise(this));
|
|
return;
|
|
}
|
|
|
|
mLoaded->MaybeResolve(this);
|
|
}
|
|
|
|
void FontFace::MaybeReject(nsresult aResult) {
|
|
gfxFontUtils::AssertSafeThreadOrServoFontMetricsLocked();
|
|
|
|
if (ServoStyleSet* ss = gfxFontUtils::CurrentServoStyleSet()) {
|
|
// See comments in Gecko_GetFontMetrics.
|
|
ss->AppendTask(
|
|
PostTraversalTask::RejectFontFaceLoadedPromise(this, aResult));
|
|
return;
|
|
}
|
|
|
|
if (mLoaded) {
|
|
mLoaded->MaybeReject(aResult);
|
|
} else if (mLoadedRejection == NS_OK) {
|
|
mLoadedRejection = aResult;
|
|
}
|
|
}
|
|
|
|
void FontFace::EnsurePromise() {
|
|
if (mLoaded || !mImpl || !GetOwnerGlobal()) {
|
|
return;
|
|
}
|
|
|
|
ErrorResult rv;
|
|
mLoaded = Promise::Create(GetOwnerGlobal(), rv);
|
|
|
|
if (mImpl->Status() == FontFaceLoadStatus::Loaded) {
|
|
mLoaded->MaybeResolve(this);
|
|
} else if (mLoadedRejection != NS_OK) {
|
|
mLoaded->MaybeReject(mLoadedRejection);
|
|
}
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|