#ifndef SRC_NODE_BUILTINS_H_ #define SRC_NODE_BUILTINS_H_ #if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS #include #include #include #include #include #include #include #include "node_mutex.h" #include "node_threadsafe_cow.h" #include "node_union_bytes.h" #include "v8.h" // Forward declare test fixture for `friend` declaration. class PerProcessTest; namespace node { class SnapshotBuilder; class ExternalReferenceRegistry; class Realm; namespace builtins { using BuiltinSourceMap = std::map; using BuiltinCodeCacheMap = std::unordered_map>; struct CodeCacheInfo { std::string id; std::vector data; }; // Handles compilation and caching of built-in JavaScript modules and // bootstrap scripts, whose source are bundled into the binary as static data. class NODE_EXTERN_PRIVATE BuiltinLoader { public: BuiltinLoader(); BuiltinLoader(const BuiltinLoader&) = delete; BuiltinLoader& operator=(const BuiltinLoader&) = delete; static void RegisterExternalReferences(ExternalReferenceRegistry* registry); static void CreatePerIsolateProperties( IsolateData* isolate_data, v8::Local target); static void CreatePerContextProperties(v8::Local target, v8::Local unused, v8::Local context, void* priv); // The parameters used to compile the scripts are detected based on // the pattern of the id. v8::MaybeLocal LookupAndCompile(v8::Local context, const char* id, Realm* optional_realm); v8::MaybeLocal CompileAndCall(v8::Local context, const char* id, int argc, v8::Local argv[], Realm* optional_realm); v8::MaybeLocal CompileAndCall(v8::Local context, const char* id, Realm* realm); v8::Local GetSourceObject(v8::Local context); // Returns config.gypi as a JSON string v8::Local GetConfigString(v8::Isolate* isolate); bool Exists(const char* id); bool Add(const char* id, const UnionBytes& source); bool Add(const char* id, std::string_view utf8source); bool CompileAllBuiltins(v8::Local context); void RefreshCodeCache(const std::vector& in); void CopyCodeCache(std::vector* out) const; void CopySourceAndCodeCacheReferenceFrom(const BuiltinLoader* other); private: // Only allow access from friends. friend class CodeCacheBuilder; // Generated by tools/js2c.py as node_javascript.cc void LoadJavaScriptSource(); // Loads data into source_ UnionBytes GetConfig(); // Return data for config.gypi std::vector GetBuiltinIds() const; struct BuiltinCategories { std::set can_be_required; std::set cannot_be_required; }; // This method builds `BuiltinCategories` from scratch every time, // and is therefore somewhat expensive, but also currently only being // used for testing, so that should not be an issue. BuiltinCategories GetBuiltinCategories() const; const v8::ScriptCompiler::CachedData* GetCodeCache(const char* id) const; enum class Result { kWithCache, kWithoutCache }; v8::MaybeLocal LoadBuiltinSource(v8::Isolate* isolate, const char* id) const; // If an exception is encountered (e.g. source code contains // syntax error), the returned value is empty. v8::MaybeLocal LookupAndCompileInternal( v8::Local context, const char* id, std::vector>* parameters, Result* result); static void RecordResult(const char* id, BuiltinLoader::Result result, Realm* realm); static void GetBuiltinCategories( v8::Local property, const v8::PropertyCallbackInfo& info); static void GetCacheUsage(const v8::FunctionCallbackInfo& args); // Passing ids of built-in source code into JS land as // internalBinding('builtins').builtinIds static void BuiltinIdsGetter(v8::Local property, const v8::PropertyCallbackInfo& info); // Passing config.gypi into JS land as internalBinding('builtins').config static void ConfigStringGetter( v8::Local property, const v8::PropertyCallbackInfo& info); // Compile a specific built-in as a function static void CompileFunction(const v8::FunctionCallbackInfo& args); static void HasCachedBuiltins( const v8::FunctionCallbackInfo& args); void AddExternalizedBuiltin(const char* id, const char* filename); ThreadsafeCopyOnWrite source_; const UnionBytes config_; struct BuiltinCodeCache { RwLock mutex; BuiltinCodeCacheMap map; bool has_code_cache = false; }; std::shared_ptr code_cache_; friend class ::PerProcessTest; }; } // namespace builtins } // namespace node #endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS #endif // SRC_NODE_BUILTINS_H_