Skip to content

Commit 7778c03

Browse files
committedNov 19, 2018
src: use STL containers instead of v8 values for static module data
Instead of putting the source code and the cache in v8::Objects, put them in per-process std::maps. This has the following benefits: - It's slightly lighter in weight compared to storing things on the v8 heap. Also it may be slightly faster since the preivous v8::Object is already in dictionary mode - though the difference is very small given the number of native modules is limited. - The source and code cache generation templates are now much simpler since they just initialize static arrays and manipulate STL constructs. - The static native module data can be accessed independently of any Environment or Isolate, and it's easy to look them up from the C++'s side. - It's now impossible to mutate the source code used to compile native modules from the JS land since it's completely separate from the v8 heap. We can still get the constant strings from process.binding('natives') but that's all. A few drive-by fixes: - Remove DecorateErrorStack in LookupAndCompile - We don't need to capture the exception to decorate when we encounter errors during native module compilation, as those errors should be syntax errors and v8 is able to decorate them well. We use CompileFunctionInContext so there is no need to worry about wrappers either. - The code cache could be rejected when node is started with v8 flags. Instead of aborting in that case, simply keep a record in the native_module_without_cache set. - Refactor js2c.py a bit, reduce code duplication and inline Render() to make the one-byte/two-byte special treatment easier to read. PR-URL: nodejs#24384 Fixes: https://github.com/Remove Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Franziska Hinkelmann <franziska.hinkelmann@gmail.com> Reviewed-By: Tiancheng "Timothy" Gu <timothygu99@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com>
1 parent 092ab7a commit 7778c03

16 files changed

+457
-419
lines changed
 

‎lib/internal/bootstrap/cache.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -9,11 +9,12 @@ const {
99
NativeModule
1010
} = require('internal/bootstrap/loaders');
1111
const {
12-
source,
12+
getSource,
1313
compileCodeCache
1414
} = internalBinding('native_module');
1515
const { hasTracing } = process.binding('config');
1616

17+
const source = getSource();
1718
const depsModule = Object.keys(source).filter(
1819
(key) => NativeModule.isDepsModule(key) || key.startsWith('internal/deps')
1920
);

‎node.gyp

+1-2
Original file line numberDiff line numberDiff line change
@@ -410,15 +410,13 @@
410410
'src/node_api.h',
411411
'src/node_api_types.h',
412412
'src/node_buffer.h',
413-
'src/node_code_cache.h',
414413
'src/node_constants.h',
415414
'src/node_contextify.h',
416415
'src/node_errors.h',
417416
'src/node_file.h',
418417
'src/node_http2.h',
419418
'src/node_http2_state.h',
420419
'src/node_internals.h',
421-
'src/node_javascript.h',
422420
'src/node_messaging.h',
423421
'src/node_mutex.h',
424422
'src/node_native_module.h',
@@ -430,6 +428,7 @@
430428
'src/node_persistent.h',
431429
'src/node_platform.h',
432430
'src/node_root_certs.h',
431+
'src/node_union_bytes.h',
433432
'src/node_version.h',
434433
'src/node_watchdog.h',
435434
'src/node_revert.h',

‎src/env.cc

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,14 @@
1-
#include "node_internals.h"
21
#include "async_wrap.h"
3-
#include "v8-profiler.h"
42
#include "node_buffer.h"
5-
#include "node_platform.h"
6-
#include "node_file.h"
73
#include "node_context_data.h"
4+
#include "node_file.h"
5+
#include "node_internals.h"
6+
#include "node_native_module.h"
7+
#include "node_platform.h"
88
#include "node_worker.h"
99
#include "tracing/agent.h"
1010
#include "tracing/traced_value.h"
11+
#include "v8-profiler.h"
1112

1213
#include <stdio.h>
1314
#include <algorithm>

‎src/env.h

+6-9
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,13 @@
2929
#include "inspector_agent.h"
3030
#endif
3131
#include "handle_wrap.h"
32+
#include "node.h"
33+
#include "node_http2_state.h"
34+
#include "node_options.h"
3235
#include "req_wrap.h"
3336
#include "util.h"
3437
#include "uv.h"
3538
#include "v8.h"
36-
#include "node.h"
37-
#include "node_options.h"
38-
#include "node_http2_state.h"
3939

