forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			507 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			507 lines
		
	
	
	
		
			16 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/ServoStyleSheet.h"
 | |
| 
 | |
| #include "mozilla/css/Rule.h"
 | |
| #include "mozilla/StyleBackendType.h"
 | |
| #include "mozilla/ServoBindings.h"
 | |
| #include "mozilla/ServoCSSRuleList.h"
 | |
| #include "mozilla/ServoImportRule.h"
 | |
| #include "mozilla/ServoMediaList.h"
 | |
| #include "mozilla/ServoStyleSet.h"
 | |
| #include "mozilla/css/GroupRule.h"
 | |
| #include "mozilla/dom/CSSRuleList.h"
 | |
| #include "mozilla/dom/MediaList.h"
 | |
| #include "nsIStyleSheetLinkingElement.h"
 | |
| #include "Loader.h"
 | |
| 
 | |
| 
 | |
| #include "mozAutoDocUpdate.h"
 | |
| #include "nsIDOMCSSStyleSheet.h"
 | |
| 
 | |
| using namespace mozilla::dom;
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| // -------------------------------
 | |
| // CSS Style Sheet Inner Data Container
 | |
| //
 | |
| 
 | |
| ServoStyleSheetInner::ServoStyleSheetInner(CORSMode aCORSMode,
 | |
|                                            ReferrerPolicy aReferrerPolicy,
 | |
|                                            const SRIMetadata& aIntegrity,
 | |
|                                            css::SheetParsingMode aParsingMode)
 | |
|   : StyleSheetInfo(aCORSMode, aReferrerPolicy, aIntegrity)
 | |
| {
 | |
|   mContents = Servo_StyleSheet_Empty(aParsingMode).Consume();
 | |
|   mURLData = URLExtraData::Dummy();
 | |
|   MOZ_COUNT_CTOR(ServoStyleSheetInner);
 | |
| }
 | |
| 
 | |
| ServoStyleSheetInner::ServoStyleSheetInner(ServoStyleSheetInner& aCopy,
 | |
|                                            ServoStyleSheet* aPrimarySheet)
 | |
|   : StyleSheetInfo(aCopy, aPrimarySheet)
 | |
|   , mURLData(aCopy.mURLData)
 | |
| {
 | |
|   MOZ_COUNT_CTOR(ServoStyleSheetInner);
 | |
| 
 | |
|   // Actually clone aCopy's mContents and use that as ours.
 | |
|   mContents = Servo_StyleSheet_Clone(
 | |
|     aCopy.mContents.get(), aPrimarySheet).Consume();
 | |
| 
 | |
|   // Our child list is fixed up by our parent.
 | |
| }
 | |
| 
 | |
| void
 | |
| ServoStyleSheet::BuildChildListAfterInnerClone()
 | |
| {
 | |
|   MOZ_ASSERT(Inner()->mSheets.Length() == 1, "Should've just cloned");
 | |
|   MOZ_ASSERT(Inner()->mSheets[0] == this);
 | |
|   MOZ_ASSERT(!Inner()->mFirstChild);
 | |
| 
 | |
|   auto* contents = Inner()->mContents.get();
 | |
|   RefPtr<ServoCssRules> rules =
 | |
|     Servo_StyleSheet_GetRules(contents).Consume();
 | |
| 
 | |
|   uint32_t index = 0;
 | |
|   while (true) {
 | |
|     uint32_t line, column; // Actually unused.
 | |
|     RefPtr<RawServoImportRule> import =
 | |
|       Servo_CssRules_GetImportRuleAt(rules, index, &line, &column).Consume();
 | |
|     if (!import) {
 | |
|       // Note that only @charset rules come before @import rules, and @charset
 | |
|       // rules are parsed but skipped, so we can stop iterating as soon as we
 | |
|       // find something that isn't an @import rule.
 | |
|       break;
 | |
|     }
 | |
|     auto* sheet =
 | |
|       const_cast<ServoStyleSheet*>(Servo_ImportRule_GetSheet(import));
 | |
|     MOZ_ASSERT(sheet);
 | |
|     PrependStyleSheetSilently(sheet);
 | |
|     index++;
 | |
|   }
 | |
| }
 | |
