1 // Copyright 2017 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <unordered_map>
6 #include <unordered_set>
7 
8 #include "src/objects/module.h"
9 
10 #include "src/accessors.h"
11 #include "src/api-inl.h"
12 #include "src/ast/modules.h"
13 #include "src/objects-inl.h"
14 #include "src/objects/hash-table-inl.h"
15 #include "src/objects/js-generator-inl.h"
16 #include "src/objects/module-inl.h"
17 
18 namespace v8 {
19 namespace internal {
20 
21 namespace {
22 
23 struct ModuleHandleHash {
operator ()v8::internal::__anon266fc0380111::ModuleHandleHash24   V8_INLINE size_t operator()(Handle<Module> module) const {
25     return module->hash();
26   }
27 };
28 
29 struct ModuleHandleEqual {
operator ()v8::internal::__anon266fc0380111::ModuleHandleEqual30   V8_INLINE bool operator()(Handle<Module> lhs, Handle<Module> rhs) const {
31     return *lhs == *rhs;
32   }
33 };
34 
35 struct StringHandleHash {
operator ()v8::internal::__anon266fc0380111::StringHandleHash36   V8_INLINE size_t operator()(Handle<String> string) const {
37     return string->Hash();
38   }
39 };
40 
41 struct StringHandleEqual {
operator ()v8::internal::__anon266fc0380111::StringHandleEqual42   V8_INLINE bool operator()(Handle<String> lhs, Handle<String> rhs) const {
43     return lhs->Equals(*rhs);
44   }
45 };
46 
47 class UnorderedStringSet
48     : public std::unordered_set<Handle<String>, StringHandleHash,
49                                 StringHandleEqual,
50                                 ZoneAllocator<Handle<String>>> {
51  public:
UnorderedStringSet(Zone * zone)52   explicit UnorderedStringSet(Zone* zone)
53       : std::unordered_set<Handle<String>, StringHandleHash, StringHandleEqual,
54                            ZoneAllocator<Handle<String>>>(
55             2 /* bucket count */, StringHandleHash(), StringHandleEqual(),
56             ZoneAllocator<Handle<String>>(zone)) {}
57 };
58 
59 class UnorderedModuleSet
60     : public std::unordered_set<Handle<Module>, ModuleHandleHash,
61                                 ModuleHandleEqual,
62                                 ZoneAllocator<Handle<Module>>> {
63  public:
UnorderedModuleSet(Zone * zone)64   explicit UnorderedModuleSet(Zone* zone)
65       : std::unordered_set<Handle<Module>, ModuleHandleHash, ModuleHandleEqual,
66                            ZoneAllocator<Handle<Module>>>(
67             2 /* bucket count */, ModuleHandleHash(), ModuleHandleEqual(),
68             ZoneAllocator<Handle<Module>>(zone)) {}
69 };
70 
71 class UnorderedStringMap
72     : public std::unordered_map<
73           Handle<String>, Handle<Object>, StringHandleHash, StringHandleEqual,
74           ZoneAllocator<std::pair<const Handle<String>, Handle<Object>>>> {
75  public:
UnorderedStringMap(Zone * zone)76   explicit UnorderedStringMap(Zone* zone)
77       : std::unordered_map<
78             Handle<String>, Handle<Object>, StringHandleHash, StringHandleEqual,
79             ZoneAllocator<std::pair<const Handle<String>, Handle<Object>>>>(
80             2 /* bucket count */, StringHandleHash(), StringHandleEqual(),
81             ZoneAllocator<std::pair<const Handle<String>, Handle<Object>>>(
82                 zone)) {}
83 };
84 
85 }  // anonymous namespace
86 
87 class Module::ResolveSet
88     : public std::unordered_map<
89           Handle<Module>, UnorderedStringSet*, ModuleHandleHash,
90           ModuleHandleEqual,
91           ZoneAllocator<std::pair<const Handle<Module>, UnorderedStringSet*>>> {
92  public:
ResolveSet(Zone * zone)93   explicit ResolveSet(Zone* zone)
94       : std::unordered_map<Handle<Module>, UnorderedStringSet*,
95                            ModuleHandleHash, ModuleHandleEqual,
96                            ZoneAllocator<std::pair<const Handle<Module>,
97                                                    UnorderedStringSet*>>>(
98             2 /* bucket count */, ModuleHandleHash(), ModuleHandleEqual(),
99             ZoneAllocator<std::pair<const Handle<Module>, UnorderedStringSet*>>(
100                 zone)),
101         zone_(zone) {}
102 
zone() const103   Zone* zone() const { return zone_; }
104 
105  private:
106   Zone* zone_;
107 };
108 
109 namespace {
110 
ExportIndex(int cell_index)111 int ExportIndex(int cell_index) {
112   DCHECK_EQ(ModuleDescriptor::GetCellIndexKind(cell_index),
113             ModuleDescriptor::kExport);
114   return cell_index - 1;
115 }
116 
ImportIndex(int cell_index)117 int ImportIndex(int cell_index) {
118   DCHECK_EQ(ModuleDescriptor::GetCellIndexKind(cell_index),
119             ModuleDescriptor::kImport);
120   return -cell_index - 1;
121 }
122 
123 }  // anonymous namespace
124 
CreateIndirectExport(Isolate * isolate,Handle<Module> module,Handle<String> name,Handle<ModuleInfoEntry> entry)125 void Module::CreateIndirectExport(Isolate* isolate, Handle<Module> module,
126                                   Handle<String> name,
127                                   Handle<ModuleInfoEntry> entry) {
128   Handle<ObjectHashTable> exports(module->exports(), isolate);
129   DCHECK(exports->Lookup(name)->IsTheHole(isolate));
130   exports = ObjectHashTable::Put(exports, name, entry);
131   module->set_exports(*exports);
132 }
133 
CreateExport(Isolate * isolate,Handle<Module> module,int cell_index,Handle<FixedArray> names)134 void Module::CreateExport(Isolate* isolate, Handle<Module> module,
135                           int cell_index, Handle<FixedArray> names) {
136   DCHECK_LT(0, names->length());
137   Handle<Cell> cell =
138       isolate->factory()->NewCell(isolate->factory()->undefined_value());
139   module->regular_exports()->set(ExportIndex(cell_index), *cell);
140 
141   Handle<ObjectHashTable> exports(module->exports(), isolate);
142   for (int i = 0, n = names->length(); i < n; ++i) {
143     Handle<String> name(String::cast(names->get(i)), isolate);
144     DCHECK(exports->Lookup(name)->IsTheHole(isolate));
145     exports = ObjectHashTable::Put(exports, name, cell);
146   }
147   module->set_exports(*exports);
148 }
149 
GetCell(int cell_index)150 Cell* Module::GetCell(int cell_index) {
151   DisallowHeapAllocation no_gc;
152   Object* cell;
153   switch (ModuleDescriptor::GetCellIndexKind(cell_index)) {
154     case ModuleDescriptor::kImport:
155       cell = regular_imports()->get(ImportIndex(cell_index));
156       break;
157     case ModuleDescriptor::kExport:
158       cell = regular_exports()->get(ExportIndex(cell_index));
159       break;
160     case ModuleDescriptor::kInvalid:
161       UNREACHABLE();
162       break;
163   }
164   return Cell::cast(cell);
165 }
166 
LoadVariable(Isolate * isolate,Handle<Module> module,int cell_index)167 Handle<Object> Module::LoadVariable(Isolate* isolate, Handle<Module> module,
168                                     int cell_index) {
169   return handle(module->GetCell(cell_index)->value(), isolate);
170 }
171 
StoreVariable(Handle<Module> module,int cell_index,Handle<Object> value)172 void Module::StoreVariable(Handle<Module> module, int cell_index,
173                            Handle<Object> value) {
174   DCHECK_EQ(ModuleDescriptor::GetCellIndexKind(cell_index),
175             ModuleDescriptor::kExport);
176   module->GetCell(cell_index)->set_value(*value);
177 }
178 
179 #ifdef DEBUG
PrintStatusTransition(Status new_status)180 void Module::PrintStatusTransition(Status new_status) {
181   if (FLAG_trace_module_status) {
182     StdoutStream os;
183     os << "Changing module status from " << status() << " to " << new_status
184        << " for ";
185     script()->GetNameOrSourceURL()->Print(os);
186 #ifndef OBJECT_PRINT
187     os << "\n";
188 #endif  // OBJECT_PRINT
189   }
190 }
191 #endif  // DEBUG
192 
SetStatus(Status new_status)193 void Module::SetStatus(Status new_status) {
194   DisallowHeapAllocation no_alloc;
195   DCHECK_LE(status(), new_status);
196   DCHECK_NE(new_status, Module::kErrored);
197 #ifdef DEBUG
198   PrintStatusTransition(new_status);
199 #endif  // DEBUG
200   set_status(new_status);
201 }
202 
ResetGraph(Isolate * isolate,Handle<Module> module)203 void Module::ResetGraph(Isolate* isolate, Handle<Module> module) {
204   DCHECK_NE(module->status(), kInstantiating);
205   DCHECK_NE(module->status(), kEvaluating);
206   if (module->status() != kPreInstantiating) return;
207   Handle<FixedArray> requested_modules(module->requested_modules(), isolate);
208   Reset(isolate, module);
209   for (int i = 0; i < requested_modules->length(); ++i) {
210     Handle<Object> descendant(requested_modules->get(i), isolate);
211     if (descendant->IsModule()) {
212       ResetGraph(isolate, Handle<Module>::cast(descendant));
213     } else {
214       DCHECK(descendant->IsUndefined(isolate));
215     }
216   }
217 }
218 
Reset(Isolate * isolate,Handle<Module> module)219 void Module::Reset(Isolate* isolate, Handle<Module> module) {
220   Factory* factory = isolate->factory();
221 
222   DCHECK(module->status() == kPreInstantiating ||
223          module->status() == kInstantiating);
224   DCHECK(module->exception()->IsTheHole(isolate));
225   DCHECK(module->import_meta()->IsTheHole(isolate));
226   // The namespace object cannot exist, because it would have been created
227   // by RunInitializationCode, which is called only after this module's SCC
228   // succeeds instantiation.
229   DCHECK(!module->module_namespace()->IsJSModuleNamespace());
230 
231   Handle<ObjectHashTable> exports =
232       ObjectHashTable::New(isolate, module->info()->RegularExportCount());
233   Handle<FixedArray> regular_exports =
234       factory->NewFixedArray(module->regular_exports()->length());
235   Handle<FixedArray> regular_imports =
236       factory->NewFixedArray(module->regular_imports()->length());
237   Handle<FixedArray> requested_modules =
238       factory->NewFixedArray(module->requested_modules()->length());
239 
240   if (module->status() == kInstantiating) {
241     module->set_code(JSFunction::cast(module->code())->shared());
242   }
243 #ifdef DEBUG
244   module->PrintStatusTransition(kUninstantiated);
245 #endif  // DEBUG
246   module->set_status(kUninstantiated);
247   module->set_exports(*exports);
248   module->set_regular_exports(*regular_exports);
249   module->set_regular_imports(*regular_imports);
250   module->set_requested_modules(*requested_modules);
251   module->set_dfs_index(-1);
252   module->set_dfs_ancestor_index(-1);
253 }
254 
RecordError(Isolate * isolate)255 void Module::RecordError(Isolate* isolate) {
256   DisallowHeapAllocation no_alloc;
257   DCHECK(exception()->IsTheHole(isolate));
258   Object* the_exception = isolate->pending_exception();
259   DCHECK(!the_exception->IsTheHole(isolate));
260 
261   set_code(info());
262 #ifdef DEBUG
263   PrintStatusTransition(Module::kErrored);
264 #endif  // DEBUG
265   set_status(Module::kErrored);
266   set_exception(the_exception);
267 }
268 
GetException()269 Object* Module::GetException() {
270   DisallowHeapAllocation no_alloc;
271   DCHECK_EQ(status(), Module::kErrored);
272   DCHECK(!exception()->IsTheHole());
273   return exception();
274 }
275 
GetSharedFunctionInfo() const276 SharedFunctionInfo* Module::GetSharedFunctionInfo() const {
277   DisallowHeapAllocation no_alloc;
278   DCHECK_NE(status(), Module::kEvaluating);
279   DCHECK_NE(status(), Module::kEvaluated);
280   switch (status()) {
281     case kUninstantiated:
282     case kPreInstantiating:
283       DCHECK(code()->IsSharedFunctionInfo());
284       return SharedFunctionInfo::cast(code());
285     case kInstantiating:
286       DCHECK(code()->IsJSFunction());
287       return JSFunction::cast(code())->shared();
288     case kInstantiated:
289       DCHECK(code()->IsJSGeneratorObject());
290       return JSGeneratorObject::cast(code())->function()->shared();
291     case kEvaluating:
292     case kEvaluated:
293     case kErrored:
294       UNREACHABLE();
295   }
296 
297   UNREACHABLE();
298 }
299 
ResolveImport(Isolate * isolate,Handle<Module> module,Handle<String> name,int module_request,MessageLocation loc,bool must_resolve,Module::ResolveSet * resolve_set)300 MaybeHandle<Cell> Module::ResolveImport(Isolate* isolate, Handle<Module> module,
301                                         Handle<String> name, int module_request,
302                                         MessageLocation loc, bool must_resolve,
303                                         Module::ResolveSet* resolve_set) {
304   Handle<Module> requested_module(
305       Module::cast(module->requested_modules()->get(module_request)), isolate);
306   Handle<String> specifier(
307       String::cast(module->info()->module_requests()->get(module_request)),
308       isolate);
309   MaybeHandle<Cell> result =
310       Module::ResolveExport(isolate, requested_module, specifier, name, loc,
311                             must_resolve, resolve_set);
312   DCHECK_IMPLIES(isolate->has_pending_exception(), result.is_null());
313   return result;
314 }
315 
ResolveExport(Isolate * isolate,Handle<Module> module,Handle<String> module_specifier,Handle<String> export_name,MessageLocation loc,bool must_resolve,Module::ResolveSet * resolve_set)316 MaybeHandle<Cell> Module::ResolveExport(Isolate* isolate, Handle<Module> module,
317                                         Handle<String> module_specifier,
318                                         Handle<String> export_name,
319                                         MessageLocation loc, bool must_resolve,
320                                         Module::ResolveSet* resolve_set) {
321   DCHECK_GE(module->status(), kPreInstantiating);
322   DCHECK_NE(module->status(), kEvaluating);
323   Handle<Object> object(module->exports()->Lookup(export_name), isolate);
324   if (object->IsCell()) {
325     // Already resolved (e.g. because it's a local export).
326     return Handle<Cell>::cast(object);
327   }
328 
329   // Check for cycle before recursing.
330   {
331     // Attempt insertion with a null string set.
332     auto result = resolve_set->insert({module, nullptr});
333     UnorderedStringSet*& name_set = result.first->second;
334     if (result.second) {
335       // |module| wasn't in the map previously, so allocate a new name set.
336       Zone* zone = resolve_set->zone();
337       name_set =
338           new (zone->New(sizeof(UnorderedStringSet))) UnorderedStringSet(zone);
339     } else if (name_set->count(export_name)) {
340       // Cycle detected.
341       if (must_resolve) {
342         return isolate->Throw<Cell>(
343             isolate->factory()->NewSyntaxError(
344                 MessageTemplate::kCyclicModuleDependency, export_name,
345                 module_specifier),
346             &loc);
347       }
348       return MaybeHandle<Cell>();
349     }
350     name_set->insert(export_name);
351   }
352 
353   if (object->IsModuleInfoEntry()) {
354     // Not yet resolved indirect export.
355     Handle<ModuleInfoEntry> entry = Handle<ModuleInfoEntry>::cast(object);
356     Handle<String> import_name(String::cast(entry->import_name()), isolate);
357     Handle<Script> script(module->script(), isolate);
358     MessageLocation new_loc(script, entry->beg_pos(), entry->end_pos());
359 
360     Handle<Cell> cell;
361     if (!ResolveImport(isolate, module, import_name, entry->module_request(),
362                        new_loc, true, resolve_set)
363              .ToHandle(&cell)) {
364       DCHECK(isolate->has_pending_exception());
365       return MaybeHandle<Cell>();
366     }
367 
368     // The export table may have changed but the entry in question should be
369     // unchanged.
370     Handle<ObjectHashTable> exports(module->exports(), isolate);
371     DCHECK(exports->Lookup(export_name)->IsModuleInfoEntry());
372 
373     exports = ObjectHashTable::Put(exports, export_name, cell);
374     module->set_exports(*exports);
375     return cell;
376   }
377 
378   DCHECK(object->IsTheHole(isolate));
379   return Module::ResolveExportUsingStarExports(isolate, module,
380                                                module_specifier, export_name,
381                                                loc, must_resolve, resolve_set);
382 }
383 
ResolveExportUsingStarExports(Isolate * isolate,Handle<Module> module,Handle<String> module_specifier,Handle<String> export_name,MessageLocation loc,bool must_resolve,Module::ResolveSet * resolve_set)384 MaybeHandle<Cell> Module::ResolveExportUsingStarExports(
385     Isolate* isolate, Handle<Module> module, Handle<String> module_specifier,
386     Handle<String> export_name, MessageLocation loc, bool must_resolve,
387     Module::ResolveSet* resolve_set) {
388   if (!export_name->Equals(ReadOnlyRoots(isolate).default_string())) {
389     // Go through all star exports looking for the given name.  If multiple star
390     // exports provide the name, make sure they all map it to the same cell.
391     Handle<Cell> unique_cell;
392     Handle<FixedArray> special_exports(module->info()->special_exports(),
393                                        isolate);
394     for (int i = 0, n = special_exports->length(); i < n; ++i) {
395       i::Handle<i::ModuleInfoEntry> entry(
396           i::ModuleInfoEntry::cast(special_exports->get(i)), isolate);
397       if (!entry->export_name()->IsUndefined(isolate)) {
398         continue;  // Indirect export.
399       }
400 
401       Handle<Script> script(module->script(), isolate);
402       MessageLocation new_loc(script, entry->beg_pos(), entry->end_pos());
403 
404       Handle<Cell> cell;
405       if (ResolveImport(isolate, module, export_name, entry->module_request(),
406                         new_loc, false, resolve_set)
407               .ToHandle(&cell)) {
408         if (unique_cell.is_null()) unique_cell = cell;
409         if (*unique_cell != *cell) {
410           return isolate->Throw<Cell>(isolate->factory()->NewSyntaxError(
411                                           MessageTemplate::kAmbiguousExport,
412                                           module_specifier, export_name),
413                                       &loc);
414         }
415       } else if (isolate->has_pending_exception()) {
416         return MaybeHandle<Cell>();
417       }
418     }
419 
420     if (!unique_cell.is_null()) {
421       // Found a unique star export for this name.
422       Handle<ObjectHashTable> exports(module->exports(), isolate);
423       DCHECK(exports->Lookup(export_name)->IsTheHole(isolate));
424       exports = ObjectHashTable::Put(exports, export_name, unique_cell);
425       module->set_exports(*exports);
426       return unique_cell;
427     }
428   }
429 
430   // Unresolvable.
431   if (must_resolve) {
432     return isolate->Throw<Cell>(
433         isolate->factory()->NewSyntaxError(MessageTemplate::kUnresolvableExport,
434                                            module_specifier, export_name),
435         &loc);
436   }
437   return MaybeHandle<Cell>();
438 }
439 
Instantiate(Isolate * isolate,Handle<Module> module,v8::Local<v8::Context> context,v8::Module::ResolveCallback callback)440 bool Module::Instantiate(Isolate* isolate, Handle<Module> module,
441                          v8::Local<v8::Context> context,
442                          v8::Module::ResolveCallback callback) {
443 #ifdef DEBUG
444   if (FLAG_trace_module_status) {
445     StdoutStream os;
446     os << "Instantiating module ";
447     module->script()->GetNameOrSourceURL()->Print(os);
448 #ifndef OBJECT_PRINT
449     os << "\n";
450 #endif  // OBJECT_PRINT
451   }
452 #endif  // DEBUG
453 
454   if (!PrepareInstantiate(isolate, module, context, callback)) {
455     ResetGraph(isolate, module);
456     return false;
457   }
458   Zone zone(isolate->allocator(), ZONE_NAME);
459   ZoneForwardList<Handle<Module>> stack(&zone);
460   unsigned dfs_index = 0;
461   if (!FinishInstantiate(isolate, module, &stack, &dfs_index, &zone)) {
462     for (auto& descendant : stack) {
463       Reset(isolate, descendant);
464     }
465     DCHECK_EQ(module->status(), kUninstantiated);
466     return false;
467   }
468   DCHECK(module->status() == kInstantiated || module->status() == kEvaluated ||
469          module->status() == kErrored);
470   DCHECK(stack.empty());
471   return true;
472 }
473 
PrepareInstantiate(Isolate * isolate,Handle<Module> module,v8::Local<v8::Context> context,v8::Module::ResolveCallback callback)474 bool Module::PrepareInstantiate(Isolate* isolate, Handle<Module> module,
475                                 v8::Local<v8::Context> context,
476                                 v8::Module::ResolveCallback callback) {
477   DCHECK_NE(module->status(), kEvaluating);
478   DCHECK_NE(module->status(), kInstantiating);
479   if (module->status() >= kPreInstantiating) return true;
480   module->SetStatus(kPreInstantiating);
481   STACK_CHECK(isolate, false);
482 
483   // Obtain requested modules.
484   Handle<ModuleInfo> module_info(module->info(), isolate);
485   Handle<FixedArray> module_requests(module_info->module_requests(), isolate);
486   Handle<FixedArray> requested_modules(module->requested_modules(), isolate);
487   for (int i = 0, length = module_requests->length(); i < length; ++i) {
488     Handle<String> specifier(String::cast(module_requests->get(i)), isolate);
489     v8::Local<v8::Module> api_requested_module;
490     if (!callback(context, v8::Utils::ToLocal(specifier),
491                   v8::Utils::ToLocal(module))
492              .ToLocal(&api_requested_module)) {
493       isolate->PromoteScheduledException();
494       return false;
495     }
496     Handle<Module> requested_module = Utils::OpenHandle(*api_requested_module);
497     requested_modules->set(i, *requested_module);
498   }
499 
500   // Recurse.
501   for (int i = 0, length = requested_modules->length(); i < length; ++i) {
502     Handle<Module> requested_module(Module::cast(requested_modules->get(i)),
503                                     isolate);
504     if (!PrepareInstantiate(isolate, requested_module, context, callback)) {
505       return false;
506     }
507   }
508 
509   // Set up local exports.
510   // TODO(neis): Create regular_exports array here instead of in factory method?
511   for (int i = 0, n = module_info->RegularExportCount(); i < n; ++i) {
512     int cell_index = module_info->RegularExportCellIndex(i);
513     Handle<FixedArray> export_names(module_info->RegularExportExportNames(i),
514                                     isolate);
515     CreateExport(isolate, module, cell_index, export_names);
516   }
517 
518   // Partially set up indirect exports.
519   // For each indirect export, we create the appropriate slot in the export
520   // table and store its ModuleInfoEntry there.  When we later find the correct
521   // Cell in the module that actually provides the value, we replace the
522   // ModuleInfoEntry by that Cell (see ResolveExport).
523   Handle<FixedArray> special_exports(module_info->special_exports(), isolate);
524   for (int i = 0, n = special_exports->length(); i < n; ++i) {
525     Handle<ModuleInfoEntry> entry(
526         ModuleInfoEntry::cast(special_exports->get(i)), isolate);
527     Handle<Object> export_name(entry->export_name(), isolate);
528     if (export_name->IsUndefined(isolate)) continue;  // Star export.
529     CreateIndirectExport(isolate, module, Handle<String>::cast(export_name),
530                          entry);
531   }
532 
533   DCHECK_EQ(module->status(), kPreInstantiating);
534   return true;
535 }
536 
RunInitializationCode(Isolate * isolate,Handle<Module> module)537 bool Module::RunInitializationCode(Isolate* isolate, Handle<Module> module) {
538   DCHECK_EQ(module->status(), kInstantiating);
539   Handle<JSFunction> function(JSFunction::cast(module->code()), isolate);
540   DCHECK_EQ(MODULE_SCOPE, function->shared()->scope_info()->scope_type());
541   Handle<Object> receiver = isolate->factory()->undefined_value();
542   Handle<Object> argv[] = {module};
543   MaybeHandle<Object> maybe_generator =
544       Execution::Call(isolate, function, receiver, arraysize(argv), argv);
545   Handle<Object> generator;
546   if (!maybe_generator.ToHandle(&generator)) {
547     DCHECK(isolate->has_pending_exception());
548     return false;
549   }
550   DCHECK_EQ(*function, Handle<JSGeneratorObject>::cast(generator)->function());
551   module->set_code(*generator);
552   return true;
553 }
554 
MaybeTransitionComponent(Isolate * isolate,Handle<Module> module,ZoneForwardList<Handle<Module>> * stack,Status new_status)555 bool Module::MaybeTransitionComponent(Isolate* isolate, Handle<Module> module,
556                                       ZoneForwardList<Handle<Module>>* stack,
557                                       Status new_status) {
558   DCHECK(new_status == kInstantiated || new_status == kEvaluated);
559   SLOW_DCHECK(
560       // {module} is on the {stack}.
561       std::count_if(stack->begin(), stack->end(),
562                     [&](Handle<Module> m) { return *m == *module; }) == 1);
563   DCHECK_LE(module->dfs_ancestor_index(), module->dfs_index());
564   if (module->dfs_ancestor_index() == module->dfs_index()) {
565     // This is the root of its strongly connected component.
566     Handle<Module> ancestor;
567     do {
568       ancestor = stack->front();
569       stack->pop_front();
570       DCHECK_EQ(ancestor->status(),
571                 new_status == kInstantiated ? kInstantiating : kEvaluating);
572       if (new_status == kInstantiated) {
573         if (!RunInitializationCode(isolate, ancestor)) return false;
574       }
575       ancestor->SetStatus(new_status);
576     } while (*ancestor != *module);
577   }
578   return true;
579 }
580 
FinishInstantiate(Isolate * isolate,Handle<Module> module,ZoneForwardList<Handle<Module>> * stack,unsigned * dfs_index,Zone * zone)581 bool Module::FinishInstantiate(Isolate* isolate, Handle<Module> module,
582                                ZoneForwardList<Handle<Module>>* stack,
583                                unsigned* dfs_index, Zone* zone) {
584   DCHECK_NE(module->status(), kEvaluating);
585   if (module->status() >= kInstantiating) return true;
586   DCHECK_EQ(module->status(), kPreInstantiating);
587   STACK_CHECK(isolate, false);
588 
589   // Instantiate SharedFunctionInfo and mark module as instantiating for
590   // the recursion.
591   Handle<SharedFunctionInfo> shared(SharedFunctionInfo::cast(module->code()),
592                                     isolate);
593   Handle<JSFunction> function =
594       isolate->factory()->NewFunctionFromSharedFunctionInfo(
595           shared, isolate->native_context());
596   module->set_code(*function);
597   module->SetStatus(kInstantiating);
598   module->set_dfs_index(*dfs_index);
599   module->set_dfs_ancestor_index(*dfs_index);
600   stack->push_front(module);
601   (*dfs_index)++;
602 
603   // Recurse.
604   Handle<FixedArray> requested_modules(module->requested_modules(), isolate);
605   for (int i = 0, length = requested_modules->length(); i < length; ++i) {
606     Handle<Module> requested_module(Module::cast(requested_modules->get(i)),
607                                     isolate);
608     if (!FinishInstantiate(isolate, requested_module, stack, dfs_index, zone)) {
609       return false;
610     }
611 
612     DCHECK_NE(requested_module->status(), kEvaluating);
613     DCHECK_GE(requested_module->status(), kInstantiating);
614     SLOW_DCHECK(
615         // {requested_module} is instantiating iff it's on the {stack}.
616         (requested_module->status() == kInstantiating) ==
617         std::count_if(stack->begin(), stack->end(), [&](Handle<Module> m) {
618           return *m == *requested_module;
619         }));
620 
621     if (requested_module->status() == kInstantiating) {
622       module->set_dfs_ancestor_index(
623           std::min(module->dfs_ancestor_index(),
624                    requested_module->dfs_ancestor_index()));
625     }
626   }
627 
628   Handle<Script> script(module->script(), isolate);
629   Handle<ModuleInfo> module_info(module->info(), isolate);
630 
631   // Resolve imports.
632   Handle<FixedArray> regular_imports(module_info->regular_imports(), isolate);
633   for (int i = 0, n = regular_imports->length(); i < n; ++i) {
634     Handle<ModuleInfoEntry> entry(
635         ModuleInfoEntry::cast(regular_imports->get(i)), isolate);
636     Handle<String> name(String::cast(entry->import_name()), isolate);
637     MessageLocation loc(script, entry->beg_pos(), entry->end_pos());
638     ResolveSet resolve_set(zone);
639     Handle<Cell> cell;
640     if (!ResolveImport(isolate, module, name, entry->module_request(), loc,
641                        true, &resolve_set)
642              .ToHandle(&cell)) {
643       return false;
644     }
645     module->regular_imports()->set(ImportIndex(entry->cell_index()), *cell);
646   }
647 
648   // Resolve indirect exports.
649   Handle<FixedArray> special_exports(module_info->special_exports(), isolate);
650   for (int i = 0, n = special_exports->length(); i < n; ++i) {
651     Handle<ModuleInfoEntry> entry(
652         ModuleInfoEntry::cast(special_exports->get(i)), isolate);
653     Handle<Object> name(entry->export_name(), isolate);
654     if (name->IsUndefined(isolate)) continue;  // Star export.
655     MessageLocation loc(script, entry->beg_pos(), entry->end_pos());
656     ResolveSet resolve_set(zone);
657     if (ResolveExport(isolate, module, Handle<String>(),
658                       Handle<String>::cast(name), loc, true, &resolve_set)
659             .is_null()) {
660       return false;
661     }
662   }
663 
664   return MaybeTransitionComponent(isolate, module, stack, kInstantiated);
665 }
666 
Evaluate(Isolate * isolate,Handle<Module> module)667 MaybeHandle<Object> Module::Evaluate(Isolate* isolate, Handle<Module> module) {
668 #ifdef DEBUG
669   if (FLAG_trace_module_status) {
670     StdoutStream os;
671     os << "Evaluating module ";
672     module->script()->GetNameOrSourceURL()->Print(os);
673 #ifndef OBJECT_PRINT
674     os << "\n";
675 #endif  // OBJECT_PRINT
676   }
677 #endif  // DEBUG
678   if (module->status() == kErrored) {
679     isolate->Throw(module->GetException());
680     return MaybeHandle<Object>();
681   }
682   DCHECK_NE(module->status(), kEvaluating);
683   DCHECK_GE(module->status(), kInstantiated);
684   Zone zone(isolate->allocator(), ZONE_NAME);
685 
686   ZoneForwardList<Handle<Module>> stack(&zone);
687   unsigned dfs_index = 0;
688   Handle<Object> result;
689   if (!Evaluate(isolate, module, &stack, &dfs_index).ToHandle(&result)) {
690     for (auto& descendant : stack) {
691       DCHECK_EQ(descendant->status(), kEvaluating);
692       descendant->RecordError(isolate);
693     }
694     DCHECK_EQ(module->GetException(), isolate->pending_exception());
695     return MaybeHandle<Object>();
696   }
697   DCHECK_EQ(module->status(), kEvaluated);
698   DCHECK(stack.empty());
699   return result;
700 }
701 
Evaluate(Isolate * isolate,Handle<Module> module,ZoneForwardList<Handle<Module>> * stack,unsigned * dfs_index)702 MaybeHandle<Object> Module::Evaluate(Isolate* isolate, Handle<Module> module,
703                                      ZoneForwardList<Handle<Module>>* stack,
704                                      unsigned* dfs_index) {
705   if (module->status() == kErrored) {
706     isolate->Throw(module->GetException());
707     return MaybeHandle<Object>();
708   }
709   if (module->status() >= kEvaluating) {
710     return isolate->factory()->undefined_value();
711   }
712   DCHECK_EQ(module->status(), kInstantiated);
713   STACK_CHECK(isolate, MaybeHandle<Object>());
714 
715   Handle<JSGeneratorObject> generator(JSGeneratorObject::cast(module->code()),
716                                       isolate);
717   module->set_code(
718       generator->function()->shared()->scope_info()->ModuleDescriptorInfo());
719   module->SetStatus(kEvaluating);
720   module->set_dfs_index(*dfs_index);
721   module->set_dfs_ancestor_index(*dfs_index);
722   stack->push_front(module);
723   (*dfs_index)++;
724 
725   // Recursion.
726   Handle<FixedArray> requested_modules(module->requested_modules(), isolate);
727   for (int i = 0, length = requested_modules->length(); i < length; ++i) {
728     Handle<Module> requested_module(Module::cast(requested_modules->get(i)),
729                                     isolate);
730     RETURN_ON_EXCEPTION(
731         isolate, Evaluate(isolate, requested_module, stack, dfs_index), Object);
732 
733     DCHECK_GE(requested_module->status(), kEvaluating);
734     DCHECK_NE(requested_module->status(), kErrored);
735     SLOW_DCHECK(
736         // {requested_module} is evaluating iff it's on the {stack}.
737         (requested_module->status() == kEvaluating) ==
738         std::count_if(stack->begin(), stack->end(), [&](Handle<Module> m) {
739           return *m == *requested_module;
740         }));
741 
742     if (requested_module->status() == kEvaluating) {
743       module->set_dfs_ancestor_index(
744           std::min(module->dfs_ancestor_index(),
745                    requested_module->dfs_ancestor_index()));
746     }
747   }
748 
749   // Evaluation of module body.
750   Handle<JSFunction> resume(
751       isolate->native_context()->generator_next_internal(), isolate);
752   Handle<Object> result;
753   ASSIGN_RETURN_ON_EXCEPTION(
754       isolate, result, Execution::Call(isolate, resume, generator, 0, nullptr),
755       Object);
756   DCHECK(static_cast<JSIteratorResult*>(JSObject::cast(*result))
757              ->done()
758              ->BooleanValue(isolate));
759 
760   CHECK(MaybeTransitionComponent(isolate, module, stack, kEvaluated));
761   return handle(
762       static_cast<JSIteratorResult*>(JSObject::cast(*result))->value(),
763       isolate);
764 }
765 
766 namespace {
767 
FetchStarExports(Isolate * isolate,Handle<Module> module,Zone * zone,UnorderedModuleSet * visited)768 void FetchStarExports(Isolate* isolate, Handle<Module> module, Zone* zone,
769                       UnorderedModuleSet* visited) {
770   DCHECK_GE(module->status(), Module::kInstantiating);
771 
772   if (module->module_namespace()->IsJSModuleNamespace()) return;  // Shortcut.
773 
774   bool cycle = !visited->insert(module).second;
775   if (cycle) return;
776   Handle<ObjectHashTable> exports(module->exports(), isolate);
777   UnorderedStringMap more_exports(zone);
778 
779   // TODO(neis): Only allocate more_exports if there are star exports.
780   // Maybe split special_exports into indirect_exports and star_exports.
781 
782   ReadOnlyRoots roots(isolate);
783   Handle<FixedArray> special_exports(module->info()->special_exports(),
784                                      isolate);
785   for (int i = 0, n = special_exports->length(); i < n; ++i) {
786     Handle<ModuleInfoEntry> entry(
787         ModuleInfoEntry::cast(special_exports->get(i)), isolate);
788     if (!entry->export_name()->IsUndefined(roots)) {
789       continue;  // Indirect export.
790     }
791 
792     Handle<Module> requested_module(
793         Module::cast(module->requested_modules()->get(entry->module_request())),
794         isolate);
795 
796     // Recurse.
797     FetchStarExports(isolate, requested_module, zone, visited);
798 
799     // Collect all of [requested_module]'s exports that must be added to
800     // [module]'s exports (i.e. to [exports]).  We record these in
801     // [more_exports].  Ambiguities (conflicting exports) are marked by mapping
802     // the name to undefined instead of a Cell.
803     Handle<ObjectHashTable> requested_exports(requested_module->exports(),
804                                               isolate);
805     for (int i = 0, n = requested_exports->Capacity(); i < n; ++i) {
806       Object* key;
807       if (!requested_exports->ToKey(roots, i, &key)) continue;
808       Handle<String> name(String::cast(key), isolate);
809 
810       if (name->Equals(roots.default_string())) continue;
811       if (!exports->Lookup(name)->IsTheHole(roots)) continue;
812 
813       Handle<Cell> cell(Cell::cast(requested_exports->ValueAt(i)), isolate);
814       auto insert_result = more_exports.insert(std::make_pair(name, cell));
815       if (!insert_result.second) {
816         auto it = insert_result.first;
817         if (*it->second == *cell || it->second->IsUndefined(roots)) {
818           // We already recorded this mapping before, or the name is already
819           // known to be ambiguous.  In either case, there's nothing to do.
820         } else {
821           DCHECK(it->second->IsCell());
822           // Different star exports provide different cells for this name, hence
823           // mark the name as ambiguous.
824           it->second = roots.undefined_value_handle();
825         }
826       }
827     }
828   }
829 
830   // Copy [more_exports] into [exports].
831   for (const auto& elem : more_exports) {
832     if (elem.second->IsUndefined(isolate)) continue;  // Ambiguous export.
833     DCHECK(!elem.first->Equals(ReadOnlyRoots(isolate).default_string()));
834     DCHECK(elem.second->IsCell());
835     exports = ObjectHashTable::Put(exports, elem.first, elem.second);
836   }
837   module->set_exports(*exports);
838 }
839 
840 }  // anonymous namespace
841 
GetModuleNamespace(Isolate * isolate,Handle<Module> module,int module_request)842 Handle<JSModuleNamespace> Module::GetModuleNamespace(Isolate* isolate,
843                                                      Handle<Module> module,
844                                                      int module_request) {
845   Handle<Module> requested_module(
846       Module::cast(module->requested_modules()->get(module_request)), isolate);
847   return Module::GetModuleNamespace(isolate, requested_module);
848 }
849 
GetModuleNamespace(Isolate * isolate,Handle<Module> module)850 Handle<JSModuleNamespace> Module::GetModuleNamespace(Isolate* isolate,
851                                                      Handle<Module> module) {
852   Handle<HeapObject> object(module->module_namespace(), isolate);
853   ReadOnlyRoots roots(isolate);
854   if (!object->IsUndefined(roots)) {
855     // Namespace object already exists.
856     return Handle<JSModuleNamespace>::cast(object);
857   }
858 
859   // Collect the export names.
860   Zone zone(isolate->allocator(), ZONE_NAME);
861   UnorderedModuleSet visited(&zone);
862   FetchStarExports(isolate, module, &zone, &visited);
863   Handle<ObjectHashTable> exports(module->exports(), isolate);
864   ZoneVector<Handle<String>> names(&zone);
865   names.reserve(exports->NumberOfElements());
866   for (int i = 0, n = exports->Capacity(); i < n; ++i) {
867     Object* key;
868     if (!exports->ToKey(roots, i, &key)) continue;
869     names.push_back(handle(String::cast(key), isolate));
870   }
871   DCHECK_EQ(static_cast<int>(names.size()), exports->NumberOfElements());
872 
873   // Sort them alphabetically.
874   std::sort(names.begin(), names.end(),
875             [&isolate](Handle<String> a, Handle<String> b) {
876               return String::Compare(isolate, a, b) ==
877                      ComparisonResult::kLessThan;
878             });
879 
880   // Create the namespace object (initially empty).
881   Handle<JSModuleNamespace> ns = isolate->factory()->NewJSModuleNamespace();
882   ns->set_module(*module);
883   module->set_module_namespace(*ns);
884 
885   // Create the properties in the namespace object. Transition the object
886   // to dictionary mode so that property addition is faster.
887   PropertyAttributes attr = DONT_DELETE;
888   JSObject::NormalizeProperties(ns, CLEAR_INOBJECT_PROPERTIES,
889                                 static_cast<int>(names.size()),
890                                 "JSModuleNamespace");
891   for (const auto& name : names) {
892     JSObject::SetNormalizedProperty(
893         ns, name, Accessors::MakeModuleNamespaceEntryInfo(isolate, name),
894         PropertyDetails(kAccessor, attr, PropertyCellType::kMutable));
895   }
896   JSObject::PreventExtensions(ns, kThrowOnError).ToChecked();
897 
898   // Optimize the namespace object as a prototype, for two reasons:
899   // - The object's map is guaranteed not to be shared. ICs rely on this.
900   // - We can store a pointer from the map back to the namespace object.
901   //   Turbofan can use this for inlining the access.
902   JSObject::OptimizeAsPrototype(ns);
903 
904   Handle<PrototypeInfo> proto_info =
905       Map::GetOrCreatePrototypeInfo(Handle<JSObject>::cast(ns), isolate);
906   proto_info->set_module_namespace(*ns);
907   return ns;
908 }
909 
GetExport(Isolate * isolate,Handle<String> name)910 MaybeHandle<Object> JSModuleNamespace::GetExport(Isolate* isolate,
911                                                  Handle<String> name) {
912   Handle<Object> object(module()->exports()->Lookup(name), isolate);
913   if (object->IsTheHole(isolate)) {
914     return isolate->factory()->undefined_value();
915   }
916 
917   Handle<Object> value(Handle<Cell>::cast(object)->value(), isolate);
918   if (value->IsTheHole(isolate)) {
919     THROW_NEW_ERROR(
920         isolate, NewReferenceError(MessageTemplate::kNotDefined, name), Object);
921   }
922 
923   return value;
924 }
925 
GetPropertyAttributes(LookupIterator * it)926 Maybe<PropertyAttributes> JSModuleNamespace::GetPropertyAttributes(
927     LookupIterator* it) {
928   Handle<JSModuleNamespace> object = it->GetHolder<JSModuleNamespace>();
929   Handle<String> name = Handle<String>::cast(it->GetName());
930   DCHECK_EQ(it->state(), LookupIterator::ACCESSOR);
931 
932   Isolate* isolate = it->isolate();
933 
934   Handle<Object> lookup(object->module()->exports()->Lookup(name), isolate);
935   if (lookup->IsTheHole(isolate)) {
936     return Just(ABSENT);
937   }
938 
939   Handle<Object> value(Handle<Cell>::cast(lookup)->value(), isolate);
940   if (value->IsTheHole(isolate)) {
941     isolate->Throw(*isolate->factory()->NewReferenceError(
942         MessageTemplate::kNotDefined, name));
943     return Nothing<PropertyAttributes>();
944   }
945 
946   return Just(it->property_attributes());
947 }
948 
949 }  // namespace internal
950 }  // namespace v8
951