4040
#include <list>
4141
#include <stdint.h>
@@ -347,12 +347,6 @@ constexpr size_t kFsStatsBufferLength = kFsStatsFieldsNumber * 2;
347347
V(libuv_stream_wrap_ctor_template, v8::FunctionTemplate) \
348348
V(message_port, v8::Object) \
349349
V(message_port_constructor_template, v8::FunctionTemplate) \
350-
V(native_modules_code_cache, v8::Object) \
351-
V(native_modules_code_cache_hash, v8::Object) \
352-
V(native_modules_source, v8::Object) \
353-
V(native_modules_source_hash, v8::Object) \
354-
V(native_modules_with_cache, v8::Set) \
355-
V(native_modules_without_cache, v8::Set) \
356350
V(performance_entry_callback, v8::Function) \
357351
V(performance_entry_template, v8::Function) \
358352
V(pipe_constructor_template, v8::FunctionTemplate) \
@@ -684,6 +678,9 @@ class Environment {
684678
// List of id's that have been destroyed and need the destroy() cb called.
685679
inline std::vector<double>* destroy_async_id_list();
686680

681+
std::set<std::string> native_modules_with_cache;
682+
std::set<std::string> native_modules_without_cache;
683+
687684
std::unordered_multimap<int, loader::ModuleWrap*> hash_to_module_map;
688685
std::unordered_map<uint32_t, loader::ModuleWrap*> id_to_module_map;
689686
std::unordered_map<uint32_t, contextify::ContextifyScript*>

‎src/node.cc

+19-17
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424
#include "node_context_data.h"
2525
#include "node_errors.h"
2626
#include "node_internals.h"
27-
#include "node_javascript.h"
2827
#include "node_native_module.h"
2928
#include "node_perf.h"
3029
#include "node_platform.h"
@@ -130,7 +129,7 @@ typedef int mode_t;
130129

131130
namespace node {
132131

133-
using native_module::NativeModule;
132+
using native_module::NativeModuleLoader;
134133
using options_parser::kAllowedInEnvironment;
135134
using options_parser::kDisallowedInEnvironment;
136135
using v8::Array;
@@ -212,7 +211,7 @@ double prog_start_time;
212211
Mutex per_process_opts_mutex;
213212
std::shared_ptr<PerProcessOptions> per_process_opts {
214213
new PerProcessOptions() };
215-
214+
NativeModuleLoader per_process_loader;
216215
static Mutex node_isolate_mutex;
217216
static Isolate* node_isolate;
218217

@@ -1243,8 +1242,7 @@ static void GetInternalBinding(const FunctionCallbackInfo<Value>& args) {
12431242
Null(env->isolate())).FromJust());
12441243
DefineConstants(env->isolate(), exports);
12451244
} else if (!strcmp(*module_v, "natives")) {
1246-
exports = Object::New(env->isolate());
1247-
NativeModule::GetNatives(env, exports);
1245+
exports = per_process_loader.GetSourceObject(env->context());
12481246
} else {
12491247
return ThrowIfNoSuchModule(env, *module_v);
12501248
}
@@ -1780,18 +1778,24 @@ void LoadEnvironment(Environment* env) {
17801778

17811779
// The bootstrapper scripts are lib/internal/bootstrap/loaders.js and
17821780
// lib/internal/bootstrap/node.js, each included as a static C string
1783-
// defined in node_javascript.h, generated in node_javascript.cc by
1784-
// node_js2c.
1781+
// generated in node_javascript.cc by node_js2c.
17851782

1786-
// TODO(joyeecheung): use NativeModule::Compile
1783+
// TODO(joyeecheung): use NativeModuleLoader::Compile
1784+
// We duplicate the string literals here since once we refactor the bootstrap
1785+
// compilation out to NativeModuleLoader none of this is going to matter
1786+
Isolate* isolate = env->isolate();
17871787
Local<String> loaders_name =
1788-
FIXED_ONE_BYTE_STRING(env->isolate(), "internal/bootstrap/loaders.js");
1788+
FIXED_ONE_BYTE_STRING(isolate, "internal/bootstrap/loaders.js");
1789+
Local<String> loaders_source =
1790+
per_process_loader.GetSource(isolate, "internal/bootstrap/loaders");
17891791
MaybeLocal<Function> loaders_bootstrapper =
1790-
GetBootstrapper(env, LoadersBootstrapperSource(env), loaders_name);
1792+
GetBootstrapper(env, loaders_source, loaders_name);
17911793
Local<String> node_name =
1792-
FIXED_ONE_BYTE_STRING(env->isolate(), "internal/bootstrap/node.js");
1794+
FIXED_ONE_BYTE_STRING(isolate, "internal/bootstrap/node.js");
1795+
Local<String> node_source =
1796+
per_process_loader.GetSource(isolate, "internal/bootstrap/node");
17931797
MaybeLocal<Function> node_bootstrapper =
1794-
GetBootstrapper(env, NodeBootstrapperSource(env), node_name);
1798+
GetBootstrapper(env, node_source, node_name);
17951799

17961800
if (loaders_bootstrapper.IsEmpty() || node_bootstrapper.IsEmpty()) {
17971801
// Execution was interrupted.
@@ -1843,8 +1847,6 @@ void LoadEnvironment(Environment* env) {
18431847
env->options()->debug_options->break_node_first_line)
18441848
};
18451849

1846-
NativeModule::LoadBindings(env);
1847-
18481850
// Bootstrap internal loaders
18491851
Local<Value> bootstrapped_loaders;
18501852
if (!ExecuteBootstrapper(env, loaders_bootstrapper.ToLocalChecked(),
@@ -2485,7 +2487,6 @@ void FreePlatform(MultiIsolatePlatform* platform) {
24852487
delete platform;
24862488
}
24872489

2488-
24892490
Local<Context> NewContext(Isolate* isolate,
24902491
Local<ObjectTemplate> object_template) {
24912492
auto context = Context::New(isolate, nullptr, object_template);
@@ -2499,8 +2500,9 @@ Local<Context> NewContext(Isolate* isolate,
24992500
// Run lib/internal/per_context.js
25002501
Context::Scope context_scope(context);
25012502

2502-
// TODO(joyeecheung): use NativeModule::Compile
2503-
Local<String> per_context = NodePerContextSource(isolate);
2503+
// TODO(joyeecheung): use NativeModuleLoader::Compile
2504+
Local<String> per_context =
2505+
per_process_loader.GetSource(isolate, "internal/per_context");
25042506
ScriptCompiler::Source per_context_src(per_context, nullptr);
25052507
Local<Script> s = ScriptCompiler::Compile(
25062508
context,

‎src/node_code_cache.h

-19
This file was deleted.

‎src/node_code_cache_stub.cc

+9-13
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,19 @@
11

2-
#include "node_code_cache.h"
2+
#include "node_native_module.h"
33

44
// This is supposed to be generated by tools/generate_code_cache.js
55
// The stub here is used when configure is run without `--code-cache-path`
66

77
namespace node {
8+
namespace native_module {
89

9-
const bool native_module_has_code_cache = false;
10+
// The generated source code would insert <std::string, UnionString> pairs
11+
// into native_module_loader.code_cache_.
12+
void NativeModuleLoader::LoadCodeCache() {}
1013

11-
void DefineCodeCache(Environment* env, v8::Local<v8::Object> target) {
12-
// When we do not produce code cache for builtin modules,
13-
// `internalBinding('code_cache')` returns an empty object
14-
// (here as `target`) so this is a noop.
15-
}
16-
17-
void DefineCodeCacheHash(Environment* env, v8::Local<v8::Object> target) {
18-
// When we do not produce code cache for builtin modules,
19-
// `internalBinding('code_cache_hash')` returns an empty object
20-
// (here as `target`) so this is a noop.
21-
}
14+
// The generated source code would instert <std::string, std::string> pairs
15+
// into native_module_loader.code_cache_hash_.
16+
void NativeModuleLoader::LoadCodeCacheHash() {}
2217

18+
} // namespace native_module
2319
} // namespace node

‎src/node_internals.h

+5
Original file line numberDiff line numberDiff line change
@@ -171,6 +171,10 @@ struct sockaddr;
171171

172172
namespace node {
173173

174+
namespace native_module {
175+
class NativeModuleLoader;
176+
}
177+
174178
extern Mutex process_mutex;
175179
extern Mutex environ_mutex;
176180

@@ -179,6 +183,7 @@ extern bool v8_initialized;
179183

180184
extern Mutex per_process_opts_mutex;
181185
extern std::shared_ptr<PerProcessOptions> per_process_opts;
186+
extern native_module::NativeModuleLoader per_process_loader;
182187

183188
// Forward declaration
184189
class Environment;

‎src/node_javascript.h

-41
This file was deleted.

‎src/node_native_module.cc

+162-162
Large diffs are not rendered by default.

‎src/node_native_module.h

+49-15
Original file line numberDiff line numberDiff line change
@@ -3,34 +3,68 @@
33

44
#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
55

6-
#include "node_internals.h"
6+
#include <map>
7+
#include <set>
8+
#include <string>
9+
#include "env.h"
10+
#include "node_union_bytes.h"
11+
#include "v8.h"
712

813
namespace node {
914
namespace native_module {
1015

11-
// The native (C++) side of the native module compilation.
16+
using NativeModuleRecordMap = std::map<std::string, UnionBytes>;
17+
using NativeModuleHashMap = std::map<std::string, std::string>;
1218

13-
class NativeModule {
19+
// The native (C++) side of the native module compilation.
20+
// This class should not depend on Environment
21+
class NativeModuleLoader {
1422
public:
15-
// For legacy process.binding('natives') which is mutable
16-
static void GetNatives(Environment* env, v8::Local<v8::Object> exports);
17-
// Loads the static JavaScript source code and the cache into Environment
18-
static void LoadBindings(Environment* env);
23+
NativeModuleLoader();
24+
static void Initialize(v8::Local<v8::Object> target,
25+
v8::Local<v8::Value> unused,
26+
v8::Local<v8::Context> context);
27+
v8::Local<v8::Object> GetSourceObject(v8::Local<v8::Context> context) const;
28+
v8::Local<v8::String> GetSource(v8::Isolate* isolate, const char* id) const;
29+
30+
private:
31+
static void GetCacheUsage(const v8::FunctionCallbackInfo<v8::Value>& args);
32+
// For legacy process.binding('natives') which is mutable, and for
33+
// internalBinding('native_module').source for internal use
34+
static void GetSourceObject(const v8::FunctionCallbackInfo<v8::Value>& args);
1935
// Compile code cache for a specific native module
2036
static void CompileCodeCache(const v8::FunctionCallbackInfo<v8::Value>& args);
2137
// Compile a specific native module as a function
2238
static void CompileFunction(const v8::FunctionCallbackInfo<v8::Value>& args);
2339

24-
private:
40+
// Generated by tools/js2c.py as node_javascript.cc
41+
void LoadJavaScriptSource(); // Loads data into source_
42+
void LoadJavaScriptHash(); // Loads data into source_hash_
43+
44+
// Generated by tools/generate_code_cache.js as node_code_cache.cc when
45+
// the build is configured with --code-cache-path=.... They are noops
46+
// in node_code_cache_stub.cc
47+
void LoadCodeCache(); // Loads data into code_cache_
48+
void LoadCodeCacheHash(); // Loads data into code_cache_hash_
49+
50+
v8::ScriptCompiler::CachedData* GetCachedData(const char* id) const;
2551
static v8::Local<v8::Value> CompileAsModule(Environment* env,
26-
v8::Local<v8::String> id,
52+
const char* id,
2753
bool produce_code_cache);
28-
// TODO(joyeecheung): make this public and reuse it to compile bootstrappers
29-
static v8::Local<v8::Value> Compile(Environment* env,
30-
v8::Local<v8::String> id,
31-
v8::Local<v8::String> parameters[],
32-
size_t parameters_count,
33-
bool produce_code_cache);
54+
// TODO(joyeecheung): make this public and reuse it to compile bootstrappers.
55+
// For bootstrappers optional_env may be a nullptr.
56+
// This method magically knows what parameter it should pass to
57+
// the function to be compiled.
58+
v8::Local<v8::Value> LookupAndCompile(v8::Local<v8::Context> context,
59+
const char* id,
60+
bool produce_code_cache,
61+
Environment* optional_env);
62+
63+
bool has_code_cache_ = false;
64+
NativeModuleRecordMap source_;
65+
NativeModuleRecordMap code_cache_;
66+
NativeModuleHashMap source_hash_;
67+
NativeModuleHashMap code_cache_hash_;
3468
};
3569

3670
} // namespace native_module

‎src/node_union_bytes.h

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
2+
#ifndef SRC_NODE_UNION_BYTES_H_
3+
#define SRC_NODE_UNION_BYTES_H_
4+
5+
#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
6+
7+
// A union of const uint8_t* or const uint16_t* data that can be
8+
// turned into external v8::String when given an isolate.
9+
10+
#include "env.h"
11+
#include "v8.h"
12+
13+
namespace node {
14+
15+
class NonOwningExternalOneByteResource
16+
: public v8::String::ExternalOneByteStringResource {
17+
public:
18+
explicit NonOwningExternalOneByteResource(const uint8_t* data, size_t length)
19+
: data_(data), length_(length) {}
20+
~NonOwningExternalOneByteResource() override = default;
21+
22+
const char* data() const override {
23+
return reinterpret_cast<const char*>(data_);
24+
}
25+
size_t length() const override { return length_; }
26+
27+
private:
28+
const uint8_t* data_;
29+
size_t length_;
30+
DISALLOW_COPY_AND_ASSIGN(NonOwningExternalOneByteResource);
31+
};
32+
33+
class NonOwningExternalTwoByteResource
34+
: public v8::String::ExternalStringResource {
35+
public:
36+
explicit NonOwningExternalTwoByteResource(const uint16_t* data, size_t length)
37+
: data_(data), length_(length) {}
38+
~NonOwningExternalTwoByteResource() override = default;
39+
40+
const uint16_t* data() const override { return data_; }
41+
size_t length() const override { return length_; }
42+
43+
private:
44+
const uint16_t* data_;
45+
size_t length_;
46+
DISALLOW_COPY_AND_ASSIGN(NonOwningExternalTwoByteResource);
47+
};
48+
49+
// Similar to a v8::String, but it's independent from Isolates
50+
// and can be materialized in Isolates as external Strings
51+
// via ToStringChecked. The data pointers are owned by the caller.
52+
class UnionBytes {
53+
public:
54+
UnionBytes(const uint16_t* data, size_t length)
55+
: is_one_byte_(false), two_bytes_(data), length_(length) {}
56+
UnionBytes(const uint8_t* data, size_t length)
57+
: is_one_byte_(true), one_bytes_(data), length_(length) {}
58+
bool is_one_byte() const { return is_one_byte_; }
59+
const uint16_t* two_bytes_data() const {
60+
CHECK(!is_one_byte_);
61+
return two_bytes_;
62+
}
63+
const uint8_t* one_bytes_data() const {
64+
CHECK(is_one_byte_);
65+
return one_bytes_;
66+
}
67+
v8::Local<v8::String> ToStringChecked(v8::Isolate* isolate) const {
68+
if (is_one_byte_) {
69+
NonOwningExternalOneByteResource* source =
70+
new NonOwningExternalOneByteResource(one_bytes_, length_);
71+
return v8::String::NewExternalOneByte(isolate, source).ToLocalChecked();
72+
} else {
73+
NonOwningExternalTwoByteResource* source =
74+
new NonOwningExternalTwoByteResource(two_bytes_, length_);
75+
return v8::String::NewExternalTwoByte(isolate, source).ToLocalChecked();
76+
}
77+
}
78+
size_t length() { return length_; }
79+
80+
private:
81+
bool is_one_byte_;
82+
union {
83+
const uint8_t* one_bytes_;
84+
const uint16_t* two_bytes_;
85+
};
86+
size_t length_;
87+
};
88+
89+
} // namespace node
90+
91+
#endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS
92+
93+
#endif // SRC_NODE_UNION_BYTES_H_

‎test/code-cache/test-code-cache-generator.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -30,10 +30,10 @@ if (child.status !== 0) {
3030
}
3131

3232
// Verifies that:
33-
// - node::DefineCodeCache()
34-
// - node::DefineCodeCacheHash()
33+
// - node::LoadCodeCache()
34+
// - node::LoadCodeCacheHash()
3535
// are defined in the generated code.
36-
// See src/node_code_cache_stub.cc for explanations.
36+
// See src/node_native_module.h for explanations.
3737

3838
const rl = readline.createInterface({
3939
input: fs.createReadStream(dest),
@@ -44,10 +44,10 @@ let hasCacheDef = false;
4444
let hasHashDef = false;
4545

4646
rl.on('line', common.mustCallAtLeast((line) => {
47-
if (line.includes('DefineCodeCache(')) {
47+
if (line.includes('LoadCodeCache(')) {
4848
hasCacheDef = true;
4949
}
50-
if (line.includes('DefineCodeCacheHash(')) {
50+
if (line.includes('LoadCodeCacheHash(')) {
5151
hasHashDef = true;
5252
}
5353
}, 2));

‎test/code-cache/test-code-cache.js

+12-3
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,7 @@ const {
1919
internalBinding
2020
} = require('internal/test/binding');
2121
const {
22-
compiledWithoutCache,
23-
compiledWithCache
22+
getCacheUsage
2423
} = internalBinding('native_module');
2524

2625
for (const key of cachableBuiltins) {
@@ -30,6 +29,12 @@ for (const key of cachableBuiltins) {
3029
require(key);
3130
}
3231

32+
// The computation has to be delayed until we have done loading modules
33+
const {
34+
compiledWithoutCache,
35+
compiledWithCache
36+
} = getCacheUsage();
37+
3338
const loadedModules = process.moduleLoadList
3439
.filter((m) => m.startsWith('NativeModule'))
3540
.map((m) => m.replace('NativeModule ', ''));
@@ -39,7 +44,11 @@ const loadedModules = process.moduleLoadList
3944
if (process.config.variables.node_code_cache_path === undefined) {
4045
console.log('The binary is not configured with code cache');
4146
assert.deepStrictEqual(compiledWithCache, new Set());
42-
assert.deepStrictEqual(compiledWithoutCache, new Set(loadedModules));
47+
48+
for (const key of loadedModules) {
49+
assert(compiledWithoutCache.has(key),
50+
`"${key}" should've been compiled without code cache`);
51+
}
4352
} else {
4453
console.log('The binary is configured with code cache');
4554
assert.strictEqual(

‎tools/generate_code_cache.js

+42-40
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,12 @@ const {
1313
cachableBuiltins
1414
} = require('internal/bootstrap/cache');
1515

16+
const {
17+
types: {
18+
isUint8Array
19+
}
20+
} = require('util');
21+
1622
function hash(str) {
1723
if (process.versions.openssl) {
1824
return require('crypto').createHash('sha256').update(str).digest('hex');
@@ -52,29 +58,25 @@ function formatSize(num) {
5258
* initializers of the code cache.
5359
*
5460
* @param {string} key ID of the builtin module
55-
* @param {Buffer} cache Code cache of the builtin module
61+
* @param {Uint8Array} cache Code cache of the builtin module
5662
* @return { definition: string, initializer: string }
5763
*/
5864
function getInitalizer(key, cache) {
59-
const defName = key.replace(/\//g, '_').replace(/-/g, '_');
60-
const definition = `static uint8_t ${defName}_raw[] = {\n` +
65+
const defName = `${key.replace(/\//g, '_').replace(/-/g, '_')}_raw`;
66+
const definition = `static const uint8_t ${defName}[] = {\n` +
6167
`${cache.join(',')}\n};`;
6268
const source = getSource(key);
6369
const sourceHash = hash(source);
64-
const initializer = `
65-
v8::Local<v8::ArrayBuffer> ${defName}_ab =
66-
v8::ArrayBuffer::New(isolate, ${defName}_raw, ${cache.length});
67-
v8::Local<v8::Uint8Array> ${defName}_array =
68-
v8::Uint8Array::New(${defName}_ab, 0, ${cache.length});
69-
target->Set(context,
70-
FIXED_ONE_BYTE_STRING(isolate, "${key}"),
71-
${defName}_array).FromJust();
72-
`;
73-
const hashIntializer = `
74-
target->Set(context,
75-
FIXED_ONE_BYTE_STRING(isolate, "${key}"),
76-
OneByteString(isolate, "${sourceHash}")).FromJust();
77-
`;
70+
const initializer =
71+
'code_cache_.emplace(\n' +
72+
` "${key}",\n` +
73+
` UnionBytes(${defName}, arraysize(${defName}))\n` +
74+
');';
75+
const hashIntializer =
76+
'code_cache_hash_.emplace(\n' +
77+
` "${key}",\n` +
78+
` "${sourceHash}"\n` +
79+
');';
7880
return {
7981
definition, initializer, hashIntializer, sourceHash
8082
};
@@ -85,55 +87,55 @@ const cacheInitializers = [];
8587
const cacheHashInitializers = [];
8688
let totalCacheSize = 0;
8789

90+
function lexical(a, b) {
91+
if (a < b) {
92+
return -1;
93+
}
94+
if (a > b) {
95+
return 1;
96+
}
97+
return 0;
98+
}
8899

89-
for (const key of cachableBuiltins) {
100+
for (const key of cachableBuiltins.sort(lexical)) {
90101
const cachedData = getCodeCache(key);
91-
if (!cachedData.length) {
102+
if (!isUint8Array(cachedData)) {
92103
console.error(`Failed to generate code cache for '${key}'`);
93104
process.exit(1);
94105
}
95106

96-
const length = cachedData.length;
97-
totalCacheSize += length;
107+
const size = cachedData.byteLength;
108+
totalCacheSize += size;
98109
const {
99110
definition, initializer, hashIntializer, sourceHash
100111
} = getInitalizer(key, cachedData);
101112
cacheDefinitions.push(definition);
102113
cacheInitializers.push(initializer);
103114
cacheHashInitializers.push(hashIntializer);
104-
console.log(`Generated cache for '${key}', size = ${formatSize(length)}` +
115+
console.log(`Generated cache for '${key}', size = ${formatSize(size)}` +
105116
`, hash = ${sourceHash}, total = ${formatSize(totalCacheSize)}`);
106117
}
107118

108-
const result = `#include "node.h"
109-
#include "node_code_cache.h"
110-
#include "v8.h"
111-
#include "env.h"
112-
#include "env-inl.h"
119+
const result = `#include "node_native_module.h"
120+
#include "node_internals.h"
113121
114122
// This file is generated by tools/generate_code_cache.js
115123
// and is used when configure is run with \`--code-cache-path\`
116124
117125
namespace node {
118-
126+
namespace native_module {
119127
${cacheDefinitions.join('\n\n')}
120128
121-
const bool native_module_has_code_cache = true;
122-
123-
// The target here will be returned as \`internalBinding('code_cache')\`
124-
void DefineCodeCache(Environment* env, v8::Local<v8::Object> target) {
125-
v8::Isolate* isolate = env->isolate();
126-
v8::Local<v8::Context> context = env->context();
127-
${cacheInitializers.join('\n')}
129+
void NativeModuleLoader::LoadCodeCache() {
130+
has_code_cache_ = true;
131+
${cacheInitializers.join('\n ')}
128132
}
129133
130-
// The target here will be returned as \`internalBinding('code_cache_hash')\`
131-
void DefineCodeCacheHash(Environment* env, v8::Local<v8::Object> target) {
132-
v8::Isolate* isolate = env->isolate();
133-
v8::Local<v8::Context> context = env->context();
134-
${cacheHashInitializers.join('\n')}
134+
void NativeModuleLoader::LoadCodeCacheHash() {
135+
${cacheHashInitializers.join('\n ')}
135136
}
136137
138+
} // namespace native_module
137139
} // namespace node
138140
`;
139141

‎tools/js2c.py

+47-88
Original file line numberDiff line numberDiff line change
@@ -43,11 +43,6 @@ def ToCArray(elements, step=10):
4343
slices = map(lambda s: ','.join(str(x) for x in s), slices)
4444
return ',\n'.join(slices)
4545

46-
47-
def ToCString(contents):
48-
return ToCArray(map(ord, contents), step=20)
49-
50-
5146
def ReadFile(filename):
5247
file = open(filename, "rt")
5348
try:
@@ -176,79 +171,49 @@ def ReadMacros(lines):
176171

177172

178173
TEMPLATE = """
179-
#include "node.h"
180-
#include "node_javascript.h"
181-
#include "v8.h"
182-
#include "env.h"
183-
#include "env-inl.h"
174+
#include "node_native_module.h"
175+
#include "node_internals.h"
184176
185177
namespace node {{
186178
187-
namespace {{
179+
namespace native_module {{
188180
189181
{definitions}
190182
191-
}} // anonymous namespace
192-
193-
v8::Local<v8::String> NodePerContextSource(v8::Isolate* isolate) {{
194-
return internal_per_context_value.ToStringChecked(isolate);
195-
}}
196-
197-
v8::Local<v8::String> LoadersBootstrapperSource(Environment* env) {{
198-
return internal_bootstrap_loaders_value.ToStringChecked(env->isolate());
199-
}}
200-
201-
v8::Local<v8::String> NodeBootstrapperSource(Environment* env) {{
202-
return internal_bootstrap_node_value.ToStringChecked(env->isolate());
203-
}}
204-
205-
void DefineJavaScript(Environment* env, v8::Local<v8::Object> target) {{
183+
void NativeModuleLoader::LoadJavaScriptSource() {{
206184
{initializers}
207185
}}
208186
209-
void DefineJavaScriptHash(Environment* env, v8::Local<v8::Object> target) {{
187+
void NativeModuleLoader::LoadJavaScriptHash() {{
210188
{hash_initializers}
211189
}}
212190
191+
}} // namespace native_module
192+
213193
}} // namespace node
214194
"""
215195

216196
ONE_BYTE_STRING = """
217-
static const uint8_t raw_{var}[] = {{ {data} }};
218-
static struct : public v8::String::ExternalOneByteStringResource {{
219-
const char* data() const override {{
220-
return reinterpret_cast<const char*>(raw_{var});
221-
}}
222-
size_t length() const override {{ return arraysize(raw_{var}); }}
223-
void Dispose() override {{ /* Default calls `delete this`. */ }}
224-
v8::Local<v8::String> ToStringChecked(v8::Isolate* isolate) {{
225-
return v8::String::NewExternalOneByte(isolate, this).ToLocalChecked();
226-
}}
227-
}} {var};
197+
static const uint8_t {var}[] = {{ {data} }};
228198
"""
229199

230200
TWO_BYTE_STRING = """
231-
static const uint16_t raw_{var}[] = {{ {data} }};
232-
static struct : public v8::String::ExternalStringResource {{
233-
const uint16_t* data() const override {{ return raw_{var}; }}
234-
size_t length() const override {{ return arraysize(raw_{var}); }}
235-
void Dispose() override {{ /* Default calls `delete this`. */ }}
236-
v8::Local<v8::String> ToStringChecked(v8::Isolate* isolate) {{
237-
return v8::String::NewExternalTwoByte(isolate, this).ToLocalChecked();
238-
}}
239-
}} {var};
201+
static const uint16_t {var}[] = {{ {data} }};
240202
"""
241203

242-
INITIALIZER = """\
243-
CHECK(target->Set(env->context(),
244-
{key}.ToStringChecked(env->isolate()),
245-
{value}.ToStringChecked(env->isolate())).FromJust());
204+
205+
INITIALIZER = """
206+
source_.emplace(
207+
"{module}",
208+
UnionBytes({var}, arraysize({var}))
209+
);
246210
"""
247211

248212
HASH_INITIALIZER = """\
249-
CHECK(target->Set(env->context(),
250-
FIXED_ONE_BYTE_STRING(env->isolate(), "{key}"),
251-
FIXED_ONE_BYTE_STRING(env->isolate(), "{value}")).FromJust());
213+
source_hash_.emplace(
214+
"{module}",
215+
"{hash_value}"
216+
);
252217
"""
253218

254219
DEPRECATED_DEPS = """\
@@ -259,20 +224,6 @@ def ReadMacros(lines):
259224
module.exports = require('internal/deps/{module}');
260225
"""
261226

262-
263-
def Render(var, data):
264-
# Treat non-ASCII as UTF-8 and convert it to UTF-16.
265-
if any(ord(c) > 127 for c in data):
266-
template = TWO_BYTE_STRING
267-
data = map(ord, data.decode('utf-8').encode('utf-16be'))
268-
data = [data[i] * 256 + data[i+1] for i in xrange(0, len(data), 2)]
269-
data = ToCArray(data)
270-
else:
271-
template = ONE_BYTE_STRING
272-
data = ToCString(data)
273-
return template.format(var=var, data=data)
274-
275-
276227
def JS2C(source, target):
277228
modules = []
278229
consts = {}
@@ -291,7 +242,29 @@ def JS2C(source, target):
291242
# Build source code lines
292243
definitions = []
293244
initializers = []
294-
hash_initializers = [];
245+
hash_initializers = []
246+
247+
def AddModule(module, source):
248+
var = '%s_raw' % (module.replace('-', '_').replace('/', '_'))
249+
source_hash = hashlib.sha256(source).hexdigest()
250+
251+
# Treat non-ASCII as UTF-8 and convert it to UTF-16.
252+
if any(ord(c) > 127 for c in source):
253+
source = map(ord, source.decode('utf-8').encode('utf-16be'))
254+
source = [source[i] * 256 + source[i+1] for i in xrange(0, len(source), 2)]
255+
source = ToCArray(source)
256+
definition = TWO_BYTE_STRING.format(var=var, data=source)
257+
else:
258+
source = ToCArray(map(ord, source), step=20)
259+
definition = ONE_BYTE_STRING.format(var=var, data=source)
260+
261+
initializer = INITIALIZER.format(module=module,
262+
var=var)
263+
hash_initializer = HASH_INITIALIZER.format(module=module,
264+
hash_value=source_hash)
265+
definitions.append(definition)
266+
initializers.append(initializer)
267+
hash_initializers.append(hash_initializer)
295268

296269
for name in modules:
297270
lines = ReadFile(str(name))
@@ -317,28 +290,14 @@ def JS2C(source, target):
317290
if name.endswith(".gypi"):
318291
lines = re.sub(r'#.*?\n', '', lines)
319292
lines = re.sub(r'\'', '"', lines)
320-
name = name.split('.', 1)[0]
321-
var = name.replace('-', '_').replace('/', '_')
322-
key = '%s_key' % var
323-
value = '%s_value' % var
324-
hash_value = hashlib.sha256(lines).hexdigest()
325293

326-
definitions.append(Render(key, name))
327-
definitions.append(Render(value, lines))
328-
initializers.append(INITIALIZER.format(key=key, value=value))
329-
hash_initializers.append(HASH_INITIALIZER.format(key=name, value=hash_value))
294+
AddModule(name.split('.', 1)[0], lines)
330295

296+
# Add deprecated aliases for deps without 'deps/'
331297
if deprecated_deps is not None:
332-
name = '/'.join(deprecated_deps)
333-
name = name.split('.', 1)[0]
334-
var = name.replace('-', '_').replace('/', '_')
335-
key = '%s_key' % var
336-
value = '%s_value' % var
337-
338-
definitions.append(Render(key, name))
339-
definitions.append(Render(value, DEPRECATED_DEPS.format(module=name)))
340-
initializers.append(INITIALIZER.format(key=key, value=value))
341-
hash_initializers.append(HASH_INITIALIZER.format(key=name, value=hash_value))
298+
module = '/'.join(deprecated_deps).split('.', 1)[0]
299+
source = DEPRECATED_DEPS.format(module=module)
300+
AddModule(module, source)
342301

343302
# Emit result
344303
output = open(str(target[0]), "w")

0 commit comments

Comments
 (0)
Please sign in to comment.