| 
 | |
| already_AddRefed<ServoStyleSheet>
 | |
| ServoStyleSheet::CreateEmptyChildSheet(
 | |
|     already_AddRefed<dom::MediaList> aMediaList) const
 | |
| {
 | |
|   RefPtr<ServoStyleSheet> child =
 | |
|     new ServoStyleSheet(
 | |
|         ParsingMode(),
 | |
|         CORSMode::CORS_NONE,
 | |
|         GetReferrerPolicy(),
 | |
|         SRIMetadata());
 | |
| 
 | |
|   child->mMedia = aMediaList;
 | |
|   return child.forget();
 | |
| }
 | |
| 
 | |
| 
 | |
| ServoStyleSheetInner::~ServoStyleSheetInner()
 | |
| {
 | |
|   MOZ_COUNT_DTOR(ServoStyleSheetInner);
 | |
| }
 | |
| 
 | |
| StyleSheetInfo*
 | |
| ServoStyleSheetInner::CloneFor(StyleSheet* aPrimarySheet)
 | |
| {
 | |
|   return new ServoStyleSheetInner(*this,
 | |
|                                   static_cast<ServoStyleSheet*>(aPrimarySheet));
 | |
| }
 | |
| 
 | |
| MOZ_DEFINE_MALLOC_SIZE_OF(ServoStyleSheetMallocSizeOf)
 | |
| MOZ_DEFINE_MALLOC_ENCLOSING_SIZE_OF(ServoStyleSheetMallocEnclosingSizeOf)
 | |
| 
 | |
| size_t
 | |
| ServoStyleSheetInner::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
 | |
| {
 | |
|   size_t n = aMallocSizeOf(this);
 | |
|   n += Servo_StyleSheet_SizeOfIncludingThis(
 | |
|       ServoStyleSheetMallocSizeOf,
 | |
|       ServoStyleSheetMallocEnclosingSizeOf,
 | |
|       mContents);
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| ServoStyleSheet::ServoStyleSheet(css::SheetParsingMode aParsingMode,
 | |
|                                  CORSMode aCORSMode,
 | |
|                                  net::ReferrerPolicy aReferrerPolicy,
 | |
|                                  const dom::SRIMetadata& aIntegrity)
 | |
|   : StyleSheet(StyleBackendType::Servo, aParsingMode)
 | |
| {
 | |
|   mInner = new ServoStyleSheetInner(
 | |
|     aCORSMode, aReferrerPolicy, aIntegrity, aParsingMode);
 | |
|   mInner->AddSheet(this);
 | |
| }
 | |
| 
 | |
| ServoStyleSheet::ServoStyleSheet(const ServoStyleSheet& aCopy,
 | |
|                                  ServoStyleSheet* aParentToUse,
 | |
|                                  dom::CSSImportRule* aOwnerRuleToUse,
 | |
|                                  nsIDocument* aDocumentToUse,
 | |
|                                  nsINode* aOwningNodeToUse)
 | |
|   : StyleSheet(aCopy,
 | |
|                aParentToUse,
 | |
|                aOwnerRuleToUse,
 | |
|                aDocumentToUse,
 | |
|                aOwningNodeToUse)
 | |
| {
 | |
|   if (mDirty) { // CSSOM's been there, force full copy now
 | |
|     NS_ASSERTION(mInner->mComplete,
 | |
|                  "Why have rules been accessed on an incomplete sheet?");
 | |
|     // FIXME: handle failure?
 | |
|     //
 | |
|     // NOTE: It's important to call this from the subclass, since this could
 | |
|     // access uninitialized members otherwise.
 | |
|     EnsureUniqueInner();
 | |
|   }
 | |
| }
 | |
| 
 | |
| ServoStyleSheet::~ServoStyleSheet()
 | |
| {
 | |
| }
 | |
| 
 | |
| void
 | |
| ServoStyleSheet::LastRelease()
 | |
| {
 | |
|   DropRuleList();
 | |
| }
 | |
| 
 | |
| // QueryInterface implementation for ServoStyleSheet
 | |
| NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(ServoStyleSheet)
 | |
|   NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDOMCSSStyleSheet)
 | |
