Commit 1a73f73b authored by Clemens Hammacher's avatar Clemens Hammacher Committed by Commit Bot

[wasm] Implement extensible name section

The format of the name section changed recently. It now contains
subsections of different type (currently for function names or local
variable names).
This CL changes our internal wasm module builders (in JS and C++) to
emit this new format, and changes the decoder to understand it.
We currently only parse the function name section, and ignore names of
local variables. I will later extend this to parse local variable names
when needed for debugging.

R=ahaas@chromium.org, rossberg@chromium.org
BUG=v8:6222

Change-Id: I2627160c25c9209a3f09abe0b88941ec48b24434
Reviewed-on: https://chromium-review.googlesource.com/470247
Commit-Queue: Clemens Hammacher <clemensh@chromium.org>
Reviewed-by: 's avatarAndreas Rossberg <rossberg@chromium.org>
Cr-Commit-Position: refs/heads/master@{#44492}
parent ceb3f4b1
...@@ -30,7 +30,7 @@ namespace wasm { ...@@ -30,7 +30,7 @@ namespace wasm {
#define TRACE(...) #define TRACE(...)
#endif #endif
const char* SectionName(WasmSectionCode code) { const char* SectionName(SectionCode code) {
switch (code) { switch (code) {
case kUnknownSectionCode: case kUnknownSectionCode:
return "Unknown"; return "Unknown";
...@@ -106,7 +106,7 @@ class WasmSectionIterator { ...@@ -106,7 +106,7 @@ class WasmSectionIterator {
return section_code_ != kUnknownSectionCode && decoder_.more(); return section_code_ != kUnknownSectionCode && decoder_.more();
} }
inline WasmSectionCode section_code() const { return section_code_; } inline SectionCode section_code() const { return section_code_; }
inline const byte* section_start() const { return section_start_; } inline const byte* section_start() const { return section_start_; }
...@@ -138,7 +138,7 @@ class WasmSectionIterator { ...@@ -138,7 +138,7 @@ class WasmSectionIterator {
private: private:
Decoder& decoder_; Decoder& decoder_;
WasmSectionCode section_code_; SectionCode section_code_;
const byte* section_start_; const byte* section_start_;
const byte* payload_start_; const byte* payload_start_;
const byte* section_end_; const byte* section_end_;
...@@ -184,15 +184,13 @@ class WasmSectionIterator { ...@@ -184,15 +184,13 @@ class WasmSectionIterator {
strncmp(reinterpret_cast<const char*>(section_name_start), strncmp(reinterpret_cast<const char*>(section_name_start),
kNameString, kNameStringLength) == 0) { kNameString, kNameStringLength) == 0) {
section_code = kNameSectionCode; section_code = kNameSectionCode;
} else {
section_code = kUnknownSectionCode;
} }
} else if (!IsValidSectionCode(section_code)) { } else if (!IsValidSectionCode(section_code)) {
decoder_.errorf(decoder_.pc(), "unknown section code #0x%02x", decoder_.errorf(decoder_.pc(), "unknown section code #0x%02x",
section_code); section_code);
section_code = kUnknownSectionCode; section_code = kUnknownSectionCode;
} }
section_code_ = static_cast<WasmSectionCode>(section_code); section_code_ = static_cast<SectionCode>(section_code);
TRACE("Section: %s\n", SectionName(section_code_)); TRACE("Section: %s\n", SectionName(section_code_));
if (section_code_ == kUnknownSectionCode && if (section_code_ == kUnknownSectionCode &&
...@@ -651,22 +649,36 @@ class ModuleDecoder : public Decoder { ...@@ -651,22 +649,36 @@ class ModuleDecoder : public Decoder {
// TODO(titzer): find a way to report name errors as warnings. // TODO(titzer): find a way to report name errors as warnings.
// Use an inner decoder so that errors don't fail the outer decoder. // Use an inner decoder so that errors don't fail the outer decoder.
Decoder inner(start_, pc_, end_); Decoder inner(start_, pc_, end_);
uint32_t functions_count = inner.consume_u32v("functions count"); // Decode all name subsections.
// Be lenient with their order.
for (uint32_t i = 0; inner.ok() && i < functions_count; ++i) { while (inner.ok() && inner.more()) {
uint32_t function_name_length = 0; uint8_t name_type = inner.consume_u8("name type");
uint32_t name_offset = if (name_type & 0x80) inner.error("name type if not varuint7");
consume_string(inner, &function_name_length, false);
uint32_t func_index = i; uint32_t name_payload_len = inner.consume_u32v("name payload length");
if (inner.ok() && func_index < module->functions.size()) { if (!inner.checkAvailable(name_payload_len)) break;
module->functions[func_index].name_offset = name_offset;
module->functions[func_index].name_length = function_name_length; // Decode function names, ignore the rest.
} // Local names will be decoded when needed.
if (name_type == NameSectionType::kFunction) {
uint32_t local_names_count = inner.consume_u32v("local names count"); uint32_t functions_count = inner.consume_u32v("functions count");
for (uint32_t j = 0; inner.ok() && j < local_names_count; j++) {
uint32_t length = inner.consume_u32v("string length"); for (; inner.ok() && functions_count > 0; --functions_count) {
inner.consume_bytes(length, "string"); uint32_t function_index = inner.consume_u32v("function index");
uint32_t name_length = 0;
uint32_t name_offset = consume_string(inner, &name_length, false);
// Be lenient with errors in the name section: Ignore illegal
// or out-of-order indexes and non-UTF8 names. You can even assign
// to the same function multiple times (last valid one wins).
if (inner.ok() && function_index < module->functions.size() &&
unibrow::Utf8::Validate(inner.start() + name_offset,
name_length)) {
module->functions[function_index].name_offset = name_offset;
module->functions[function_index].name_length = name_length;
}
}
} else {
inner.consume_bytes(name_payload_len, "name subsection payload");
} }
} }
// Skip the whole names section in the outer decoder. // Skip the whole names section in the outer decoder.
......
...@@ -20,7 +20,7 @@ const uint8_t kWasmFunctionTypeForm = 0x60; ...@@ -20,7 +20,7 @@ const uint8_t kWasmFunctionTypeForm = 0x60;
const uint8_t kWasmAnyFunctionTypeForm = 0x70; const uint8_t kWasmAnyFunctionTypeForm = 0x70;
const uint8_t kResizableMaximumFlag = 1; const uint8_t kResizableMaximumFlag = 1;
enum WasmSectionCode { enum SectionCode {
kUnknownSectionCode = 0, // code for unknown sections kUnknownSectionCode = 0, // code for unknown sections
kTypeSectionCode = 1, // Function signature declarations kTypeSectionCode = 1, // Function signature declarations
kImportSectionCode = 2, // Import declarations kImportSectionCode = 2, // Import declarations
...@@ -36,11 +36,13 @@ enum WasmSectionCode { ...@@ -36,11 +36,13 @@ enum WasmSectionCode {
kNameSectionCode = 12, // Name section (encoded as a string) kNameSectionCode = 12, // Name section (encoded as a string)
}; };
enum NameSectionType : uint8_t { kFunction = 1, kLocal = 2 };
inline bool IsValidSectionCode(uint8_t byte) { inline bool IsValidSectionCode(uint8_t byte) {
return kTypeSectionCode <= byte && byte <= kDataSectionCode; return kTypeSectionCode <= byte && byte <= kDataSectionCode;
} }
const char* SectionName(WasmSectionCode code); const char* SectionName(SectionCode code);
typedef Result<const WasmModule*> ModuleResult; typedef Result<const WasmModule*> ModuleResult;
typedef Result<WasmFunction*> FunctionResult; typedef Result<WasmFunction*> FunctionResult;
......
...@@ -34,7 +34,7 @@ namespace wasm { ...@@ -34,7 +34,7 @@ namespace wasm {
// Emit a section code and the size as a padded varint that can be patched // Emit a section code and the size as a padded varint that can be patched
// later. // later.
size_t EmitSection(WasmSectionCode code, ZoneBuffer& buffer) { size_t EmitSection(SectionCode code, ZoneBuffer& buffer) {
// Emit the section code. // Emit the section code.
buffer.write_u8(code); buffer.write_u8(code);
...@@ -206,8 +206,7 @@ void WasmFunctionBuilder::WriteExports(ZoneBuffer& buffer) const { ...@@ -206,8 +206,7 @@ void WasmFunctionBuilder::WriteExports(ZoneBuffer& buffer) const {
buffer.write_size(name.size()); buffer.write_size(name.size());
buffer.write(reinterpret_cast<const byte*>(name.data()), name.size()); buffer.write(reinterpret_cast<const byte*>(name.data()), name.size());
buffer.write_u8(kExternalFunction); buffer.write_u8(kExternalFunction);
buffer.write_u32v(func_index_ + static_cast<uint32_t>( buffer.write_size(func_index_ + builder_->function_imports_.size());
builder_->function_imports_.size()));
} }
} }
...@@ -391,14 +390,14 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const { ...@@ -391,14 +390,14 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const {
} }
// == Emit function signatures =============================================== // == Emit function signatures ===============================================
bool has_names = false; uint32_t num_function_names = 0;
if (functions_.size() > 0) { if (functions_.size() > 0) {
size_t start = EmitSection(kFunctionSectionCode, buffer); size_t start = EmitSection(kFunctionSectionCode, buffer);
buffer.write_size(functions_.size()); buffer.write_size(functions_.size());
for (auto function : functions_) { for (auto function : functions_) {
function->WriteSignature(buffer); function->WriteSignature(buffer);
exports += static_cast<uint32_t>(function->exported_names_.size()); exports += static_cast<uint32_t>(function->exported_names_.size());
if (function->name_.size() > 0) has_names = true; if (!function->name_.empty()) ++num_function_names;
} }
FixupSection(buffer, start); FixupSection(buffer, start);
} }
...@@ -507,8 +506,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const { ...@@ -507,8 +506,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const {
// == emit start function index ============================================== // == emit start function index ==============================================
if (start_function_index_ >= 0) { if (start_function_index_ >= 0) {
size_t start = EmitSection(kStartSectionCode, buffer); size_t start = EmitSection(kStartSectionCode, buffer);
buffer.write_u32v(start_function_index_ + buffer.write_size(start_function_index_ + function_imports_.size());
static_cast<uint32_t>(function_imports_.size()));
FixupSection(buffer, start); FixupSection(buffer, start);
} }
...@@ -523,8 +521,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const { ...@@ -523,8 +521,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const {
buffer.write_size(indirect_functions_.size()); // element count buffer.write_size(indirect_functions_.size()); // element count
for (auto index : indirect_functions_) { for (auto index : indirect_functions_) {
buffer.write_u32v(index + buffer.write_size(index + function_imports_.size());
static_cast<uint32_t>(function_imports_.size()));
} }
FixupSection(buffer, start); FixupSection(buffer, start);
...@@ -557,7 +554,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const { ...@@ -557,7 +554,7 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const {
} }
// == Emit names ============================================================= // == Emit names =============================================================
if (has_names) { if (num_function_names > 0 || !function_imports_.empty()) {
// Emit the section code. // Emit the section code.
buffer.write_u8(kUnknownSectionCode); buffer.write_u8(kUnknownSectionCode);
// Emit a placeholder for the length. // Emit a placeholder for the length.
...@@ -565,19 +562,37 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const { ...@@ -565,19 +562,37 @@ void WasmModuleBuilder::WriteTo(ZoneBuffer& buffer) const {
// Emit the section string. // Emit the section string.
buffer.write_size(4); buffer.write_size(4);
buffer.write(reinterpret_cast<const byte*>("name"), 4); buffer.write(reinterpret_cast<const byte*>("name"), 4);
// Emit the names. // Emit a subsection for the function names.
size_t count = functions_.size() + function_imports_.size(); buffer.write_u8(NameSectionType::kFunction);
buffer.write_size(count); // Emit a placeholder for the subsection length.
for (size_t i = 0; i < function_imports_.size(); i++) { size_t functions_start = buffer.reserve_u32v();
buffer.write_u8(0); // empty name for import // Emit the function names.
buffer.write_u8(0); // no local variables // Imports are always named.
uint32_t num_imports = static_cast<uint32_t>(function_imports_.size());
buffer.write_size(num_imports + num_function_names);
uint32_t function_index = 0;
for (; function_index < num_imports; ++function_index) {
const WasmFunctionImport* import = &function_imports_[function_index];
DCHECK_NOT_NULL(import->name);
buffer.write_u32v(function_index);
uint32_t name_len = static_cast<uint32_t>(import->name_length);
buffer.write_u32v(name_len);
buffer.write(reinterpret_cast<const byte*>(import->name), name_len);
} }
for (auto function : functions_) { if (num_function_names > 0) {
buffer.write_size(function->name_.size()); for (auto function : functions_) {
buffer.write(reinterpret_cast<const byte*>(function->name_.data()), DCHECK_EQ(function_index,
function->name_.size()); function->func_index() + function_imports_.size());
buffer.write_u8(0); if (!function->name_.empty()) {
buffer.write_u32v(function_index);
buffer.write_size(function->name_.size());
buffer.write(reinterpret_cast<const byte*>(function->name_.data()),
function->name_.size());
}
++function_index;
}
} }
FixupSection(buffer, functions_start);
FixupSection(buffer, start); FixupSection(buffer, start);
} }
} }
......
...@@ -1132,9 +1132,10 @@ MaybeHandle<String> WasmCompiledModule::ExtractUtf8StringFromModuleBytes( ...@@ -1132,9 +1132,10 @@ MaybeHandle<String> WasmCompiledModule::ExtractUtf8StringFromModuleBytes(
isolate); isolate);
DCHECK_GE(module_bytes->length(), offset); DCHECK_GE(module_bytes->length(), offset);
DCHECK_GE(module_bytes->length() - offset, size); DCHECK_GE(module_bytes->length() - offset, size);
Address raw = module_bytes->GetCharsAddress() + offset; // UTF8 validation happens at decode time.
if (!unibrow::Utf8::Validate(reinterpret_cast<const byte*>(raw), size)) DCHECK(unibrow::Utf8::Validate(
return {}; // UTF8 decoding error for name. reinterpret_cast<const byte*>(module_bytes->GetCharsAddress() + offset),
size));
DCHECK_GE(kMaxInt, offset); DCHECK_GE(kMaxInt, offset);
DCHECK_GE(kMaxInt, size); DCHECK_GE(kMaxInt, size);
return isolate->factory()->NewStringFromUtf8SubString( return isolate->factory()->NewStringFromUtf8SubString(
...@@ -1219,6 +1220,8 @@ MaybeHandle<String> WasmCompiledModule::GetFunctionNameOrNull( ...@@ -1219,6 +1220,8 @@ MaybeHandle<String> WasmCompiledModule::GetFunctionNameOrNull(
uint32_t func_index) { uint32_t func_index) {
DCHECK_LT(func_index, compiled_module->module()->functions.size()); DCHECK_LT(func_index, compiled_module->module()->functions.size());
WasmFunction& function = compiled_module->module()->functions[func_index]; WasmFunction& function = compiled_module->module()->functions[func_index];
DCHECK_IMPLIES(function.name_offset == 0, function.name_length == 0);
if (!function.name_offset) return {};
return WasmCompiledModule::ExtractUtf8StringFromModuleBytes( return WasmCompiledModule::ExtractUtf8StringFromModuleBytes(
isolate, compiled_module, function.name_offset, function.name_length); isolate, compiled_module, function.name_offset, function.name_length);
} }
......
...@@ -308,14 +308,17 @@ class TestingModule : public ModuleEnv { ...@@ -308,14 +308,17 @@ class TestingModule : public ModuleEnv {
Handle<SeqOneByteString> old_bytes( Handle<SeqOneByteString> old_bytes(
instance_object_->compiled_module()->module_bytes(), isolate_); instance_object_->compiled_module()->module_bytes(), isolate_);
uint32_t old_size = static_cast<uint32_t>(old_bytes->length()); uint32_t old_size = static_cast<uint32_t>(old_bytes->length());
ScopedVector<byte> new_bytes(old_size + bytes.length()); // Avoid placing strings at offset 0, this might be interpreted as "not
// set", e.g. for function names.
uint32_t bytes_offset = old_size ? old_size : 1;
ScopedVector<byte> new_bytes(bytes_offset + bytes.length());
memcpy(new_bytes.start(), old_bytes->GetChars(), old_size); memcpy(new_bytes.start(), old_bytes->GetChars(), old_size);
memcpy(new_bytes.start() + old_size, bytes.start(), bytes.length()); memcpy(new_bytes.start() + bytes_offset, bytes.start(), bytes.length());
Handle<SeqOneByteString> new_bytes_str = Handle<SeqOneByteString>::cast( Handle<SeqOneByteString> new_bytes_str = Handle<SeqOneByteString>::cast(
isolate_->factory()->NewStringFromOneByte(new_bytes).ToHandleChecked()); isolate_->factory()->NewStringFromOneByte(new_bytes).ToHandleChecked());
instance_object_->compiled_module()->shared()->set_module_bytes( instance_object_->compiled_module()->shared()->set_module_bytes(
*new_bytes_str); *new_bytes_str);
return old_size; return bytes_offset;
} }
WasmFunction* GetFunctionAt(int index) { return &module_.functions[index]; } WasmFunction* GetFunctionAt(int index) { return &module_.functions[index]; }
......
...@@ -19,8 +19,7 @@ using namespace v8::internal::wasm; ...@@ -19,8 +19,7 @@ using namespace v8::internal::wasm;
static const char* kNameString = "name"; static const char* kNameString = "name";
static const size_t kNameStringLength = 4; static const size_t kNameStringLength = 4;
int fuzz_wasm_section(WasmSectionCode section, const uint8_t* data, int fuzz_wasm_section(SectionCode section, const uint8_t* data, size_t size) {
size_t size) {
v8_fuzzer::FuzzerSupport* support = v8_fuzzer::FuzzerSupport::Get(); v8_fuzzer::FuzzerSupport* support = v8_fuzzer::FuzzerSupport::Get();
v8::Isolate* isolate = support->GetIsolate(); v8::Isolate* isolate = support->GetIsolate();
v8::internal::Isolate* i_isolate = v8::internal::Isolate* i_isolate =
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
#include "src/wasm/module-decoder.h" #include "src/wasm/module-decoder.h"
int fuzz_wasm_section(v8::internal::wasm::WasmSectionCode section, int fuzz_wasm_section(v8::internal::wasm::SectionCode section,
const uint8_t* data, size_t size); const uint8_t* data, size_t size);
#endif // WASM_SECTION_FUZZERS_H_ #endif // WASM_SECTION_FUZZERS_H_
Running testFunction... Running testFunction...
Script nr 0 parsed. URL: v8://test/setup Script nr 0 parsed. URL: v8://test/setup
Script nr 1 parsed. URL: v8://test/runTestFunction Script nr 1 parsed. URL: v8://test/runTestFunction
Script nr 2 parsed. URL: wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0 Script nr 2 parsed. URL: wasm://wasm/wasm-354ada0e/wasm-354ada0e-0
This is a wasm script (nr 0). This is a wasm script (nr 0).
Script nr 3 parsed. URL: wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1 Script nr 3 parsed. URL: wasm://wasm/wasm-354ada0e/wasm-354ada0e-1
This is a wasm script (nr 1). This is a wasm script (nr 1).
Querying breakable locations for all wasm scripts now... Querying breakable locations for all wasm scripts now...
Requesting all breakable locations in wasm script 0 Requesting all breakable locations in wasm script 0
...@@ -37,51 +37,51 @@ Requesting breakable locations in lines [4,6) ...@@ -37,51 +37,51 @@ Requesting breakable locations in lines [4,6)
[0] 4:6 || >call 0 [0] 4:6 || >call 0
[1] 5:4 || >end [1] 5:4 || >end
Setting a breakpoint on each breakable location... Setting a breakpoint on each breakable location...
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:2:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:2:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:3:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:3:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:4:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:4:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:5:0 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:5:0
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:1:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:1:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:2:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:2:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:3:4 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:3:4
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:4:6 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:4:6
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:5:4 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:5:4
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:6:2 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:6:2
Success! Success!
Setting at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:7:0 Setting at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:7:0
Success! Success!
Running wasm code... Running wasm code...
Missing breakpoints: 11 Missing breakpoints: 11
Script nr 4 parsed. URL: v8://test/runWasm Script nr 4 parsed. URL: v8://test/runWasm
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:1:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:1:2
Missing breakpoints: 10 Missing breakpoints: 10
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:2:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:2:2
Missing breakpoints: 9 Missing breakpoints: 9
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:3:4 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:3:4
Missing breakpoints: 8 Missing breakpoints: 8
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:4:6 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:4:6
Missing breakpoints: 7 Missing breakpoints: 7
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:2:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:2:2
Missing breakpoints: 6 Missing breakpoints: 6
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:3:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:3:2
Missing breakpoints: 5 Missing breakpoints: 5
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:4:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:4:2
Missing breakpoints: 4 Missing breakpoints: 4
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-0:5:0 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-0:5:0
Missing breakpoints: 3 Missing breakpoints: 3
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:5:4 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:5:4
Missing breakpoints: 2 Missing breakpoints: 2
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:6:2 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:6:2
Missing breakpoints: 1 Missing breakpoints: 1
Stopped at wasm://wasm/wasm-eb3c2032/wasm-eb3c2032-1:7:0 Stopped at wasm://wasm/wasm-354ada0e/wasm-354ada0e-1:7:0
Missing breakpoints: 0 Missing breakpoints: 0
Finished! Finished!
Check that inspector gets two wasm scripts at module creation time. Check that inspector gets two wasm scripts at module creation time.
Script #0 parsed. URL: v8://test/testFunction Script #0 parsed. URL: v8://test/testFunction
Script #1 parsed. URL: v8://test/runTestRunction Script #1 parsed. URL: v8://test/runTestRunction
Script #2 parsed. URL: wasm://wasm/wasm-4205e2ce/wasm-4205e2ce-0 Script #2 parsed. URL: wasm://wasm/wasm-7b04570e/wasm-7b04570e-0
Script #3 parsed. URL: wasm://wasm/wasm-4205e2ce/wasm-4205e2ce-1 Script #3 parsed. URL: wasm://wasm/wasm-7b04570e/wasm-7b04570e-1
Source for wasm://wasm/wasm-4205e2ce/wasm-4205e2ce-0: Source for wasm://wasm/wasm-7b04570e/wasm-7b04570e-0:
func $nopFunction func $nopFunction
nop nop
end end
Source for wasm://wasm/wasm-4205e2ce/wasm-4205e2ce-1: Source for wasm://wasm/wasm-7b04570e/wasm-7b04570e-1:
func $main func $main
block block
i32.const 2 i32.const 2
......
...@@ -2,10 +2,10 @@ Installing code an global variable. ...@@ -2,10 +2,10 @@ Installing code an global variable.
Calling instantiate function. Calling instantiate function.
Waiting for two wasm scripts to be parsed. Waiting for two wasm scripts to be parsed.
Ignoring script with url v8://test/callInstantiate Ignoring script with url v8://test/callInstantiate
Got wasm script: wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0 Got wasm script: wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0
Requesting source for wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0... Requesting source for wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0...
Got wasm script: wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1 Got wasm script: wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1
Requesting source for wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1... Requesting source for wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1...
func $wasm_A func $wasm_A
nop nop
nop nop
...@@ -25,59 +25,59 @@ func $wasm_B (param i32) ...@@ -25,59 +25,59 @@ func $wasm_B (param i32)
end end
end end
Setting breakpoint on line 7 (on the setlocal before the call), url wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1 Setting breakpoint on line 7 (on the setlocal before the call), url wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1
{ {
columnNumber : 6 columnNumber : 6
lineNumber : 7 lineNumber : 7
scriptId : <scriptId> scriptId : <scriptId>
} }
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:7:6: >set_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:7:6: >set_local 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:8:6: >call 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:8:6: >call 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:1:2: >nop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:1:2: >nop
Step action: stepOver Step action: stepOver
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:2:2: >nop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:2:2: >nop
Step action: stepOut Step action: stepOut
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:9:6: >br 1 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:9:6: >br 1
Step action: stepOut Step action: stepOut
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:7:6: >set_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:7:6: >set_local 0
Step action: stepOver Step action: stepOver
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:8:6: >call 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:8:6: >call 0
Step action: stepOver Step action: stepOver
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:9:6: >br 1 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:9:6: >br 1
Step action: resume Step action: resume
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:7:6: >set_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:7:6: >set_local 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:8:6: >call 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:8:6: >call 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:1:2: >nop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:1:2: >nop
Step action: stepOut Step action: stepOut
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:9:6: >br 1 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:9:6: >br 1
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:1:2: >loop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:1:2: >loop
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:2:4: >get_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:2:4: >get_local 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:3:4: >if Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:3:4: >if
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:4:6: >get_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:4:6: >get_local 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:5:6: >i32.const 1 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:5:6: >i32.const 1
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:6:6: >i32.sub Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:6:6: >i32.sub
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:7:6: >set_local 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:7:6: >set_local 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:8:6: >call 0 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:8:6: >call 0
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:1:2: >nop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:1:2: >nop
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:2:2: >nop Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:2:2: >nop
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-0:3:0: >end Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-0:3:0: >end
Step action: stepInto Step action: stepInto
Paused at wasm://wasm/wasm-0e9ccf72/wasm-0e9ccf72-1:9:6: >br 1 Paused at wasm://wasm/wasm-0c10a5fe/wasm-0c10a5fe-1:9:6: >br 1
Step action: resume Step action: resume
exports.main returned! exports.main returned!
Finished! Finished!
...@@ -10,7 +10,7 @@ var name = 'regression_684858'; ...@@ -10,7 +10,7 @@ var name = 'regression_684858';
function patchNameLength(buffer) { function patchNameLength(buffer) {
var count = 0; var count = 0;
var view = new Uint8Array(buffer); var view = new Uint8Array(buffer);
for (var i = 0, e = view.length - name.length; i < e; ++i) { for (var i = 0, e = view.length - name.length; i <= e; ++i) {
var subs = String.fromCharCode.apply(null, view.slice(i, i + name.length)); var subs = String.fromCharCode.apply(null, view.slice(i, i + name.length));
if (subs != name) continue; if (subs != name) continue;
++count; ++count;
......
...@@ -54,16 +54,16 @@ Error.prepareStackTrace = function(error, frames) { ...@@ -54,16 +54,16 @@ Error.prepareStackTrace = function(error, frames) {
return frames; return frames;
}; };
(function testFunctionNamesAsCallSites() { (function testFunctionNamesAsCallSites() {
var names = expected_names.concat(["testFunctionNamesAsCallSites", null]); var names = expected_names.concat(['testFunctionNamesAsCallSites', null]);
try { try {
module.exports.main(); module.exports.main();
assertFalse("should throw"); assertFalse('should throw');
} catch (e) { } catch (e) {
assertEquals(names.length, e.stack.length); assertEquals(names.length, e.stack.length, 'stack length');
for (var i = 0; i < names.length; ++i) { for (var i = 0; i < names.length; ++i) {
assertEquals(names[i], e.stack[i].getFunctionName()); assertEquals(
names[i], e.stack[i].getFunctionName(), 'function name at ' + i);
} }
} }
})(); })();
...@@ -25,7 +25,8 @@ function verifyStack(frames, expected) { ...@@ -25,7 +25,8 @@ function verifyStack(frames, expected) {
assertContains(exp[4], frames[i].getFileName(), "["+i+"].getFileName()"); assertContains(exp[4], frames[i].getFileName(), "["+i+"].getFileName()");
var toString; var toString;
if (exp[0]) { if (exp[0]) {
toString = exp[1] + " (<WASM>[" + exp[2] + "]+" + exp[3] + ")"; var fun_str = exp[1] !== null ? exp[1] : "<WASM UNNAMED>";
toString = fun_str + " (<WASM>[" + exp[2] + "]+" + exp[3] + ")";
} else { } else {
toString = exp[4] + ":" + exp[2] + ":"; toString = exp[4] + ":" + exp[2] + ":";
} }
...@@ -70,10 +71,10 @@ var module = builder.instantiate({mod: {func: STACK}}); ...@@ -70,10 +71,10 @@ var module = builder.instantiate({mod: {func: STACK}});
(function testSimpleStack() { (function testSimpleStack() {
var expected_string = "Error\n" + var expected_string = "Error\n" +
// The line numbers below will change as this test gains / loses lines.. // The line numbers below will change as this test gains / loses lines..
" at STACK (stack.js:39:11)\n" + // -- " at STACK (stack.js:40:11)\n" + // --
" at main (<WASM>[1]+1)\n" + // -- " at main (<WASM>[1]+1)\n" + // --
" at testSimpleStack (stack.js:78:18)\n" + // -- " at testSimpleStack (stack.js:79:18)\n" + // --
" at stack.js:80:3"; // -- " at stack.js:81:3"; // --
module.exports.main(); module.exports.main();
assertEquals(expected_string, stripPath(stack)); assertEquals(expected_string, stripPath(stack));
...@@ -90,10 +91,10 @@ Error.prepareStackTrace = function(error, frames) { ...@@ -90,10 +91,10 @@ Error.prepareStackTrace = function(error, frames) {
verifyStack(stack, [ verifyStack(stack, [
// isWasm function line pos file // isWasm function line pos file
[ false, "STACK", 39, 0, "stack.js"], [ false, "STACK", 40, 0, "stack.js"],
[ true, "main", 1, 1, null], [ true, "main", 1, 1, null],
[ false, "testStackFrames", 89, 0, "stack.js"], [ false, "testStackFrames", 90, 0, "stack.js"],
[ false, null, 98, 0, "stack.js"] [ false, null, 99, 0, "stack.js"]
]); ]);
})(); })();
...@@ -106,8 +107,8 @@ Error.prepareStackTrace = function(error, frames) { ...@@ -106,8 +107,8 @@ Error.prepareStackTrace = function(error, frames) {
verifyStack(e.stack, [ verifyStack(e.stack, [
// isWasm function line pos file // isWasm function line pos file
[ true, "exec_unreachable", 2, 1, null], [ true, "exec_unreachable", 2, 1, null],
[ false, "testWasmUnreachable", 102, 0, "stack.js"], [ false, "testWasmUnreachable", 103, 0, "stack.js"],
[ false, null, 113, 0, "stack.js"] [ false, null, 114, 0, "stack.js"]
]); ]);
} }
})(); })();
...@@ -120,10 +121,10 @@ Error.prepareStackTrace = function(error, frames) { ...@@ -120,10 +121,10 @@ Error.prepareStackTrace = function(error, frames) {
assertContains("out of bounds", e.message); assertContains("out of bounds", e.message);
verifyStack(e.stack, [ verifyStack(e.stack, [
// isWasm function line pos file // isWasm function line pos file
[ true, "", 3, 3, null], [ true, null, 3, 3, null],
[ true, "call_mem_out_of_bounds", 4, 1, null], [ true, "call_mem_out_of_bounds", 4, 1, null],
[ false, "testWasmMemOutOfBounds", 117, 0, "stack.js"], [ false, "testWasmMemOutOfBounds", 118, 0, "stack.js"],
[ false, null, 129, 0, "stack.js"] [ false, null, 130, 0, "stack.js"]
]); ]);
} }
})(); })();
......
...@@ -65,6 +65,10 @@ let kCodeSectionCode = 10; // Function code ...@@ -65,6 +65,10 @@ let kCodeSectionCode = 10; // Function code
let kDataSectionCode = 11; // Data segments let kDataSectionCode = 11; // Data segments
let kNameSectionCode = 12; // Name section (encoded as string) let kNameSectionCode = 12; // Name section (encoded as string)
// Name section types
let kFunctionNamesCode = 1;
let kLocalNamesCode = 2;
let kWasmFunctionTypeForm = 0x60; let kWasmFunctionTypeForm = 0x60;
let kWasmAnyFunctionTypeForm = 0x70; let kWasmAnyFunctionTypeForm = 0x70;
......
...@@ -336,15 +336,16 @@ class WasmModuleBuilder { ...@@ -336,15 +336,16 @@ class WasmModuleBuilder {
} }
// Add functions declarations // Add functions declarations
let has_names = false; let num_function_names = 0;
let names = false; let names = false;
if (wasm.functions.length > 0) { if (wasm.functions.length > 0) {
if (debug) print("emitting function decls @ " + binary.length); if (debug) print("emitting function decls @ " + binary.length);
binary.emit_section(kFunctionSectionCode, section => { binary.emit_section(kFunctionSectionCode, section => {
section.emit_u32v(wasm.functions.length); section.emit_u32v(wasm.functions.length);
for (let func of wasm.functions) { for (let func of wasm.functions) {
has_names = has_names || (func.name != undefined && if (func.name !== undefined) {
func.name.length > 0); ++num_function_names;
}
section.emit_u32v(func.type_index); section.emit_u32v(func.type_index);
} }
}); });
...@@ -363,7 +364,7 @@ class WasmModuleBuilder { ...@@ -363,7 +364,7 @@ class WasmModuleBuilder {
} }
// Add memory section // Add memory section
if (wasm.memory != undefined) { if (wasm.memory !== undefined) {
if (debug) print("emitting memory @ " + binary.length); if (debug) print("emitting memory @ " + binary.length);
binary.emit_section(kMemorySectionCode, section => { binary.emit_section(kMemorySectionCode, section => {
section.emit_u8(1); // one memory entry section.emit_u8(1); // one memory entry
...@@ -424,7 +425,7 @@ class WasmModuleBuilder { ...@@ -424,7 +425,7 @@ class WasmModuleBuilder {
} }
// Add export table. // Add export table.
var mem_export = (wasm.memory != undefined && wasm.memory.exp); var mem_export = (wasm.memory !== undefined && wasm.memory.exp);
var exports_count = wasm.exports.length + (mem_export ? 1 : 0); var exports_count = wasm.exports.length + (mem_export ? 1 : 0);
if (exports_count > 0) { if (exports_count > 0) {
if (debug) print("emitting exports @ " + binary.length); if (debug) print("emitting exports @ " + binary.length);
...@@ -444,7 +445,7 @@ class WasmModuleBuilder { ...@@ -444,7 +445,7 @@ class WasmModuleBuilder {
} }
// Add start function section. // Add start function section.
if (wasm.start_index != undefined) { if (wasm.start_index !== undefined) {
if (debug) print("emitting start function @ " + binary.length); if (debug) print("emitting start function @ " + binary.length);
binary.emit_section(kStartSectionCode, section => { binary.emit_section(kStartSectionCode, section => {
section.emit_u32v(wasm.start_index); section.emit_u32v(wasm.start_index);
...@@ -485,7 +486,7 @@ class WasmModuleBuilder { ...@@ -485,7 +486,7 @@ class WasmModuleBuilder {
// Function body length will be patched later. // Function body length will be patched later.
let local_decls = []; let local_decls = [];
let l = func.locals; let l = func.locals;
if (l != undefined) { if (l !== undefined) {
let local_decls_count = 0; let local_decls_count = 0;
if (l.i32_count > 0) { if (l.i32_count > 0) {
local_decls.push({count: l.i32_count, type: kWasmI32}); local_decls.push({count: l.i32_count, type: kWasmI32});
...@@ -545,21 +546,18 @@ class WasmModuleBuilder { ...@@ -545,21 +546,18 @@ class WasmModuleBuilder {
} }
// Add function names. // Add function names.
if (has_names) { if (num_function_names > 0) {
if (debug) print("emitting names @ " + binary.length); if (debug) print('emitting names @ ' + binary.length);
binary.emit_section(kUnknownSectionCode, section => { binary.emit_section(kUnknownSectionCode, section => {
section.emit_string("name"); section.emit_string('name');
var count = wasm.functions.length + wasm.num_imported_funcs; section.emit_section(kFunctionNamesCode, name_section => {
section.emit_u32v(count); name_section.emit_u32v(num_function_names);
for (var i = 0; i < wasm.num_imported_funcs; i++) { for (let func of wasm.functions) {
section.emit_u8(0); // empty string if (func.name === undefined) continue;
section.emit_u8(0); // local names count == 0 name_section.emit_u32v(func.index);
} name_section.emit_string(func.name);
for (let func of wasm.functions) { }
var name = func.name == undefined ? "" : func.name; });
section.emit_string(name);
section.emit_u8(0); // local names count == 0
}
}); });
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment