From 1ec3876afc071a79bb2d104b0b79306392e20bc4 Mon Sep 17 00:00:00 2001 From: Marcel Greter Date: Fri, 18 Oct 2019 07:22:52 +0200 Subject: [PATCH 1/2] Fix extend issue being stuck in endless loop Fixes https://github.com/sass/libsass/issues/3006 --- src/eval_selectors.cpp | 6 +++--- src/expand.cpp | 29 +++++++++++++++++++++++------ src/expand.hpp | 3 +++ src/extender.hpp | 4 ++-- 4 files changed, 31 insertions(+), 11 deletions(-) diff --git a/src/eval_selectors.cpp b/src/eval_selectors.cpp index cc7d3409c..db797e5bd 100644 --- a/src/eval_selectors.cpp +++ b/src/eval_selectors.cpp @@ -46,10 +46,10 @@ namespace Sass { SelectorList* Eval::operator()(ComplexSelector* s) { bool implicit_parent = !exp.old_at_root_without_rule; - if (is_in_selector_schema) exp.pushToSelectorStack({}); + if (is_in_selector_schema) exp.pushNullSelector(); SelectorListObj other = s->resolve_parent_refs( - exp.getSelectorStack(), traces, implicit_parent); - if (is_in_selector_schema) exp.popFromSelectorStack(); + exp.getOriginalStack(), traces, implicit_parent); + if (is_in_selector_schema) exp.popNullSelector(); for (size_t i = 0; i < other->length(); i++) { ComplexSelectorObj sel = other->at(i); diff --git a/src/expand.cpp b/src/expand.cpp index 1b9bf9158..0380e1cec 100644 --- a/src/expand.cpp +++ b/src/expand.cpp @@ -62,6 +62,23 @@ namespace Sass { return 0; } + void Expand::pushNullSelector() + { + pushToSelectorStack({}); + pushToOriginalStack({}); + } + + void Expand::popNullSelector() + { + popFromOriginalStack(); + popFromSelectorStack(); + } + + SelectorStack Expand::getOriginalStack() + { + return originalStack; + } + SelectorStack Expand::getSelectorStack() { return selector_stack; @@ -153,15 +170,15 @@ namespace Sass { Block* bb = operator()(r->block()); Keyframe_Rule_Obj k = SASS_MEMORY_NEW(Keyframe_Rule, r->pstate(), bb); if (r->schema()) { - pushToSelectorStack({}); + pushNullSelector(); k->name(eval(r->schema())); - popFromSelectorStack(); + popNullSelector(); } else if (r->selector()) { if (SelectorListObj s = r->selector()) { - pushToSelectorStack({}); + pushNullSelector(); k->name(eval(s)); - popFromSelectorStack(); + popNullSelector(); } } @@ -288,10 +305,10 @@ namespace Sass { Block* ab = a->block(); SelectorList* as = a->selector(); Expression* av = a->value(); - pushToSelectorStack({}); + pushNullSelector(); if (av) av = av->perform(&eval); if (as) as = eval(as); - popFromSelectorStack(); + popNullSelector(); Block* bb = ab ? operator()(ab) : NULL; Directive* aa = SASS_MEMORY_NEW(Directive, a->pstate(), diff --git a/src/expand.hpp b/src/expand.hpp index db5852ce4..90edbf4ca 100644 --- a/src/expand.hpp +++ b/src/expand.hpp @@ -22,7 +22,10 @@ namespace Sass { SelectorListObj& selector(); SelectorListObj& original(); SelectorListObj popFromSelectorStack(); + SelectorStack getOriginalStack(); SelectorStack getSelectorStack(); + void pushNullSelector(); + void popNullSelector(); void pushToSelectorStack(SelectorListObj selector); SelectorListObj popFromOriginalStack(); diff --git a/src/extender.hpp b/src/extender.hpp index 4b00cc3a0..5f161786a 100644 --- a/src/extender.hpp +++ b/src/extender.hpp @@ -126,8 +126,8 @@ namespace Sass { std::unordered_map< SimpleSelectorObj, size_t, - ObjHash, - ObjEquality + ObjPtrHash, + ObjPtrEquality > sourceSpecificity; // ########################################################################## From fdf63e4c93d40c1c4da43fdf13010d590dc133e0 Mon Sep 17 00:00:00 2001 From: Marcel Greter Date: Sun, 20 Oct 2019 04:45:29 +0200 Subject: [PATCH 2/2] Fix trim loop condition and outer continue --- src/extender.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/extender.cpp b/src/extender.cpp index 2ae54df98..b0add3857 100644 --- a/src/extender.cpp +++ b/src/extender.cpp @@ -1070,10 +1070,10 @@ namespace Sass { // the result so that, if two selectors are identical, we keep the first one. std::vector result; size_t numOriginals = 0; - // Use label to quit outer loop - redo: + size_t i = selectors.size(); + outer: // Use label to continue loop + while (--i != std::string::npos) { - for (size_t i = selectors.size() - 1; i != std::string::npos; i--) { const ComplexSelectorObj& complex1 = selectors[i]; // Check if selector in known in existing "originals" // For custom behavior dart-sass had `isOriginal(complex1)` @@ -1083,7 +1083,7 @@ namespace Sass { for (size_t j = 0; j < numOriginals; j++) { if (ObjEqualityFn(result[j], complex1)) { rotateSlice(result, 0, j + 1); - goto redo; + goto outer; } } result.insert(result.begin(), complex1);