|   if (aIID.Equals(NS_GET_IID(ServoStyleSheet)))
 | |
|     foundInterface = reinterpret_cast<nsISupports*>(this);
 | |
|   else
 | |
| NS_INTERFACE_MAP_END_INHERITING(StyleSheet)
 | |
| 
 | |
| NS_IMPL_ADDREF_INHERITED(ServoStyleSheet, StyleSheet)
 | |
| NS_IMPL_RELEASE_INHERITED(ServoStyleSheet, StyleSheet)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_CLASS(ServoStyleSheet)
 | |
| 
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(ServoStyleSheet)
 | |
|   tmp->DropRuleList();
 | |
| NS_IMPL_CYCLE_COLLECTION_UNLINK_END_INHERITED(StyleSheet)
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(ServoStyleSheet, StyleSheet)
 | |
|   NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRuleList)
 | |
| NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | |
| 
 | |
| bool
 | |
| ServoStyleSheet::HasRules() const
 | |
| {
 | |
|   return Servo_StyleSheet_HasRules(Inner()->mContents);
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| ServoStyleSheet::ParseSheet(css::Loader* aLoader,
 | |
|                             Span<const uint8_t> aInput,
 | |
|                             nsIURI* aSheetURI,
 | |
|                             nsIURI* aBaseURI,
 | |
|                             nsIPrincipal* aSheetPrincipal,
 | |
|                             uint32_t aLineNumber,
 | |
|                             nsCompatibility aCompatMode,
 | |
|                             css::LoaderReusableStyleSheets* aReusableSheets)
 | |
| {
 | |
|   MOZ_ASSERT(!mMedia || mMedia->IsServo());
 | |
|   RefPtr<URLExtraData> extraData =
 | |
|     new URLExtraData(aBaseURI, aSheetURI, aSheetPrincipal);
 | |
| 
 | |
|   Inner()->mContents = Servo_StyleSheet_FromUTF8Bytes(aLoader,
 | |
|                                                       this,
 | |
|                                                       aInput.Elements(),
 | |
|                                                       aInput.Length(),
 | |
|                                                       mParsingMode,
 | |
|                                                       extraData,
 | |
|                                                       aLineNumber,
 | |
|                                                       aCompatMode,
 | |
|                                                       aReusableSheets)
 | |
|                          .Consume();
 | |
| 
 | |
|   nsString sourceMapURL;
 | |
|   Servo_StyleSheet_GetSourceMapURL(Inner()->mContents, &sourceMapURL);
 | |
|   SetSourceMapURLFromComment(sourceMapURL);
 | |
| 
 | |
|   nsString sourceURL;
 | |
|   Servo_StyleSheet_GetSourceURL(Inner()->mContents, &sourceURL);
 | |
|   SetSourceURL(sourceURL);
 | |
| 
 | |
|   Inner()->mURLData = extraData.forget();
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| ServoStyleSheet::ReparseSheet(const nsAString& aInput)
 | |
| {
 | |
|   if (!mInner->mComplete) {
 | |
|     return NS_ERROR_DOM_INVALID_ACCESS_ERR;
 | |
|   }
 | |
| 
 | |
|   // Hold strong ref to the CSSLoader in case the document update
 | |
|   // kills the document
 | |
|   RefPtr<css::Loader> loader;
 | |
|   if (mDocument) {
 | |
|     loader = mDocument->CSSLoader();
 | |
|     NS_ASSERTION(loader, "Document with no CSS loader!");
 | |
|   } else {
 | |
|     loader = new css::Loader(StyleBackendType::Servo, nullptr);
 | |
|   }
 | |
| 
 | |
|   mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
 | |
| 
 | |
|   WillDirty();
 | |
| 
 | |
|   // cache child sheets to reuse
 | |
|   css::LoaderReusableStyleSheets reusableSheets;
 | |
|   for (StyleSheet* child = GetFirstChild(); child; child = child->mNext) {
 | |
|     if (child->GetOriginalURI()) {
 | |
|       reusableSheets.AddReusableSheet(child);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // clean up child sheets list
 | |
|   for (StyleSheet* child = GetFirstChild(); child; ) {
 | |
|     StyleSheet* next = child->mNext;
 | |
|     child->mParent = nullptr;
 | |
|     child->SetAssociatedDocument(nullptr, NotOwnedByDocument);
 | |
|     child->mNext = nullptr;
 | |
|     child = next;
 | |
|   }
 | |
|   Inner()->mFirstChild = nullptr;
 | |
| 
 | |
|   uint32_t lineNumber = 1;
 | |
|   if (mOwningNode) {
 | |
|     nsCOMPtr<nsIStyleSheetLinkingElement> link = do_QueryInterface(mOwningNode);
 | |
|     if (link) {
 | |
|       lineNumber = link->GetLineNumber();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Notify mDocument that all our rules are removed.
 | |
|   if (mDocument) {
 | |
|     // Get the rule list.
 | |
|     ServoCSSRuleList* ruleList = GetCssRulesInternal();
 | |
|     MOZ_ASSERT(ruleList);
 | |
| 
 | |
|     uint32_t ruleCount = ruleList->Length();
 | |
|     for (uint32_t i = 0; i < ruleCount; ++i) {
 | |
|       css::Rule* rule = ruleList->GetRule(i);
 | |
|       MOZ_ASSERT(rule);
 | |
|       if (rule->GetType() == css::Rule::IMPORT_RULE &&
 | |
|           RuleHasPendingChildSheet(rule)) {
 | |
|         continue; // notify when loaded (see StyleSheetLoaded)
 | |
|       }
 | |
|       mDocument->StyleRuleRemoved(this, rule);
 | |
| 
 | |
|       // Document observers could possibly detach document from this sheet.
 | |
|       if (!mDocument) {
 | |
|         // If detached, don't process any more rules.
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   DropRuleList();
 | |
| 
 | |
|   nsresult rv = ParseSheet(loader,
 | |
|                            NS_ConvertUTF16toUTF8(aInput),
 | |
|                            mInner->mSheetURI,
 | |
|                            mInner->mBaseURI,
 | |
|                            mInner->mPrincipal,
 | |
|                            lineNumber,
 | |
|                            eCompatibility_FullStandards,
 | |
|                            &reusableSheets);
 | |
|   DidDirty();
 | |
|   NS_ENSURE_SUCCESS(rv, rv);
 | |
| 
 | |
|   // Notify mDocument that all our new rules are added.
 | |
|   if (mDocument) {
 | |
|     // Get the rule list (which will need to be regenerated after ParseSheet).
 | |
|     ServoCSSRuleList* ruleList = GetCssRulesInternal();
 | |
|     MOZ_ASSERT(ruleList);
 | |
| 
 | |
|     uint32_t ruleCount = ruleList->Length();
 | |
|     for (uint32_t i = 0; i < ruleCount; ++i) {
 | |
|       css::Rule* rule = ruleList->GetRule(i);
 | |
|       MOZ_ASSERT(rule);
 | |
|       if (rule->GetType() == css::Rule::IMPORT_RULE &&
 | |
|           RuleHasPendingChildSheet(rule)) {
 | |
|         continue; // notify when loaded (see StyleSheetLoaded)
 | |
|       }
 | |
| 
 | |
|       mDocument->StyleRuleAdded(this, rule);
 | |
| 
 | |
|       // Document observers could possibly detach document from this sheet.
 | |
|       if (!mDocument) {
 | |
|         // If detached, don't process any more rules.
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // FIXME(emilio): This is kind-of a hack for bug 1420713. As you may notice,
 | |
|   // there's nothing that triggers a style flush or anything similar (neither
 | |
|   // here or in the relevant Gecko path inside DidDirty).
 | |
|   //
 | |
|   // The tl;dr is: if we want to make sure scripted changes to sheets not
 | |
|   // associated with any document get properly reflected, we need to rejigger a
 | |
|   // fair amount of stuff. I'm probably doing that work as part of the shadow
 | |
|   // DOM stuff.
 | |
|   for (StyleSetHandle handle : mStyleSets) {
 | |
|     handle->AsServo()->RecordStyleSheetChange(
 | |
|       this, StyleSheet::ChangeType::ReparsedFromInspector);
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| // nsICSSLoaderObserver implementation
 | |
| NS_IMETHODIMP
 | |
| ServoStyleSheet::StyleSheetLoaded(StyleSheet* aSheet,
 | |
|                                   bool aWasAlternate,
 | |
|                                   nsresult aStatus)
 | |
| {
 | |
|   MOZ_ASSERT(aSheet->IsServo(),
 | |
|              "why we were called back with a CSSStyleSheet?");
 | |
| 
 | |
|   ServoStyleSheet* sheet = aSheet->AsServo();
 | |
|   if (sheet->GetParentSheet() == nullptr) {
 | |
|     return NS_OK; // ignore if sheet has been detached already
 | |
|   }
 | |
|   NS_ASSERTION(this == sheet->GetParentSheet(),
 | |
|                "We are being notified of a sheet load for a sheet that is not our child!");
 | |
| 
 | |
|   if (mDocument && NS_SUCCEEDED(aStatus)) {
 | |
|     mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
 | |
|     mDocument->StyleRuleAdded(this, sheet->GetOwnerRule());
 | |
|   }
 | |
| 
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void
 | |
| ServoStyleSheet::DropRuleList()
 | |
| {
 | |
|   if (mRuleList) {
 | |
|     mRuleList->DropReference();
 | |
|     mRuleList = nullptr;
 | |
|   }
 | |
| }
 | |
| 
 | |
| already_AddRefed<StyleSheet>
 | |
| ServoStyleSheet::Clone(StyleSheet* aCloneParent,
 | |
|                        dom::CSSImportRule* aCloneOwnerRule,
 | |
|                        nsIDocument* aCloneDocument,
 | |
|                        nsINode* aCloneOwningNode) const
 | |
| {
 | |
|   RefPtr<StyleSheet> clone = new ServoStyleSheet(*this,
 | |
|     static_cast<ServoStyleSheet*>(aCloneParent),
 | |
|     aCloneOwnerRule,
 | |
|     aCloneDocument,
 | |
|     aCloneOwningNode);
 | |
|   return clone.forget();
 | |
| }
 | |
| 
 | |
| ServoCSSRuleList*
 | |
| ServoStyleSheet::GetCssRulesInternal()
 | |
| {
 | |
|   if (!mRuleList) {
 | |
|     EnsureUniqueInner();
 | |
| 
 | |
|     RefPtr<ServoCssRules> rawRules =
 | |
|       Servo_StyleSheet_GetRules(Inner()->mContents).Consume();
 | |
|     MOZ_ASSERT(rawRules);
 | |
|     mRuleList = new ServoCSSRuleList(rawRules.forget(), this);
 | |
|   }
 | |
|   return mRuleList;
 | |
| }
 | |
| 
 | |
| uint32_t
 | |
| ServoStyleSheet::InsertRuleInternal(const nsAString& aRule,
 | |
|                                     uint32_t aIndex, ErrorResult& aRv)
 | |
| {
 | |
|   // Ensure mRuleList is constructed.
 | |
|   GetCssRulesInternal();
 | |
| 
 | |
|   mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
 | |
|   aRv = mRuleList->InsertRule(aRule, aIndex);
 | |
|   if (aRv.Failed()) {
 | |
|     return 0;
 | |
|   }
 | |
|   if (mDocument) {
 | |
|     if (mRuleList->GetDOMCSSRuleType(aIndex) != nsIDOMCSSRule::IMPORT_RULE ||
 | |
|         !RuleHasPendingChildSheet(mRuleList->GetRule(aIndex))) {
 | |
|       // XXX We may not want to get the rule when stylesheet change event
 | |
|       // is not enabled.
 | |
|       mDocument->StyleRuleAdded(this, mRuleList->GetRule(aIndex));
 | |
|     }
 | |
|   }
 | |
|   return aIndex;
 | |
| }
 | |
| 
 | |
| void
 | |
| ServoStyleSheet::DeleteRuleInternal(uint32_t aIndex, ErrorResult& aRv)
 | |
| {
 | |
|   // Ensure mRuleList is constructed.
 | |
|   GetCssRulesInternal();
 | |
|   if (aIndex >= mRuleList->Length()) {
 | |
|     aRv.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
 | |
|   // Hold a strong ref to the rule so it doesn't die when we remove it
 | |
|   // from the list. XXX We may not want to hold it if stylesheet change
 | |
|   // event is not enabled.
 | |
|   RefPtr<css::Rule> rule = mRuleList->GetRule(aIndex);
 | |
|   aRv = mRuleList->DeleteRule(aIndex);
 | |
|   MOZ_ASSERT(!aRv.ErrorCodeIs(NS_ERROR_DOM_INDEX_SIZE_ERR),
 | |
|              "IndexSizeError should have been handled earlier");
 | |
|   if (!aRv.Failed() && mDocument) {
 | |
|     mDocument->StyleRuleRemoved(this, rule);
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| ServoStyleSheet::InsertRuleIntoGroupInternal(const nsAString& aRule,
 | |
|                                              css::GroupRule* aGroup,
 | |
|                                              uint32_t aIndex)
 | |
| {
 | |
|   auto rules = static_cast<ServoCSSRuleList*>(aGroup->CssRules());
 | |
|   MOZ_ASSERT(rules->GetParentRule() == aGroup);
 | |
|   return rules->InsertRule(aRule, aIndex);
 | |
| }
 | |
| 
 | |
| size_t
 | |
| ServoStyleSheet::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
 | |
| {
 | |
|   size_t n = StyleSheet::SizeOfIncludingThis(aMallocSizeOf);
 | |
|   const ServoStyleSheet* s = this;
 | |
|   while (s) {
 | |
|     // See the comment in CSSStyleSheet::SizeOfIncludingThis() for an
 | |
|     // explanation of this.
 | |
|     if (s->Inner()->mSheets.LastElement() == s) {
 | |
|       n += s->Inner()->SizeOfIncludingThis(aMallocSizeOf);
 | |
|     }
 | |
| 
 | |
|     // Measurement of the following members may be added later if DMD finds it
 | |
|     // is worthwhile:
 | |
|     // - s->mRuleList
 | |
| 
 | |
|     s = s->mNext ? s->mNext->AsServo() : nullptr;
 | |
|   }
 | |
|   return n;
 | |
| }
 | |
| 
 | |
| OriginFlags
 | |
| ServoStyleSheet::GetOrigin()
 | |
| {
 | |
|   return static_cast<OriginFlags>(
 | |
|     Servo_StyleSheet_GetOrigin(Inner()->mContents));
 | |
| }
 | |
| 
 | |
| } // namespace mozilla
 | 
