Skip to content

Commit

Permalink
OS#14115684: Function-in-block semantics don't work for defer parse
Browse files Browse the repository at this point in the history
In below code repro, we do not create var decl binding for `foo` inside the `nest` function in defer-parse mode. Due to this, we leave the reference to `foo` in `nest` on the pid ref stack and bind it to the formal `foo` parameter from the `test` function. This leads us to think formal `foo` is captured and try to put it into a scope slot. However, formal `foo` is not actually captured so no scope slot is allocated. When we actually generate bytecode for `test`, we hit a fail fast.

```js
function test(foo) {
    function nest() {
        {
            function foo() {
                console.log('pass');
            }
        }
        foo();
    }
    nest();
}
test(()=>console.log('fail'));
```

Fix is to create the var binding for `foo` at the function declaration for `foo` even in defer-parse mode.

Fixes:
https://microsoft.visualstudio.com/web/wi.aspx?id=14115684
  • Loading branch information
boingoing committed Feb 13, 2018
1 parent ee5ffb3 commit c3d4e4d
Show file tree
Hide file tree
Showing 4 changed files with 43 additions and 13 deletions.
28 changes: 17 additions & 11 deletions lib/Parser/Parse.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1632,7 +1632,7 @@ ParseNodePtr Parser::ParseBlock(ParseNodePtr pnodeLabel, LabelId* pLabelId)
&& outerBlockInfo->pnodeBlock->sxBlock.scope != nullptr
&& outerBlockInfo->pnodeBlock->sxBlock.scope->GetScopeType() == ScopeType_CatchParamPattern)
{
// If we are parsing the catch block then destructured params can have let declrations. Let's add them to the new block.
// If we are parsing the catch block then destructured params can have let declarations. Let's add them to the new block.
for (ParseNodePtr pnode = m_currentBlockInfo->pBlockInfoOuter->pnodeBlock->sxBlock.pnodeLexVars; pnode; pnode = pnode->sxVar.pnodeNext)
{
PidRefStack* ref = PushPidRef(pnode->sxVar.sym->GetPid());
Expand All @@ -1659,7 +1659,6 @@ ParseNodePtr Parser::ParseBlock(ParseNodePtr pnodeLabel, LabelId* pLabelId)

ChkCurTok(tkRCurly, ERRnoRcurly);


return pnodeBlock;
}

Expand Down Expand Up @@ -5130,8 +5129,9 @@ ParseNodePtr Parser::ParseFncDecl(ushort flags, LPCOLESTR pNameHint, const bool
pnodeFnc->sxFnc.SetIsClassConstructor((flags & fFncClassConstructor) != 0);
pnodeFnc->sxFnc.SetIsBaseClassConstructor((flags & fFncBaseClassConstructor) != 0);

IdentPtr pFncNamePid = nullptr;
bool needScanRCurly = true;
bool result = ParseFncDeclHelper<buildAST>(pnodeFnc, pNameHint, flags, &funcHasName, fUnaryOrParen, noStmtContext, &needScanRCurly, fModule);
bool result = ParseFncDeclHelper<buildAST>(pnodeFnc, pNameHint, flags, &funcHasName, fUnaryOrParen, noStmtContext, &needScanRCurly, fModule, &pFncNamePid);
if (!result)
{
Assert(!pnodeFncBlockScope);
Expand Down Expand Up @@ -5214,9 +5214,10 @@ ParseNodePtr Parser::ParseFncDecl(ushort flags, LPCOLESTR pNameHint, const bool

m_scopeCountNoAst = scopeCountNoAstSave;

if (buildAST && fDeclaration && !IsStrictMode())
if (fDeclaration && !IsStrictMode())
{
if (pnodeFnc->sxFnc.pnodeName != nullptr && pnodeFnc->sxFnc.pnodeName->nop == knopVarDecl &&
if (pFncNamePid != nullptr &&
GetCurrentBlock() &&
GetCurrentBlock()->sxBlock.blockType == PnodeBlockType::Regular)
{
// Add a function-scoped VarDecl with the same name as the function for
Expand All @@ -5225,9 +5226,9 @@ ParseNodePtr Parser::ParseFncDecl(ushort flags, LPCOLESTR pNameHint, const bool
// level and we accomplish this by having each block scoped function
// declaration assign to both the block scoped "let" binding, as well
// as the function scoped "var" binding.
ParseNodePtr vardecl = CreateVarDeclNode(pnodeFnc->sxFnc.pnodeName->sxVar.pid, STVariable, false, nullptr, false);
ParseNodePtr vardecl = CreateVarDeclNode(pFncNamePid, STVariable, false, nullptr, false);
vardecl->sxVar.isBlockScopeFncDeclVar = true;
if (vardecl->sxVar.sym->GetIsFormal())
if (GetCurrentFunctionNode() && vardecl->sxVar.sym->GetIsFormal())
{
GetCurrentFunctionNode()->sxFnc.SetHasAnyWriteToFormals(true);
}
Expand Down Expand Up @@ -5310,7 +5311,7 @@ void Parser::AppendFunctionToScopeList(bool fDeclaration, ParseNodePtr pnodeFnc)
Parse a function definition.
***************************************************************************/
template<bool buildAST>
bool Parser::ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, ushort flags, bool *pHasName, bool fUnaryOrParen, bool noStmtContext, bool *pNeedScanRCurly, bool skipFormals)
bool Parser::ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, ushort flags, bool *pHasName, bool fUnaryOrParen, bool noStmtContext, bool *pNeedScanRCurly, bool skipFormals, IdentPtr* pFncNamePid)
{
ParseNodePtr pnodeFncParent = GetCurrentFunctionNode();
// is the following correct? When buildAST is false, m_currentNodeDeferredFunc can be nullptr on transition to deferred parse from non-deferred
Expand All @@ -5326,6 +5327,7 @@ bool Parser::ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, usho
StmtNest *pstmtSave;
ParseNodePtr *lastNodeRef = nullptr;
bool fFunctionInBlock = false;

if (buildAST)
{
fFunctionInBlock = GetCurrentBlockInfo() != GetCurrentFunctionBlockInfo() &&
Expand Down Expand Up @@ -5353,7 +5355,7 @@ bool Parser::ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, usho
this->UpdateCurrentNodeFunc<buildAST>(pnodeFnc, fLambda);
}

*pHasName = !fLambda && !fModule && this->ParseFncNames<buildAST>(pnodeFnc, pnodeFncSave, flags, &lastNodeRef);
*pHasName = !fLambda && !fModule && this->ParseFncNames<buildAST>(pnodeFnc, pnodeFncSave, flags, &lastNodeRef, pFncNamePid);

if (fDeclaration)
{
Expand Down Expand Up @@ -6453,7 +6455,7 @@ void Parser::ParseNestedDeferredFunc(ParseNodePtr pnodeFnc, bool fLambda, bool *
}

template<bool buildAST>
bool Parser::ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, ushort flags, ParseNodePtr **pLastNodeRef)
bool Parser::ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, ushort flags, ParseNodePtr **pLastNodeRef, IdentPtr* pFncNamePid)
{
BOOL fDeclaration = flags & fFncDeclaration;
BOOL fIsAsync = flags & fFncAsync;
Expand Down Expand Up @@ -6543,7 +6545,6 @@ bool Parser::ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, u

ichMinNames = m_pscan->IchMinTok();


Assert(m_token.tk == tkID || (m_token.tk == tkYIELD && !fDeclaration));

if (IsStrictMode())
Expand All @@ -6561,6 +6562,11 @@ bool Parser::ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, u
pnodeT->ichMin = ichMinBase;
pnodeT->ichLim = ichLimBase;

if (pFncNamePid != nullptr)
{
*pFncNamePid = pidBase;
}

if (fDeclaration &&
pnodeFncParent &&
pnodeFncParent->sxFnc.pnodeName &&
Expand Down
4 changes: 2 additions & 2 deletions lib/Parser/Parse.h
Original file line number Diff line number Diff line change
Expand Up @@ -808,9 +808,9 @@ class Parser
template<bool buildAST> void ParseComputedName(ParseNodePtr* ppnodeName, LPCOLESTR* ppNameHint, LPCOLESTR* ppFullNameHint = nullptr, uint32 *pNameLength = nullptr, uint32 *pShortNameOffset = nullptr);
template<bool buildAST> ParseNodePtr ParseMemberGetSet(OpCode nop, LPCOLESTR* ppNameHint);
template<bool buildAST> ParseNodePtr ParseFncDecl(ushort flags, LPCOLESTR pNameHint = NULL, const bool needsPIDOnRCurlyScan = false, bool resetParsingSuperRestrictionState = true, bool fUnaryOrParen = false);
template<bool buildAST> bool ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, ushort flags, ParseNodePtr **pLastNodeRef);
template<bool buildAST> bool ParseFncNames(ParseNodePtr pnodeFnc, ParseNodePtr pnodeFncParent, ushort flags, ParseNodePtr **pLastNodeRef, IdentPtr* pFncNamePid = nullptr);
template<bool buildAST> void ParseFncFormals(ParseNodePtr pnodeFnc, ParseNodePtr pnodeParentFnc, ushort flags, bool isTopLevelDeferredFunc = false);
template<bool buildAST> bool ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, ushort flags, bool *pHasName, bool fUnaryOrParen, bool noStmtContext, bool *pNeedScanRCurly, bool skipFormals = false);
template<bool buildAST> bool ParseFncDeclHelper(ParseNodePtr pnodeFnc, LPCOLESTR pNameHint, ushort flags, bool *pHasName, bool fUnaryOrParen, bool noStmtContext, bool *pNeedScanRCurly, bool skipFormals = false, IdentPtr* pFncNamePid = nullptr);
template<bool buildAST> void ParseExpressionLambdaBody(ParseNodePtr pnodeFnc);
template<bool buildAST> void UpdateCurrentNodeFunc(ParseNodePtr pnodeFnc, bool fLambda);
bool FncDeclAllowedWithoutContext(ushort flags);
Expand Down
17 changes: 17 additions & 0 deletions test/Bugs/bug_OS14115684.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
//-------------------------------------------------------------------------------------------------------
// Copyright (C) Microsoft. All rights reserved.
// Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
//-------------------------------------------------------------------------------------------------------

function test(foo) {
function nest() {
{
function foo() {
console.log('pass');
}
}
foo();
}
nest();
}
test(()=>console.log('fail'));
7 changes: 7 additions & 0 deletions test/Bugs/rlexe.xml
Original file line number Diff line number Diff line change
Expand Up @@ -444,4 +444,11 @@
<compile-flags>-args summary -endargs</compile-flags>
</default>
</test>
<test>
<default>
<files>bug_OS14115684.js</files>
<tags>BugFix</tags>
<compile-flags>-forceundodefer</compile-flags>
</default>
</test>
</regress-exe>

0 comments on commit c3d4e4d

Please sign in to comment.