darling-JavaScriptCore/parser/ModuleAnalyzer.cpp

149 lines
6.7 KiB
C++

/*
* Copyright (C) 2015-2019 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "ModuleAnalyzer.h"
#include "IdentifierInlines.h"
#include "JSGlobalObject.h"
#include "JSModuleRecord.h"
#include "ModuleScopeData.h"
#include "StrongInlines.h"
namespace JSC {
ModuleAnalyzer::ModuleAnalyzer(JSGlobalObject* globalObject, const Identifier& moduleKey, const SourceCode& sourceCode, const VariableEnvironment& declaredVariables, const VariableEnvironment& lexicalVariables)
: m_vm(globalObject->vm())
, m_moduleRecord(m_vm, JSModuleRecord::create(globalObject, m_vm, globalObject->moduleRecordStructure(), moduleKey, sourceCode, declaredVariables, lexicalVariables))
{
}
void ModuleAnalyzer::exportVariable(ModuleProgramNode& moduleProgramNode, const RefPtr<UniquedStringImpl>& localName, const VariableEnvironmentEntry& variable)
{
// In the parser, we already marked the variables as Exported and Imported.
// By leveraging this information, we collect the information that is needed
// to construct the module environment.
//
// I E
// * = exported module local variable
// * = imported binding
// = non-exported module local variable
// * * = indirect exported binding
//
// One exception is namespace binding (like import * as ns from "mod").
// This is annotated as an imported, but the actual binding is locate in the
// current module.
if (!variable.isExported())
return;
// Exported module local variable.
if (!variable.isImported()) {
for (auto& exportName : moduleProgramNode.moduleScopeData().exportedBindings().get(localName.get()))
moduleRecord()->addExportEntry(JSModuleRecord::ExportEntry::createLocal(Identifier::fromUid(m_vm, exportName.get()), Identifier::fromUid(m_vm, localName.get())));
return;
}
if (variable.isImportedNamespace()) {
// Exported namespace binding.
// import * as namespace from "mod"
// export { namespace }
//
// Sec 15.2.1.16.1 step 11-a-ii-2-b https://tc39.github.io/ecma262/#sec-parsemodule
// Namespace export is handled as local export since a namespace object binding itself is implemented as a local binding.
for (auto& exportName : moduleProgramNode.moduleScopeData().exportedBindings().get(localName.get()))
moduleRecord()->addExportEntry(JSModuleRecord::ExportEntry::createLocal(Identifier::fromUid(m_vm, exportName.get()), Identifier::fromUid(m_vm, localName.get())));
return;
}
// Indirectly exported binding.
// import a from "mod"
// export { a }
Optional<JSModuleRecord::ImportEntry> optionalImportEntry = moduleRecord()->tryGetImportEntry(localName.get());
ASSERT(optionalImportEntry);
const JSModuleRecord::ImportEntry& importEntry = *optionalImportEntry;
for (auto& exportName : moduleProgramNode.moduleScopeData().exportedBindings().get(localName.get()))
moduleRecord()->addExportEntry(JSModuleRecord::ExportEntry::createIndirect(Identifier::fromUid(m_vm, exportName.get()), importEntry.importName, importEntry.moduleRequest));
}
JSModuleRecord* ModuleAnalyzer::analyze(ModuleProgramNode& moduleProgramNode)
{
// Traverse the module AST and collect
// * Import entries
// * Export entries that have FromClause (e.g. export { a } from "mod")
// * Export entries that have star (e.g. export * from "mod")
// * Aliased export names (e.g. export { a as b })
moduleProgramNode.analyzeModule(*this);
// Based on the collected information, categorize export entries into 3 types.
// 1. Local export entries
// This references the local variable in the current module.
// This variable should be allocated in the current module environment as a heap variable.
//
// const variable = 20
// export { variable }
//
// 2. Namespace export entries
// This references the namespace object imported by some import entries.
// This variable itself should be allocated in the current module environment as a heap variable.
// But when the other modules attempt to resolve this export name in this module, this module
// should tell the link to the original module.
//
// import * as namespace from "mod"
// export { namespace as mod }
//
// 3. Indirect export entries
// This references the imported binding name from the other module.
// This module environment itself should hold the pointer to (1) the original module and
// (2) the binding in the original module. The variable itself is allocated in the original
// module. This indirect binding is resolved when the CodeBlock resolves the references.
//
// import mod from "mod"
// export { mod }
//
// export { a } from "mod"
//
// And separeted from the above 3 types, we also collect the star export entries.
//
// 4. Star export entries
// This exports all the names from the specified external module as the current module's name.
//
// export * from "mod"
for (const auto& pair : m_moduleRecord->declaredVariables())
exportVariable(moduleProgramNode, pair.key, pair.value);
for (const auto& pair : m_moduleRecord->lexicalVariables())
exportVariable(moduleProgramNode, pair.key, pair.value);
if (UNLIKELY(Options::dumpModuleRecord()))
m_moduleRecord->dump();
return m_moduleRecord.get();
}
} // namespace JSC