1 /* 2 * Copyright (c) 2021 Huawei Device Co., Ltd. 3 * Licensed under the Apache License, Version 2.0 (the "License"); 4 * you may not use this file except in compliance with the License. 5 * You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software 10 * distributed under the License is distributed on an "AS IS" BASIS, 11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 * See the License for the specific language governing permissions and 13 * limitations under the License. 14 */ 15 16 #include "sourceTextModuleRecord.h" 17 #include <binder/scope.h> 18 19 namespace panda::es2panda::parser { AddModuleRequest(const util::StringView source)20 int SourceTextModuleRecord::AddModuleRequest(const util::StringView source) 21 { 22 ASSERT(!source.Empty()); 23 int moduleRequestsSize = static_cast<int>(moduleRequestsMap_.size()); 24 if (moduleRequestsMap_.find(source) == moduleRequestsMap_.end()) { 25 moduleRequests_.emplace_back(source); 26 } 27 auto insertedRes = moduleRequestsMap_.insert(std::make_pair(source, moduleRequestsSize)); 28 moduleRequestsIdxMap_.insert(std::make_pair(insertedRes.first->second, source)); 29 return insertedRes.first->second; 30 } 31 32 // import x from 'test.js' 33 // import {x} from 'test.js' 34 // import {x as y} from 'test.js' 35 // import defaultExport from 'test.js' AddImportEntry(SourceTextModuleRecord::ImportEntry * entry)36 void SourceTextModuleRecord::AddImportEntry(SourceTextModuleRecord::ImportEntry *entry) 37 { 38 ASSERT(!entry->importName_.Empty()); 39 ASSERT(!entry->localName_.Empty()); 40 ASSERT(entry->moduleRequestIdx_ != -1); 41 regularImportEntries_.insert(std::make_pair(entry->localName_, entry)); 42 // the implicit indirect exports should be insert into indirectExportsEntries 43 // when add an ImportEntry. 44 // e.g. export { x }; import { x } from 'test.js' 45 CheckImplicitIndirectExport(entry); 46 } 47 48 // import * as x from 'test.js' AddStarImportEntry(SourceTextModuleRecord::ImportEntry * entry)49 void SourceTextModuleRecord::AddStarImportEntry(SourceTextModuleRecord::ImportEntry *entry) 50 { 51 ASSERT(!entry->localName_.Empty()); 52 ASSERT(entry->importName_.Empty()); 53 ASSERT(entry->moduleRequestIdx_ != -1); 54 namespaceImportEntries_.push_back(entry); 55 } 56 57 // export {x} 58 // export {x as y} 59 // export VariableStatement 60 // export Declaration 61 // export default ... AddLocalExportEntry(SourceTextModuleRecord::ExportEntry * entry)62 bool SourceTextModuleRecord::AddLocalExportEntry(SourceTextModuleRecord::ExportEntry *entry) 63 { 64 ASSERT(entry->importName_.Empty()); 65 ASSERT(!entry->localName_.Empty()); 66 ASSERT(!entry->exportName_.Empty()); 67 ASSERT(entry->moduleRequestIdx_ == -1); 68 69 // the implicit indirect exports should be insert into indirectExportsEntries 70 // when add an ExportEntry. 71 // e.g. import { x } from 'test.js'; export { x } 72 if (CheckImplicitIndirectExport(entry)) { 73 return true; 74 } 75 if (!HasDuplicateExport(entry->exportName_)) { 76 localExportEntries_.insert(std::make_pair(entry->localName_, entry)); 77 return true; 78 } 79 return false; 80 } 81 82 // export {x} from 'test.js' 83 // export {x as y} from 'test.js' 84 // import { x } from 'test.js'; export { x } AddIndirectExportEntry(SourceTextModuleRecord::ExportEntry * entry)85 bool SourceTextModuleRecord::AddIndirectExportEntry(SourceTextModuleRecord::ExportEntry *entry) 86 { 87 ASSERT(!entry->importName_.Empty()); 88 ASSERT(!entry->exportName_.Empty()); 89 ASSERT(entry->localName_.Empty()); 90 ASSERT(entry->moduleRequestIdx_ != -1); 91 if (!HasDuplicateExport(entry->exportName_)) { 92 indirectExportEntries_.push_back(entry); 93 return true; 94 } 95 return false; 96 } 97 98 // export * from 'test.js' AddStarExportEntry(SourceTextModuleRecord::ExportEntry * entry)99 void SourceTextModuleRecord::AddStarExportEntry(SourceTextModuleRecord::ExportEntry *entry) 100 { 101 ASSERT(entry->importName_.Empty()); 102 ASSERT(entry->localName_.Empty()); 103 ASSERT(entry->exportName_.Empty()); 104 ASSERT(entry->moduleRequestIdx_ != -1); 105 starExportEntries_.push_back(entry); 106 } 107 HasDuplicateExport(util::StringView exportName) const108 bool SourceTextModuleRecord::HasDuplicateExport(util::StringView exportName) const 109 { 110 for (auto const &entryUnit : localExportEntries_) { 111 const SourceTextModuleRecord::ExportEntry *e = entryUnit.second; 112 if (exportName == e->exportName_) { 113 return true; 114 } 115 } 116 117 for (const auto *e : indirectExportEntries_) { 118 if (exportName == e->exportName_) { 119 return true; 120 } 121 } 122 123 return false; 124 } 125 CheckImplicitIndirectExport(SourceTextModuleRecord::ExportEntry * exportEntry)126 bool SourceTextModuleRecord::CheckImplicitIndirectExport(SourceTextModuleRecord::ExportEntry *exportEntry) 127 { 128 ASSERT(!exportEntry->localName_.Empty()); 129 auto regularImport = regularImportEntries_.find(exportEntry->localName_); 130 if (regularImport != regularImportEntries_.end()) { 131 ConvertLocalExportToIndirect(regularImport->second, exportEntry); 132 return AddIndirectExportEntry(exportEntry); 133 } 134 return false; 135 } 136 CheckImplicitIndirectExport(SourceTextModuleRecord::ImportEntry * importEntry)137 void SourceTextModuleRecord::CheckImplicitIndirectExport(SourceTextModuleRecord::ImportEntry *importEntry) 138 { 139 ASSERT(!importEntry->localName_.Empty()); 140 auto range = localExportEntries_.equal_range(importEntry->localName_); 141 // not found implicit indirect 142 if (range.first == range.second) { 143 return; 144 } 145 146 for (auto it = range.first; it != range.second; ++it) { 147 SourceTextModuleRecord::ExportEntry *exportEntry = it->second; 148 ConvertLocalExportToIndirect(importEntry, exportEntry); 149 indirectExportEntries_.push_back(exportEntry); 150 } 151 localExportEntries_.erase(range.first, range.second); 152 } 153 ConvertLocalExportToIndirect(SourceTextModuleRecord::ImportEntry * importEntry,SourceTextModuleRecord::ExportEntry * exportEntry)154 void SourceTextModuleRecord::ConvertLocalExportToIndirect(SourceTextModuleRecord::ImportEntry *importEntry, 155 SourceTextModuleRecord::ExportEntry *exportEntry) 156 { 157 ASSERT(exportEntry->importName_.Empty()); 158 ASSERT(exportEntry->moduleRequestIdx_ == -1); 159 ASSERT(!importEntry->importName_.Empty()); 160 ASSERT(importEntry->moduleRequestIdx_ != -1); 161 exportEntry->importName_ = importEntry->importName_; 162 exportEntry->moduleRequestIdx_ = importEntry->moduleRequestIdx_; 163 exportEntry->localName_ = util::StringView(""); 164 } 165 AssignIndexToModuleVariable(binder::ModuleScope * moduleScope)166 void SourceTextModuleRecord::AssignIndexToModuleVariable(binder::ModuleScope *moduleScope) 167 { 168 uint32_t index = 0; 169 for (auto it = localExportEntries_.begin(); it != localExportEntries_.end(); 170 it = localExportEntries_.upper_bound(it->first)) { 171 CheckAndAssignIndex(moduleScope, it->first, &index); 172 } 173 174 index = 0; 175 for (const auto &elem : regularImportEntries_) { 176 CheckAndAssignIndex(moduleScope, elem.first, &index); 177 } 178 } 179 CheckAndAssignIndex(binder::ModuleScope * moduleScope,util::StringView name,uint32_t * index) const180 void SourceTextModuleRecord::CheckAndAssignIndex(binder::ModuleScope *moduleScope, 181 util::StringView name, 182 uint32_t *index) const 183 { 184 if (moduleScope->FindLocal(name) != nullptr) { 185 moduleScope->AssignIndexToModuleVariable(name, *index); 186 (*index)++; 187 } 188 } 189 RemoveDefaultLocalExportEntry()190 void SourceTextModuleRecord::RemoveDefaultLocalExportEntry() 191 { 192 util::StringView localName = parser::SourceTextModuleRecord::DEFAULT_LOCAL_NAME; 193 localExportEntries_.erase(localName); 194 } 195 } // namespace panda::es2panda::parser 196