Skip to content

Commit 43fb6ff

Browse files
committed
build: enable loading internal modules from disk
PR-URL: #31321 Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: Tobias Nießen <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Rich Trott <[email protected]> Reviewed-By: Colin Ihrig <[email protected]> Reviewed-By: Matteo Collina <[email protected]>
1 parent a171314 commit 43fb6ff

5 files changed

+85
-9
lines changed

configure.py

+16-6
Original file line numberDiff line numberDiff line change
@@ -609,6 +609,12 @@
609609
default=False,
610610
help='compile V8 with minimal optimizations and with runtime checks')
611611

612+
parser.add_option('--node-builtin-modules-path',
613+
action='store',
614+
dest='node_builtin_modules_path',
615+
default=False,
616+
help='node will load builtin modules from disk instead of from binary')
617+
612618
# Create compile_commands.json in out/Debug and out/Release.
613619
parser.add_option('-C',
614620
action='store_true',
@@ -992,18 +998,18 @@ def configure_node(o):
992998

993999
o['variables']['want_separate_host_toolset'] = int(cross_compiling)
9941000

995-
if not options.without_node_snapshot:
1001+
if options.without_node_snapshot or options.node_builtin_modules_path:
1002+
o['variables']['node_use_node_snapshot'] = 'false'
1003+
else:
9961004
o['variables']['node_use_node_snapshot'] = b(
9971005
not cross_compiling and not options.shared)
998-
else:
999-
o['variables']['node_use_node_snapshot'] = 'false'
10001006

1001-
if not options.without_node_code_cache:
1007+
if options.without_node_code_cache or options.node_builtin_modules_path:
1008+
o['variables']['node_use_node_code_cache'] = 'false'
1009+
else:
10021010
# TODO(refack): fix this when implementing embedded code-cache when cross-compiling.
10031011
o['variables']['node_use_node_code_cache'] = b(
10041012
not cross_compiling and not options.shared)
1005-
else:
1006-
o['variables']['node_use_node_code_cache'] = 'false'
10071013

10081014
if target_arch == 'arm':
10091015
configure_arm(o)
@@ -1145,6 +1151,10 @@ def configure_node(o):
11451151
else:
11461152
o['variables']['node_target_type'] = 'executable'
11471153

1154+
if options.node_builtin_modules_path:
1155+
print('Warning! Loading builtin modules from disk is for development')
1156+
o['variables']['node_builtin_modules_path'] = options.node_builtin_modules_path
1157+
11481158
def configure_napi(output):
11491159
version = getnapibuildversion.get_napi_version()
11501160
output['variables']['napi_build_version'] = version

node.gyp

+4
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
'node_core_target_name%': 'node',
2626
'node_lib_target_name%': 'libnode',
2727
'node_intermediate_lib_type%': 'static_library',
28+
'node_builtin_modules_path%': '',
2829
'library_files': [
2930
'lib/internal/bootstrap/environment.js',
3031
'lib/internal/bootstrap/loaders.js',
@@ -709,6 +710,9 @@
709710
'msvs_disabled_warnings!': [4244],
710711

711712
'conditions': [
713+
[ 'node_builtin_modules_path!=""', {
714+
'defines': [ 'NODE_BUILTIN_MODULES_PATH="<(node_builtin_modules_path)"' ]
715+
}],
712716
[ 'node_shared=="true"', {
713717
'sources': [
714718
'src/node_snapshot_stub.cc',

src/node.cc

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
#include "env-inl.h"
2828
#include "memory_tracker-inl.h"
2929
#include "node_binding.h"
30+
#include "node_errors.h"
3031
#include "node_internals.h"
3132
#include "node_main_instance.h"
3233
#include "node_metadata.h"

src/node_native_module.cc

+62-3
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,64 @@ MaybeLocal<Function> NativeModuleLoader::CompileAsModule(
174174
return LookupAndCompile(context, id, &parameters, result);
175175
}
176176

177+
#ifdef NODE_BUILTIN_MODULES_PATH
178+
static std::string OnDiskFileName(const char* id) {
179+
std::string filename = NODE_BUILTIN_MODULES_PATH;
180+
filename += "/";
181+
182+
if (strncmp(id, "internal/deps", strlen("internal/deps")) == 0) {
183+
id += strlen("internal/");
184+
} else {
185+
filename += "lib/";
186+
}
187+
filename += id;
188+
filename += ".js";
189+
190+
return filename;
191+
}
192+
#endif // NODE_BUILTIN_MODULES_PATH
193+
194+
MaybeLocal<String> NativeModuleLoader::LoadBuiltinModuleSource(Isolate* isolate,
195+
const char* id) {
196+
#ifdef NODE_BUILTIN_MODULES_PATH
197+
std::string filename = OnDiskFileName(id);
198+
199+
uv_fs_t req;
200+
uv_file file =
201+
uv_fs_open(nullptr, &req, filename.c_str(), O_RDONLY, 0, nullptr);
202+
CHECK_GE(req.result, 0);
203+
uv_fs_req_cleanup(&req);
204+
205+
std::shared_ptr<void> defer_close(nullptr, [file](...) {
206+
uv_fs_t close_req;
207+
CHECK_EQ(0, uv_fs_close(nullptr, &close_req, file, nullptr));
208+
uv_fs_req_cleanup(&close_req);
209+
});
210+
211+
std::string contents;
212+
char buffer[4096];
213+
uv_buf_t buf = uv_buf_init(buffer, sizeof(buffer));
214+
215+
while (true) {
216+
const int r =
217+
uv_fs_read(nullptr, &req, file, &buf, 1, contents.length(), nullptr);
218+
CHECK_GE(req.result, 0);
219+
uv_fs_req_cleanup(&req);
220+
if (r <= 0) {
221+
break;
222+
}
223+
contents.append(buf.base, r);
224+
}
225+
226+
return String::NewFromUtf8(
227+
isolate, contents.c_str(), v8::NewStringType::kNormal, contents.length());
228+
#else
229+
const auto source_it = source_.find(id);
230+
CHECK_NE(source_it, source_.end());
231+
return source_it->second.ToStringChecked(isolate);
232+
#endif // NODE_BUILTIN_MODULES_PATH
233+
}
234+
177235
// Returns Local<Function> of the compiled module if return_code_cache
178236
// is false (we are only compiling the function).
179237
// Otherwise return a Local<Object> containing the cache.
@@ -185,9 +243,10 @@ MaybeLocal<Function> NativeModuleLoader::LookupAndCompile(
185243
Isolate* isolate = context->GetIsolate();
186244
EscapableHandleScope scope(isolate);
187245

188-
const auto source_it = source_.find(id);
189-
CHECK_NE(source_it, source_.end());
190-
Local<String> source = source_it->second.ToStringChecked(isolate);
246+
Local<String> source;
247+
if (!LoadBuiltinModuleSource(isolate, id).ToLocal(&source)) {
248+
return {};
249+
}
191250

192251
std::string filename_s = id + std::string(".js");
193252
Local<String> filename =

src/node_native_module.h

+2
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,8 @@ class NativeModuleLoader {
6666
NativeModuleCacheMap* code_cache();
6767
v8::ScriptCompiler::CachedData* GetCodeCache(const char* id) const;
6868
enum class Result { kWithCache, kWithoutCache };
69+
v8::MaybeLocal<v8::String> LoadBuiltinModuleSource(v8::Isolate* isolate,
70+
const char* id);
6971
// If an exception is encountered (e.g. source code contains
7072
// syntax error), the returned value is empty.
7173
v8::MaybeLocal<v8::Function> LookupAndCompile(

0 commit comments

Comments
 (0)