From 7169555811f9898a5cadd8307dcf76b7f4744aea Mon Sep 17 00:00:00 2001 From: Yingchi Long Date: Thu, 21 Sep 2023 18:22:44 +0800 Subject: [PATCH] nixd/Sema: lowering for `let ... in ...` --- nixd/include/nixd/Sema/Lowering.h | 10 ++++- nixd/lib/Sema/Lowering.cpp | 49 ++++++++++++++++++++--- nixd/tools/nixd-lint/test/let-dynamic.nix | 8 ++++ nixd/tools/nixd-lint/test/let-simple.nix | 9 +++++ 4 files changed, 68 insertions(+), 8 deletions(-) create mode 100644 nixd/tools/nixd-lint/test/let-dynamic.nix create mode 100644 nixd/tools/nixd-lint/test/let-simple.nix diff --git a/nixd/include/nixd/Sema/Lowering.h b/nixd/include/nixd/Sema/Lowering.h index f6deeb349..f6d20ff7a 100644 --- a/nixd/include/nixd/Sema/Lowering.h +++ b/nixd/include/nixd/Sema/Lowering.h @@ -18,9 +18,14 @@ class ExprAttrsBuilder { Lowering &LW; nix::ExprAttrs *Result; + RangeIdx Range; + bool Recursive; - RangeIdx Range; + /// let ... in ... + /// It is not allowed to use dynamic binds here, so we want to give diagnostic + /// to each occurrence. + bool IsLet; /// Nested attributes, we create a new builder for them, and collapse the map /// while finishing @@ -34,7 +39,8 @@ class ExprAttrsBuilder { std::map Fields; public: - ExprAttrsBuilder(Lowering &LW, bool Recursive, RangeIdx Range); + ExprAttrsBuilder(Lowering &LW, RangeIdx Range, bool Recursive, bool IsLet); + void addAttr(const syntax::Node *Attr, const syntax::Node *Body, bool Recursive); void addAttribute(const syntax::Attribute &A); diff --git a/nixd/lib/Sema/Lowering.cpp b/nixd/lib/Sema/Lowering.cpp index d4bf9fa84..137b49613 100644 --- a/nixd/lib/Sema/Lowering.cpp +++ b/nixd/lib/Sema/Lowering.cpp @@ -134,8 +134,9 @@ static syntax::Diagnostic mkAttrDupDiag(std::string Name, RangeIdx Range, return Diag; } -ExprAttrsBuilder::ExprAttrsBuilder(Lowering &LW, bool Recursive, RangeIdx Range) - : LW(LW), Recursive(Recursive), Range(Range) { +ExprAttrsBuilder::ExprAttrsBuilder(Lowering &LW, RangeIdx Range, bool Recursive, + bool IsLet) + : LW(LW), Range(Range), Recursive(Recursive), IsLet(IsLet) { Result = LW.Ctx.Pool.record(new nix::ExprAttrs); } @@ -196,6 +197,15 @@ static Diagnostic mkRecDiag(RangeIdx ThisRange, RangeIdx MergingRange, return Diag; } +static Diagnostic mkLetDynamicDiag(RangeIdx Range) { + Diagnostic Diag; + Diag.Kind = Diagnostic::Error; + Diag.Msg = "dynamic attributes are not allowed in let ... in ... expression"; + Diag.Range = Range; + + return Diag; +} + void ExprAttrsBuilder::addAttrSet(const syntax::AttrSet &AS) { if (AS.Recursive != Recursive) { auto Diag = mkRecDiag(Range, AS.Range, Recursive, AS.Recursive); @@ -233,7 +243,7 @@ void ExprAttrsBuilder::addAttr(const syntax::Node *Attr, // because we have chance to merge it latter const auto *BodyAttrSet = dynamic_cast(Body); Nested[Sym] = std::make_unique( - LW, BodyAttrSet->Recursive, BodyAttrSet->Range); + LW, BodyAttrSet->Range, BodyAttrSet->Recursive, IsLet); Nested[Sym]->addBinds(*BodyAttrSet->AttrBinds); } else { nix::ExprAttrs::AttrDef Def(LW.lower(Body), Attr->Range.Begin); @@ -243,6 +253,11 @@ void ExprAttrsBuilder::addAttr(const syntax::Node *Attr, break; } default: { + if (IsLet) { + // reject dynamic attrs in let expression + LW.Diags.emplace_back(mkLetDynamicDiag(Attr->Range)); + return; + } nix::Expr *NameExpr = LW.lower(Attr); nix::Expr *ValueExpr = LW.lower(Body); nix::ExprAttrs::DynamicAttrDef DDef(NameExpr, ValueExpr, Attr->Range.Begin); @@ -281,15 +296,20 @@ void ExprAttrsBuilder::addAttribute(const syntax::Attribute &A) { } else { // create a new builder and select to it. S->Nested[Sym] = std::make_unique( - LW, /*Recursive=*/false, Name->Range); + LW, Name->Range, /*Recursive=*/false, IsLet); S->Fields[Sym] = Name; S = S->Nested[Sym].get(); } break; // case Node::NK_Identifier: } default: { + if (IsLet) { + // reject dynamic attrs in let expression + LW.Diags.emplace_back(mkLetDynamicDiag(Name->Range)); + return; + } DynamicNested[Name] = std::make_unique( - LW, /*Recursive=*/false, Name->Range); + LW, Name->Range, /*Recursive=*/false, IsLet); S = DynamicNested[Name].get(); } } @@ -389,7 +409,8 @@ nix::Expr *Lowering::lower(const syntax::Node *Root) { case Node::NK_AttrSet: { const auto *AttrSet = dynamic_cast(Root); assert(AttrSet->AttrBinds && "null AttrBinds of the AttrSet!"); - ExprAttrsBuilder Builder(*this, AttrSet->Recursive, AttrSet->Range); + ExprAttrsBuilder Builder(*this, AttrSet->Range, AttrSet->Recursive, + /*IsLet=*/false); Builder.addBinds(*AttrSet->AttrBinds); return Builder.finish(); } @@ -399,6 +420,22 @@ nix::Expr *Lowering::lower(const syntax::Node *Root) { Ctx.Pool.record(NixInt); return NixInt; } + case Node::NK_Let: { + const auto *Let = dynamic_cast(Root); + ExprAttrsBuilder Builder(*this, Let->Binds->Range, /*Recursive=*/true, + /*IsLet=*/true); + Builder.addBinds(*Let->Binds); + nix::ExprAttrs *Attrs = Builder.finish(); + nix::Expr *Body = lower(Let->Body); + + // special work for let expression, the expression there is implictly + // recursive, and we should not mark `rec` to the evaluator, because the + // official parser does not do this also. + Attrs->recursive = false; + + auto *Ret = new nix::ExprLet(Attrs, Body); + return Ret; + } } return nullptr; diff --git a/nixd/tools/nixd-lint/test/let-dynamic.nix b/nixd/tools/nixd-lint/test/let-dynamic.nix new file mode 100644 index 000000000..5d4a8dbd0 --- /dev/null +++ b/nixd/tools/nixd-lint/test/let-dynamic.nix @@ -0,0 +1,8 @@ +# RUN: nixd-lint %s | FileCheck %s + +# CHECK: dynamic attributes are not allowed in let ... in ... expression +let + ${1} = 3; + a = 1; +in +1 diff --git a/nixd/tools/nixd-lint/test/let-simple.nix b/nixd/tools/nixd-lint/test/let-simple.nix new file mode 100644 index 000000000..69ac262c9 --- /dev/null +++ b/nixd/tools/nixd-lint/test/let-simple.nix @@ -0,0 +1,9 @@ +# RUN: nixd-lint -dump-nix-ast %s | FileCheck %s + +# CHECK: ExprLet: (let x = 1; y = 2; z = 3; in 1) +let + x = 1; + y = 2; + z = 3; +in +1