/* -*- 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/ServoStyleRuleMap.h" #include "mozilla/css/GroupRule.h" #include "mozilla/dom/CSSImportRule.h" #include "mozilla/dom/CSSRuleBinding.h" #include "mozilla/dom/CSSStyleRule.h" #include "mozilla/dom/Document.h" #include "mozilla/dom/Element.h" #include "mozilla/dom/ShadowRoot.h" #include "mozilla/IntegerRange.h" #include "mozilla/ServoStyleSet.h" #include "mozilla/StyleSheetInlines.h" #include "nsStyleSheetService.h" using namespace mozilla::dom; namespace mozilla { void ServoStyleRuleMap::EnsureTable(ServoStyleSet& aStyleSet) { if (!IsEmpty()) { return; } aStyleSet.EnumerateStyleSheets( [&](StyleSheet& aSheet) { FillTableFromStyleSheet(aSheet); }); } void ServoStyleRuleMap::EnsureTable(ShadowRoot& aShadowRoot) { if (!IsEmpty()) { return; } for (auto index : IntegerRange(aShadowRoot.SheetCount())) { FillTableFromStyleSheet(*aShadowRoot.SheetAt(index)); } for (auto& sheet : aShadowRoot.AdoptedStyleSheets()) { FillTableFromStyleSheet(*sheet); } } void ServoStyleRuleMap::SheetAdded(StyleSheet& aStyleSheet) { if (!IsEmpty()) { FillTableFromStyleSheet(aStyleSheet); } } void ServoStyleRuleMap::SheetCloned(StyleSheet& aStyleSheet) { // Invalidate all data inside. We could probably track down all the individual // rules that changed etc, but it doesn't seem worth it. // // TODO: We can't do this until GetCssRulesInternal stops cloning. // mTable.Clear(); } void ServoStyleRuleMap::SheetRemoved(StyleSheet& aStyleSheet) { // Invalidate all data inside. This isn't strictly necessary since // we should always get update from document before new queries come. // But it is probably still safer if we try to avoid having invalid // pointers inside. Also if the document keep adding and removing // stylesheets, this would also prevent us from infinitely growing // memory usage. mTable.Clear(); } void ServoStyleRuleMap::RuleAdded(StyleSheet& aStyleSheet, css::Rule& aStyleRule) { if (!IsEmpty()) { FillTableFromRule(aStyleRule); } } void ServoStyleRuleMap::RuleRemoved(StyleSheet& aStyleSheet, css::Rule& aStyleRule) { if (IsEmpty()) { return; } switch (aStyleRule.Type()) { case StyleCssRuleType::Style: { auto& rule = static_cast(aStyleRule); mTable.Remove(rule.Raw()); break; } case StyleCssRuleType::Import: case StyleCssRuleType::Media: case StyleCssRuleType::Supports: case StyleCssRuleType::LayerBlock: case StyleCssRuleType::Container: case StyleCssRuleType::Document: { // See the comment in StyleSheetRemoved. mTable.Clear(); break; } case StyleCssRuleType::LayerStatement: case StyleCssRuleType::FontFace: case StyleCssRuleType::Page: case StyleCssRuleType::Keyframes: case StyleCssRuleType::Keyframe: case StyleCssRuleType::Namespace: case StyleCssRuleType::CounterStyle: case StyleCssRuleType::FontFeatureValues: case StyleCssRuleType::FontPaletteValues: case StyleCssRuleType::Viewport: break; } } size_t ServoStyleRuleMap::SizeOfIncludingThis( MallocSizeOf aMallocSizeOf) const { size_t n = aMallocSizeOf(this); n += mTable.ShallowSizeOfExcludingThis(aMallocSizeOf); return n; } void ServoStyleRuleMap::FillTableFromRule(css::Rule& aRule) { switch (aRule.Type()) { case StyleCssRuleType::Style: { auto& rule = static_cast(aRule); mTable.InsertOrUpdate(rule.Raw(), &rule); break; } case StyleCssRuleType::LayerBlock: case StyleCssRuleType::Media: case StyleCssRuleType::Supports: case StyleCssRuleType::Container: case StyleCssRuleType::Document: { auto& rule = static_cast(aRule); if (ServoCSSRuleList* ruleList = rule.GetCssRules()) { FillTableFromRuleList(*ruleList); } break; } case StyleCssRuleType::Import: { auto& rule = static_cast(aRule); MOZ_ASSERT(aRule.GetStyleSheet()); FillTableFromStyleSheet(*rule.GetStyleSheet()); break; } case StyleCssRuleType::LayerStatement: case StyleCssRuleType::FontFace: case StyleCssRuleType::Page: case StyleCssRuleType::Keyframes: case StyleCssRuleType::Keyframe: case StyleCssRuleType::Namespace: case StyleCssRuleType::CounterStyle: case StyleCssRuleType::FontFeatureValues: case StyleCssRuleType::FontPaletteValues: case StyleCssRuleType::Viewport: break; } } void ServoStyleRuleMap::FillTableFromRuleList(ServoCSSRuleList& aRuleList) { for (uint32_t i : IntegerRange(aRuleList.Length())) { FillTableFromRule(*aRuleList.GetRule(i)); } } void ServoStyleRuleMap::FillTableFromStyleSheet(StyleSheet& aSheet) { if (aSheet.IsComplete()) { FillTableFromRuleList(*aSheet.GetCssRulesInternal()); } } } // namespace mozilla