1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2024-11-21 15:04:11 -05:00

Fix snapshot source code wrapping.

This commit is contained in:
Ryan Dahl 2018-06-19 17:45:58 +02:00
parent 7a38a2f1a7
commit b2b4054f97
2 changed files with 45 additions and 34 deletions

View file

@ -1,55 +1,57 @@
// A simple runtime that doesn't involve typescript or protobufs to test // A simple runtime that doesn't involve typescript or protobufs to test
// libdeno. Invoked by mock_runtime_test.cc // libdeno. Invoked by mock_runtime_test.cc
const global = this;
function assert(cond) { function assert(cond) {
if (!cond) throw Error("mock_runtime.js assert failed"); if (!cond) throw Error("mock_runtime.js assert failed");
} }
function typedArrayToArrayBuffer(ta) { global.typedArrayToArrayBuffer = (ta) => {
return ta.buffer.slice(ta.byteOffset, ta.byteOffset + ta.byteLength); return ta.buffer.slice(ta.byteOffset, ta.byteOffset + ta.byteLength);
} };
function CanCallFunction() { global.CanCallFunction = () => {
deno.print("Hello world from foo"); deno.print("Hello world from foo");
return "foo"; return "foo";
} };
// This object is created to test snapshotting. // This object is created to test snapshotting.
// See DeserializeInternalFieldsCallback and SerializeInternalFieldsCallback. // See DeserializeInternalFieldsCallback and SerializeInternalFieldsCallback.
const snapshotted = new Uint8Array([1, 3, 3, 7]); const snapshotted = new Uint8Array([1, 3, 3, 7]);
function TypedArraySnapshots() { global.TypedArraySnapshots = () => {
assert(snapshotted[0] === 1); assert(snapshotted[0] === 1);
assert(snapshotted[1] === 3); assert(snapshotted[1] === 3);
assert(snapshotted[2] === 3); assert(snapshotted[2] === 3);
assert(snapshotted[3] === 7); assert(snapshotted[3] === 7);
} };
function PubSuccess() { global.PubSuccess = () => {
deno.sub((channel, msg) => { deno.sub((channel, msg) => {
assert(channel === "PubSuccess"); assert(channel === "PubSuccess");
deno.print("PubSuccess: ok"); deno.print("PubSuccess: ok");
}); });
} };
function PubByteLength() { global.PubByteLength = () => {
deno.sub((channel, msg) => { deno.sub((channel, msg) => {
assert(channel === "PubByteLength"); assert(channel === "PubByteLength");
assert(msg instanceof ArrayBuffer); assert(msg instanceof ArrayBuffer);
assert(msg.byteLength === 3); assert(msg.byteLength === 3);
}); });
} };
function SubReturnEmpty() { global.SubReturnEmpty = () => {
const ui8 = new Uint8Array("abc".split("").map(c => c.charCodeAt(0))); const ui8 = new Uint8Array("abc".split("").map(c => c.charCodeAt(0)));
const ab = typedArrayToArrayBuffer(ui8); const ab = typedArrayToArrayBuffer(ui8);
let r = deno.pub("SubReturnEmpty", ab); let r = deno.pub("SubReturnEmpty", ab);
assert(r == null); assert(r == null);
r = deno.pub("SubReturnEmpty", ab); r = deno.pub("SubReturnEmpty", ab);
assert(r == null); assert(r == null);
} };
function SubReturnBar() { global.SubReturnBar = () => {
const ui8 = new Uint8Array("abc".split("").map(c => c.charCodeAt(0))); const ui8 = new Uint8Array("abc".split("").map(c => c.charCodeAt(0)));
const ab = typedArrayToArrayBuffer(ui8); const ab = typedArrayToArrayBuffer(ui8);
const r = deno.pub("SubReturnBar", ab); const r = deno.pub("SubReturnBar", ab);
@ -58,18 +60,18 @@ function SubReturnBar() {
const rui8 = new Uint8Array(r); const rui8 = new Uint8Array(r);
const rstr = String.fromCharCode(...rui8); const rstr = String.fromCharCode(...rui8);
assert(rstr === "bar"); assert(rstr === "bar");
} };
function DoubleSubFails() { global.DoubleSubFails = () => {
// deno.sub is an internal function and should only be called once from the // deno.sub is an internal function and should only be called once from the
// runtime. // runtime.
deno.sub((channel, msg) => assert(false)); deno.sub((channel, msg) => assert(false));
deno.sub((channel, msg) => assert(false)); deno.sub((channel, msg) => assert(false));
} };
// The following join has caused SnapshotBug to segfault when using kKeep. // The following join has caused SnapshotBug to segfault when using kKeep.
[].join(""); [].join("");
function SnapshotBug() { global.SnapshotBug = () => {
assert("1,2,3" === String([1, 2, 3])); assert("1,2,3" === String([1, 2, 3]));
} };

View file

@ -42,6 +42,30 @@ v8::StartupData MakeSnapshot(const char* js_filename, const char* js_source) {
return snapshot_blob; return snapshot_blob;
} }
// Wrap the js_source in an IIFE to work around a bug in the V8 snapshot
// serializer. Without it, CreateBlob() triggers the following assert:
// Debug check failed : outer_scope_info()->IsScopeInfo() || is_toplevel().
// ==== C stack trace ====
// v8::internal::SharedFunctionInfo::FlushCompiled
// v8::SnapshotCreator::CreateBlob
// deno::MakeSnapshot
// Avoid misaligning the source map, and ensure that the sourceMappingUrl
// comment remains at the last line.
// Try removing this when this bug is fixed:
// https://bugs.chromium.org/p/v8/issues/detail?id=7857
std::string WrapSourceCode(const std::string& js_source) {
auto smu_offset = js_source.rfind("//# sourceMappingURL=");
std::string tail =
smu_offset == std::string::npos ? "" : js_source.substr(smu_offset);
auto wrapped_js_source =
"(function() {" + js_source.substr(0, smu_offset) + "\n})();\n" + tail;
// Double check that the source mapping url comment is at the last line.
auto last_line = wrapped_js_source.substr(wrapped_js_source.rfind('\n'));
CHECK(smu_offset == std::string::npos ||
last_line.find("sourceMappingURL") != std::string::npos);
return wrapped_js_source;
}
} // namespace deno } // namespace deno
int main(int argc, char** argv) { int main(int argc, char** argv) {
@ -57,22 +81,7 @@ int main(int argc, char** argv) {
std::string js_source; std::string js_source;
CHECK(deno::ReadFileToString(js_fn, &js_source)); CHECK(deno::ReadFileToString(js_fn, &js_source));
// Wrap the js_source in an IIFE to work around a bug in the V8 snapshot auto wrapped_js_source = deno::WrapSourceCode(js_source);
// serializer. Without it, CreateBlob() triggers the following assert:
// Debug check failed : outer_scope_info()->IsScopeInfo() || is_toplevel().
// ==== C stack trace ====
// v8::internal::SharedFunctionInfo::FlushCompiled
// v8::SnapshotCreator::CreateBlob
// deno::MakeSnapshot
// Avoid misaligning the source map, and ensure that the sourceMappingUrl
// comment remains at the last line.
auto smu_offset = js_source.rfind("//#");
CHECK(smu_offset != std::string::npos);
auto wrapped_js_source = "(function() {" + js_source.substr(0, smu_offset) +
"\n})();\n" + js_source.substr(smu_offset);
// Double check that the source mapping url comment is at the last line.
auto last_line = wrapped_js_source.substr(wrapped_js_source.rfind('\n'));
CHECK(last_line.find("sourceMappingURL") != std::string::npos);
deno_init(); deno_init();
auto snapshot_blob = deno::MakeSnapshot(js_fn, wrapped_js_source.c_str()); auto snapshot_blob = deno::MakeSnapshot(js_fn, wrapped_js_source.c_str());