// Copyright 2017 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_TORQUE_GLOBAL_CONTEXT_H_ #define V8_TORQUE_GLOBAL_CONTEXT_H_ #include "src/torque/declarable.h" #include "src/torque/declarations.h" #include "src/torque/scope.h" #include "src/torque/type-oracle.h" namespace v8 { namespace internal { namespace torque { class GlobalContext; class Scope; class TypeOracle; class Builtin; class Label; class Module { public: explicit Module(const std::string& name, bool is_default) : name_(name), is_default_(is_default) {} const std::string& name() const { return name_; } bool IsDefault() const { return is_default_; } std::ostream& source_stream() { return source_stream_; } std::ostream& header_stream() { return header_stream_; } std::string source() { return source_stream_.str(); } std::string header() { return header_stream_.str(); } private: std::string name_; bool is_default_; std::stringstream header_stream_; std::stringstream source_stream_; }; class GlobalContext { public: explicit GlobalContext(Ast ast) : verbose_(false), next_label_number_(0), default_module_(GetModule("base", true)), ast_(std::move(ast)) {} Module* GetDefaultModule() { return default_module_; } Module* GetModule(const std::string& name, bool is_default = false) { auto i = modules_.find(name); if (i != modules_.end()) { return i->second.get(); } Module* module = new Module(name, is_default); modules_[name] = std::unique_ptr(module); return module; } int GetNextLabelNumber() { return next_label_number_++; } const std::map>& GetModules() const { return modules_; } void SetVerbose() { verbose_ = true; } bool verbose() const { return verbose_; } void AddControlSplitChangedVariables(const AstNode* node, const TypeVector& specialization_types, const std::set& vars) { auto key = std::make_pair(node, specialization_types); control_split_changed_variables_[key] = vars; } const std::set& GetControlSplitChangedVariables( const AstNode* node, const TypeVector& specialization_types) { auto key = std::make_pair(node, specialization_types); assert(control_split_changed_variables_.find(key) != control_split_changed_variables_.end()); return control_split_changed_variables_.find(key)->second; } void MarkVariableChanged(const AstNode* node, const TypeVector& specialization_types, Variable* var) { auto key = std::make_pair(node, specialization_types); control_split_changed_variables_[key].insert(var); } friend class CurrentCallableActivator; friend class BreakContinueActivator; Callable* GetCurrentCallable() const { return current_callable_; } Label* GetCurrentBreak() const { return break_continue_stack_.back().first; } Label* GetCurrentContinue() const { return break_continue_stack_.back().second; } Declarations* declarations() { return &declarations_; } Ast* ast() { return &ast_; } private: bool verbose_; int next_label_number_; Declarations declarations_; Callable* current_callable_; std::vector> break_continue_stack_; std::map> modules_; Module* default_module_; std::map, std::set> control_split_changed_variables_; Ast ast_; }; class CurrentCallableActivator { public: CurrentCallableActivator(GlobalContext& context, Callable* callable, CallableNode* decl) : context_(context), scope_activator_(context.declarations(), decl) { remembered_callable_ = context_.current_callable_; context_.current_callable_ = callable; } ~CurrentCallableActivator() { context_.current_callable_ = remembered_callable_; } private: GlobalContext& context_; Callable* remembered_callable_; Declarations::NodeScopeActivator scope_activator_; }; class BreakContinueActivator { public: BreakContinueActivator(GlobalContext& context, Label* break_label, Label* continue_label) : context_(context) { context_.break_continue_stack_.push_back({break_label, continue_label}); } ~BreakContinueActivator() { context_.break_continue_stack_.pop_back(); } private: GlobalContext& context_; }; } // namespace torque } // namespace internal } // namespace v8 #endif // V8_TORQUE_GLOBAL_CONTEXT_H_