[inspector] Introduce translation of wasm frames
This allows to show wasm source (disassembled wasm code) in DevTools. See design doc for details. More tests for the disassembly will have to follow. Also, the text format (generated by V8) will be changed. BUG=chromium:659715 R=yangguo@chromium.org, kozyatinskiy@chromium.org, titzer@chromium.org, dgozman@chromium.org Review-Url: https://codereview.chromium.org/2493773003 Cr-Commit-Position: refs/heads/master@{#41055}
This commit is contained in:
parent
32793ab33f
commit
d4a42a5f89
36
src/api.cc
36
src/api.cc
@ -8910,6 +8910,7 @@ int DebugInterface::Script::ColumnOffset() const {
|
|||||||
|
|
||||||
std::vector<int> DebugInterface::Script::LineEnds() const {
|
std::vector<int> DebugInterface::Script::LineEnds() const {
|
||||||
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
||||||
|
if (script->type() == i::Script::TYPE_WASM) return std::vector<int>();
|
||||||
i::Isolate* isolate = script->GetIsolate();
|
i::Isolate* isolate = script->GetIsolate();
|
||||||
i::HandleScope scope(isolate);
|
i::HandleScope scope(isolate);
|
||||||
i::Script::InitLineEnds(script);
|
i::Script::InitLineEnds(script);
|
||||||
@ -8973,6 +8974,10 @@ MaybeLocal<String> DebugInterface::Script::Source() const {
|
|||||||
handle_scope.CloseAndEscape(i::Handle<i::String>::cast(value)));
|
handle_scope.CloseAndEscape(i::Handle<i::String>::cast(value)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool DebugInterface::Script::IsWasm() const {
|
||||||
|
return Utils::OpenHandle(this)->type() == i::Script::TYPE_WASM;
|
||||||
|
}
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
int GetSmiValue(i::Handle<i::FixedArray> array, int index) {
|
int GetSmiValue(i::Handle<i::FixedArray> array, int index) {
|
||||||
return i::Smi::cast(array->get(index))->value();
|
return i::Smi::cast(array->get(index))->value();
|
||||||
@ -8984,6 +8989,10 @@ bool DebugInterface::Script::GetPossibleBreakpoints(
|
|||||||
std::vector<Location>* locations) const {
|
std::vector<Location>* locations) const {
|
||||||
CHECK(!start.IsEmpty());
|
CHECK(!start.IsEmpty());
|
||||||
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
||||||
|
if (script->type() == i::Script::TYPE_WASM) {
|
||||||
|
// TODO(clemensh): Return the proper thing once we support wasm breakpoints.
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
i::Script::InitLineEnds(script);
|
i::Script::InitLineEnds(script);
|
||||||
CHECK(script->line_ends()->IsFixedArray());
|
CHECK(script->line_ends()->IsFixedArray());
|
||||||
@ -9029,6 +9038,10 @@ bool DebugInterface::Script::GetPossibleBreakpoints(
|
|||||||
|
|
||||||
int DebugInterface::Script::GetSourcePosition(const Location& location) const {
|
int DebugInterface::Script::GetSourcePosition(const Location& location) const {
|
||||||
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
i::Handle<i::Script> script = Utils::OpenHandle(this);
|
||||||
|
if (script->type() == i::Script::TYPE_WASM) {
|
||||||
|
// TODO(clemensh): Return the proper thing for wasm.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int line = std::max(location.GetLineNumber() - script->line_offset(), 0);
|
int line = std::max(location.GetLineNumber() - script->line_offset(), 0);
|
||||||
int column = location.GetColumnNumber();
|
int column = location.GetColumnNumber();
|
||||||
@ -9062,7 +9075,10 @@ MaybeLocal<DebugInterface::Script> DebugInterface::Script::Wrap(
|
|||||||
return MaybeLocal<Script>();
|
return MaybeLocal<Script>();
|
||||||
}
|
}
|
||||||
i::Handle<i::Script> script_obj = i::Handle<i::Script>::cast(script_value);
|
i::Handle<i::Script> script_obj = i::Handle<i::Script>::cast(script_value);
|
||||||
if (script_obj->type() != i::Script::TYPE_NORMAL) return MaybeLocal<Script>();
|
if (script_obj->type() != i::Script::TYPE_NORMAL &&
|
||||||
|
script_obj->type() != i::Script::TYPE_WASM) {
|
||||||
|
return MaybeLocal<Script>();
|
||||||
|
}
|
||||||
return ToApiHandle<DebugInterface::Script>(
|
return ToApiHandle<DebugInterface::Script>(
|
||||||
handle_scope.CloseAndEscape(script_obj));
|
handle_scope.CloseAndEscape(script_obj));
|
||||||
}
|
}
|
||||||
@ -9112,6 +9128,24 @@ void DebugInterface::GetLoadedScripts(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::pair<std::string, std::vector<std::tuple<uint32_t, int, int>>>
|
||||||
|
DebugInterface::DisassembleWasmFunction(Isolate* v8_isolate,
|
||||||
|
Local<Object> v8_script,
|
||||||
|
int function_index) {
|
||||||
|
i::Isolate* isolate = reinterpret_cast<i::Isolate*>(v8_isolate);
|
||||||
|
if (v8_script.IsEmpty()) return {};
|
||||||
|
i::Handle<i::Object> script_wrapper = Utils::OpenHandle(*v8_script);
|
||||||
|
if (!script_wrapper->IsJSValue()) return {};
|
||||||
|
i::Handle<i::Object> script_obj(
|
||||||
|
i::Handle<i::JSValue>::cast(script_wrapper)->value(), isolate);
|
||||||
|
if (!script_obj->IsScript()) return {};
|
||||||
|
i::Handle<i::Script> script = i::Handle<i::Script>::cast(script_obj);
|
||||||
|
if (script->type() != i::Script::TYPE_WASM) return {};
|
||||||
|
i::Handle<i::WasmCompiledModule> compiled_module(
|
||||||
|
i::WasmCompiledModule::cast(script->wasm_compiled_module()), isolate);
|
||||||
|
return i::wasm::DisassembleFunction(compiled_module, function_index);
|
||||||
|
}
|
||||||
|
|
||||||
Local<String> CpuProfileNode::GetFunctionName() const {
|
Local<String> CpuProfileNode::GetFunctionName() const {
|
||||||
const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
|
const i::ProfileNode* node = reinterpret_cast<const i::ProfileNode*>(this);
|
||||||
i::Isolate* isolate = node->isolate();
|
i::Isolate* isolate = node->isolate();
|
||||||
|
@ -179,6 +179,7 @@ class DebugInterface {
|
|||||||
MaybeLocal<String> SourceMappingURL() const;
|
MaybeLocal<String> SourceMappingURL() const;
|
||||||
MaybeLocal<String> ContextData() const;
|
MaybeLocal<String> ContextData() const;
|
||||||
MaybeLocal<String> Source() const;
|
MaybeLocal<String> Source() const;
|
||||||
|
bool IsWasm() const;
|
||||||
bool GetPossibleBreakpoints(const Location& start, const Location& end,
|
bool GetPossibleBreakpoints(const Location& start, const Location& end,
|
||||||
std::vector<Location>* locations) const;
|
std::vector<Location>* locations) const;
|
||||||
|
|
||||||
@ -202,6 +203,16 @@ class DebugInterface {
|
|||||||
*/
|
*/
|
||||||
static void GetLoadedScripts(Isolate* isolate,
|
static void GetLoadedScripts(Isolate* isolate,
|
||||||
PersistentValueVector<Script>& scripts);
|
PersistentValueVector<Script>& scripts);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute the disassembly of a wasm function.
|
||||||
|
* Returns the disassembly string and a list of <byte_offset, line, column>
|
||||||
|
* entries, mapping wasm byte offsets to line and column in the disassembly.
|
||||||
|
* The list is guaranteed to be ordered by the byte_offset.
|
||||||
|
*/
|
||||||
|
static std::pair<std::string, std::vector<std::tuple<uint32_t, int, int>>>
|
||||||
|
DisassembleWasmFunction(Isolate* isolate, v8::Local<v8::Object> script,
|
||||||
|
int function_index);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace v8
|
} // namespace v8
|
||||||
|
@ -186,5 +186,7 @@ v8_source_set("inspector") {
|
|||||||
"v8-stack-trace-impl.h",
|
"v8-stack-trace-impl.h",
|
||||||
"v8-value-copier.cc",
|
"v8-value-copier.cc",
|
||||||
"v8-value-copier.h",
|
"v8-value-copier.h",
|
||||||
|
"wasm-translation.cc",
|
||||||
|
"wasm-translation.h",
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -90,6 +90,8 @@
|
|||||||
'inspector/v8-stack-trace-impl.h',
|
'inspector/v8-stack-trace-impl.h',
|
||||||
'inspector/v8-value-copier.cc',
|
'inspector/v8-value-copier.cc',
|
||||||
'inspector/v8-value-copier.h',
|
'inspector/v8-value-copier.h',
|
||||||
|
'inspector/wasm-translation.cc',
|
||||||
|
'inspector/wasm-translation.h',
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -60,8 +60,29 @@ static const char kDebuggerNotEnabled[] = "Debugger agent is not enabled";
|
|||||||
static const char kDebuggerNotPaused[] =
|
static const char kDebuggerNotPaused[] =
|
||||||
"Can only perform operation while paused.";
|
"Can only perform operation while paused.";
|
||||||
|
|
||||||
static String16 breakpointIdSuffix(
|
namespace {
|
||||||
V8DebuggerAgentImpl::BreakpointSource source) {
|
|
||||||
|
void TranslateWasmStackTraceLocations(Array<CallFrame>* stackTrace,
|
||||||
|
WasmTranslation* wasmTranslation,
|
||||||
|
int context_group_id) {
|
||||||
|
for (size_t i = 0, e = stackTrace->length(); i != e; ++i) {
|
||||||
|
protocol::Debugger::Location* location = stackTrace->get(i)->getLocation();
|
||||||
|
String16 scriptId = location->getScriptId();
|
||||||
|
int lineNumber = location->getLineNumber();
|
||||||
|
int columnNumber = location->getColumnNumber(-1);
|
||||||
|
|
||||||
|
if (!wasmTranslation->TranslateWasmScriptLocationToProtocolLocation(
|
||||||
|
&scriptId, &lineNumber, &columnNumber, context_group_id)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
location->setScriptId(std::move(scriptId));
|
||||||
|
location->setLineNumber(lineNumber);
|
||||||
|
location->setColumnNumber(columnNumber);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String16 breakpointIdSuffix(V8DebuggerAgentImpl::BreakpointSource source) {
|
||||||
switch (source) {
|
switch (source) {
|
||||||
case V8DebuggerAgentImpl::UserBreakpointSource:
|
case V8DebuggerAgentImpl::UserBreakpointSource:
|
||||||
break;
|
break;
|
||||||
@ -73,9 +94,8 @@ static String16 breakpointIdSuffix(
|
|||||||
return String16();
|
return String16();
|
||||||
}
|
}
|
||||||
|
|
||||||
static String16 generateBreakpointId(
|
String16 generateBreakpointId(const ScriptBreakpoint& breakpoint,
|
||||||
const ScriptBreakpoint& breakpoint,
|
V8DebuggerAgentImpl::BreakpointSource source) {
|
||||||
V8DebuggerAgentImpl::BreakpointSource source) {
|
|
||||||
String16Builder builder;
|
String16Builder builder;
|
||||||
builder.append(breakpoint.script_id);
|
builder.append(breakpoint.script_id);
|
||||||
builder.append(':');
|
builder.append(':');
|
||||||
@ -86,13 +106,13 @@ static String16 generateBreakpointId(
|
|||||||
return builder.toString();
|
return builder.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool positionComparator(const std::pair<int, int>& a,
|
bool positionComparator(const std::pair<int, int>& a,
|
||||||
const std::pair<int, int>& b) {
|
const std::pair<int, int>& b) {
|
||||||
if (a.first != b.first) return a.first < b.first;
|
if (a.first != b.first) return a.first < b.first;
|
||||||
return a.second < b.second;
|
return a.second < b.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::unique_ptr<protocol::Debugger::Location> buildProtocolLocation(
|
std::unique_ptr<protocol::Debugger::Location> buildProtocolLocation(
|
||||||
const String16& scriptId, int lineNumber, int columnNumber) {
|
const String16& scriptId, int lineNumber, int columnNumber) {
|
||||||
return protocol::Debugger::Location::create()
|
return protocol::Debugger::Location::create()
|
||||||
.setScriptId(scriptId)
|
.setScriptId(scriptId)
|
||||||
@ -101,6 +121,8 @@ static std::unique_ptr<protocol::Debugger::Location> buildProtocolLocation(
|
|||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
V8DebuggerAgentImpl::V8DebuggerAgentImpl(
|
V8DebuggerAgentImpl::V8DebuggerAgentImpl(
|
||||||
V8InspectorSessionImpl* session, protocol::FrontendChannel* frontendChannel,
|
V8InspectorSessionImpl* session, protocol::FrontendChannel* frontendChannel,
|
||||||
protocol::DictionaryValue* state)
|
protocol::DictionaryValue* state)
|
||||||
@ -503,10 +525,18 @@ V8DebuggerAgentImpl::resolveBreakpoint(const String16& breakpointId,
|
|||||||
scriptIterator->second->endLine() < breakpoint.line_number)
|
scriptIterator->second->endLine() < breakpoint.line_number)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
ScriptBreakpoint translatedBreakpoint = breakpoint;
|
||||||
|
if (m_scripts.count(breakpoint.script_id) == 0) {
|
||||||
|
m_debugger->wasmTranslation()
|
||||||
|
->TranslateProtocolLocationToWasmScriptLocation(
|
||||||
|
&translatedBreakpoint.script_id, &translatedBreakpoint.line_number,
|
||||||
|
&translatedBreakpoint.column_number);
|
||||||
|
}
|
||||||
|
|
||||||
int actualLineNumber;
|
int actualLineNumber;
|
||||||
int actualColumnNumber;
|
int actualColumnNumber;
|
||||||
String16 debuggerBreakpointId = m_debugger->setBreakpoint(
|
String16 debuggerBreakpointId = m_debugger->setBreakpoint(
|
||||||
breakpoint, &actualLineNumber, &actualColumnNumber);
|
translatedBreakpoint, &actualLineNumber, &actualColumnNumber);
|
||||||
if (debuggerBreakpointId.isEmpty()) return nullptr;
|
if (debuggerBreakpointId.isEmpty()) return nullptr;
|
||||||
|
|
||||||
m_serverBreakpoints[debuggerBreakpointId] =
|
m_serverBreakpoints[debuggerBreakpointId] =
|
||||||
@ -515,7 +545,7 @@ V8DebuggerAgentImpl::resolveBreakpoint(const String16& breakpointId,
|
|||||||
|
|
||||||
m_breakpointIdToDebuggerBreakpointIds[breakpointId].push_back(
|
m_breakpointIdToDebuggerBreakpointIds[breakpointId].push_back(
|
||||||
debuggerBreakpointId);
|
debuggerBreakpointId);
|
||||||
return buildProtocolLocation(breakpoint.script_id, actualLineNumber,
|
return buildProtocolLocation(translatedBreakpoint.script_id, actualLineNumber,
|
||||||
actualColumnNumber);
|
actualColumnNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -529,9 +559,8 @@ Response V8DebuggerAgentImpl::searchInContent(
|
|||||||
return Response::Error("No script for id: " + scriptId);
|
return Response::Error("No script for id: " + scriptId);
|
||||||
|
|
||||||
std::vector<std::unique_ptr<protocol::Debugger::SearchMatch>> matches =
|
std::vector<std::unique_ptr<protocol::Debugger::SearchMatch>> matches =
|
||||||
searchInTextByLinesImpl(m_session,
|
searchInTextByLinesImpl(m_session, it->second->source(m_isolate), query,
|
||||||
toProtocolString(it->second->source(m_isolate)),
|
optionalCaseSensitive.fromMaybe(false),
|
||||||
query, optionalCaseSensitive.fromMaybe(false),
|
|
||||||
optionalIsRegex.fromMaybe(false));
|
optionalIsRegex.fromMaybe(false));
|
||||||
*results = protocol::Array<protocol::Debugger::SearchMatch>::create();
|
*results = protocol::Array<protocol::Debugger::SearchMatch>::create();
|
||||||
for (size_t i = 0; i < matches.size(); ++i)
|
for (size_t i = 0; i < matches.size(); ++i)
|
||||||
@ -602,7 +631,7 @@ Response V8DebuggerAgentImpl::getScriptSource(const String16& scriptId,
|
|||||||
if (it == m_scripts.end())
|
if (it == m_scripts.end())
|
||||||
return Response::Error("No script for id: " + scriptId);
|
return Response::Error("No script for id: " + scriptId);
|
||||||
v8::HandleScope handles(m_isolate);
|
v8::HandleScope handles(m_isolate);
|
||||||
*scriptSource = toProtocolString(it->second->source(m_isolate));
|
*scriptSource = it->second->source(m_isolate);
|
||||||
return Response::OK();
|
return Response::OK();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1004,6 +1033,8 @@ Response V8DebuggerAgentImpl::currentCallFrames(
|
|||||||
protocol::ErrorSupport errorSupport;
|
protocol::ErrorSupport errorSupport;
|
||||||
*result = Array<CallFrame>::parse(protocolValue.get(), &errorSupport);
|
*result = Array<CallFrame>::parse(protocolValue.get(), &errorSupport);
|
||||||
if (!*result) return Response::Error(errorSupport.errors());
|
if (!*result) return Response::Error(errorSupport.errors());
|
||||||
|
TranslateWasmStackTraceLocations(result->get(), m_debugger->wasmTranslation(),
|
||||||
|
m_session->contextGroupId());
|
||||||
return Response::OK();
|
return Response::OK();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1017,7 +1048,7 @@ std::unique_ptr<StackTrace> V8DebuggerAgentImpl::currentAsyncStackTrace() {
|
|||||||
void V8DebuggerAgentImpl::didParseSource(
|
void V8DebuggerAgentImpl::didParseSource(
|
||||||
std::unique_ptr<V8DebuggerScript> script, bool success) {
|
std::unique_ptr<V8DebuggerScript> script, bool success) {
|
||||||
v8::HandleScope handles(m_isolate);
|
v8::HandleScope handles(m_isolate);
|
||||||
String16 scriptSource = toProtocolString(script->source(m_isolate));
|
String16 scriptSource = script->source(m_isolate);
|
||||||
if (!success) script->setSourceURL(findSourceURL(scriptSource, false));
|
if (!success) script->setSourceURL(findSourceURL(scriptSource, false));
|
||||||
if (!success)
|
if (!success)
|
||||||
script->setSourceMappingURL(findSourceMapURL(scriptSource, false));
|
script->setSourceMappingURL(findSourceMapURL(scriptSource, false));
|
||||||
@ -1046,14 +1077,14 @@ void V8DebuggerAgentImpl::didParseSource(
|
|||||||
m_frontend.scriptParsed(
|
m_frontend.scriptParsed(
|
||||||
scriptId, scriptURL, scriptRef->startLine(), scriptRef->startColumn(),
|
scriptId, scriptURL, scriptRef->startLine(), scriptRef->startColumn(),
|
||||||
scriptRef->endLine(), scriptRef->endColumn(),
|
scriptRef->endLine(), scriptRef->endColumn(),
|
||||||
scriptRef->executionContextId(), scriptRef->hash(),
|
scriptRef->executionContextId(), scriptRef->hash(m_isolate),
|
||||||
std::move(executionContextAuxDataParam), isLiveEditParam,
|
std::move(executionContextAuxDataParam), isLiveEditParam,
|
||||||
std::move(sourceMapURLParam), hasSourceURLParam);
|
std::move(sourceMapURLParam), hasSourceURLParam);
|
||||||
else
|
else
|
||||||
m_frontend.scriptFailedToParse(
|
m_frontend.scriptFailedToParse(
|
||||||
scriptId, scriptURL, scriptRef->startLine(), scriptRef->startColumn(),
|
scriptId, scriptURL, scriptRef->startLine(), scriptRef->startColumn(),
|
||||||
scriptRef->endLine(), scriptRef->endColumn(),
|
scriptRef->endLine(), scriptRef->endColumn(),
|
||||||
scriptRef->executionContextId(), scriptRef->hash(),
|
scriptRef->executionContextId(), scriptRef->hash(m_isolate),
|
||||||
std::move(executionContextAuxDataParam), std::move(sourceMapURLParam),
|
std::move(executionContextAuxDataParam), std::move(sourceMapURLParam),
|
||||||
hasSourceURLParam);
|
hasSourceURLParam);
|
||||||
|
|
||||||
|
@ -116,9 +116,8 @@ V8DebuggerScript::V8DebuggerScript(v8::Isolate* isolate,
|
|||||||
m_isLiveEdit = isLiveEdit;
|
m_isLiveEdit = isLiveEdit;
|
||||||
|
|
||||||
if (script->Source().ToLocal(&tmp)) {
|
if (script->Source().ToLocal(&tmp)) {
|
||||||
m_source.Reset(m_isolate, tmp);
|
m_sourceObj.Reset(m_isolate, tmp);
|
||||||
String16 source = toProtocolString(tmp);
|
String16 source = toProtocolString(tmp);
|
||||||
m_hash = calculateHash(source);
|
|
||||||
// V8 will not count last line if script source ends with \n.
|
// V8 will not count last line if script source ends with \n.
|
||||||
if (source.length() > 1 && source[source.length() - 1] == '\n') {
|
if (source.length() > 1 && source[source.length() - 1] == '\n') {
|
||||||
m_endLine++;
|
m_endLine++;
|
||||||
@ -129,14 +128,35 @@ V8DebuggerScript::V8DebuggerScript(v8::Isolate* isolate,
|
|||||||
m_script.Reset(m_isolate, script);
|
m_script.Reset(m_isolate, script);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
V8DebuggerScript::V8DebuggerScript(String16 id, String16 url, String16 source)
|
||||||
|
: m_id(std::move(id)), m_url(std::move(url)), m_source(std::move(source)) {
|
||||||
|
int num_lines = 0;
|
||||||
|
int last_newline = -1;
|
||||||
|
size_t next_newline = m_source.find('\n', last_newline + 1);
|
||||||
|
while (next_newline != String16::kNotFound) {
|
||||||
|
last_newline = static_cast<int>(next_newline);
|
||||||
|
next_newline = m_source.find('\n', last_newline + 1);
|
||||||
|
++num_lines;
|
||||||
|
}
|
||||||
|
m_endLine = num_lines;
|
||||||
|
m_endColumn = static_cast<int>(m_source.length()) - last_newline - 1;
|
||||||
|
}
|
||||||
|
|
||||||
V8DebuggerScript::~V8DebuggerScript() {}
|
V8DebuggerScript::~V8DebuggerScript() {}
|
||||||
|
|
||||||
const String16& V8DebuggerScript::sourceURL() const {
|
const String16& V8DebuggerScript::sourceURL() const {
|
||||||
return m_sourceURL.isEmpty() ? m_url : m_sourceURL;
|
return m_sourceURL.isEmpty() ? m_url : m_sourceURL;
|
||||||
}
|
}
|
||||||
|
|
||||||
v8::Local<v8::String> V8DebuggerScript::source(v8::Isolate* isolate) const {
|
String16 V8DebuggerScript::source(v8::Isolate* isolate) const {
|
||||||
return m_source.Get(isolate);
|
if (m_sourceObj.IsEmpty()) return m_source;
|
||||||
|
return toProtocolString(m_sourceObj.Get(isolate));
|
||||||
|
}
|
||||||
|
|
||||||
|
const String16& V8DebuggerScript::hash(v8::Isolate* isolate) const {
|
||||||
|
if (m_hash.isEmpty()) m_hash = calculateHash(source(isolate));
|
||||||
|
DCHECK(!m_hash.isEmpty());
|
||||||
|
return m_hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
void V8DebuggerScript::setSourceURL(const String16& sourceURL) {
|
void V8DebuggerScript::setSourceURL(const String16& sourceURL) {
|
||||||
@ -148,8 +168,9 @@ void V8DebuggerScript::setSourceMappingURL(const String16& sourceMappingURL) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void V8DebuggerScript::setSource(v8::Local<v8::String> source) {
|
void V8DebuggerScript::setSource(v8::Local<v8::String> source) {
|
||||||
m_source.Reset(m_isolate, source);
|
m_source = String16();
|
||||||
m_hash = calculateHash(toProtocolString(source));
|
m_sourceObj.Reset(m_isolate, source);
|
||||||
|
m_hash = String16();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool V8DebuggerScript::getPossibleBreakpoints(
|
bool V8DebuggerScript::getPossibleBreakpoints(
|
||||||
|
@ -43,6 +43,7 @@ class V8DebuggerScript {
|
|||||||
V8DebuggerScript(v8::Isolate* isolate,
|
V8DebuggerScript(v8::Isolate* isolate,
|
||||||
v8::Local<v8::DebugInterface::Script> script,
|
v8::Local<v8::DebugInterface::Script> script,
|
||||||
bool isLiveEdit);
|
bool isLiveEdit);
|
||||||
|
V8DebuggerScript(String16 id, String16 url, String16 source);
|
||||||
~V8DebuggerScript();
|
~V8DebuggerScript();
|
||||||
|
|
||||||
const String16& scriptId() const { return m_id; }
|
const String16& scriptId() const { return m_id; }
|
||||||
@ -50,8 +51,8 @@ class V8DebuggerScript {
|
|||||||
bool hasSourceURL() const { return !m_sourceURL.isEmpty(); }
|
bool hasSourceURL() const { return !m_sourceURL.isEmpty(); }
|
||||||
const String16& sourceURL() const;
|
const String16& sourceURL() const;
|
||||||
const String16& sourceMappingURL() const { return m_sourceMappingURL; }
|
const String16& sourceMappingURL() const { return m_sourceMappingURL; }
|
||||||
v8::Local<v8::String> source(v8::Isolate*) const;
|
String16 source(v8::Isolate*) const;
|
||||||
const String16& hash() const { return m_hash; }
|
const String16& hash(v8::Isolate*) const;
|
||||||
int startLine() const { return m_startLine; }
|
int startLine() const { return m_startLine; }
|
||||||
int startColumn() const { return m_startColumn; }
|
int startColumn() const { return m_startColumn; }
|
||||||
int endLine() const { return m_endLine; }
|
int endLine() const { return m_endLine; }
|
||||||
@ -76,15 +77,16 @@ class V8DebuggerScript {
|
|||||||
String16 m_url;
|
String16 m_url;
|
||||||
String16 m_sourceURL;
|
String16 m_sourceURL;
|
||||||
String16 m_sourceMappingURL;
|
String16 m_sourceMappingURL;
|
||||||
v8::Global<v8::String> m_source;
|
v8::Global<v8::String> m_sourceObj;
|
||||||
String16 m_hash;
|
String16 m_source;
|
||||||
int m_startLine;
|
mutable String16 m_hash;
|
||||||
int m_startColumn;
|
int m_startLine = 0;
|
||||||
int m_endLine;
|
int m_startColumn = 0;
|
||||||
int m_endColumn;
|
int m_endLine = 0;
|
||||||
int m_executionContextId;
|
int m_endColumn = 0;
|
||||||
|
int m_executionContextId = 0;
|
||||||
String16 m_executionContextAuxData;
|
String16 m_executionContextAuxData;
|
||||||
bool m_isLiveEdit;
|
bool m_isLiveEdit = false;
|
||||||
|
|
||||||
v8::Isolate* m_isolate;
|
v8::Isolate* m_isolate;
|
||||||
v8::Global<v8::DebugInterface::Script> m_script;
|
v8::Global<v8::DebugInterface::Script> m_script;
|
||||||
|
@ -56,7 +56,8 @@ V8Debugger::V8Debugger(v8::Isolate* isolate, V8InspectorImpl* inspector)
|
|||||||
m_runningNestedMessageLoop(false),
|
m_runningNestedMessageLoop(false),
|
||||||
m_ignoreScriptParsedEventsCounter(0),
|
m_ignoreScriptParsedEventsCounter(0),
|
||||||
m_maxAsyncCallStackDepth(0),
|
m_maxAsyncCallStackDepth(0),
|
||||||
m_pauseOnExceptionsState(v8::DebugInterface::NoBreakOnException) {}
|
m_pauseOnExceptionsState(v8::DebugInterface::NoBreakOnException),
|
||||||
|
m_wasmTranslation(isolate, this) {}
|
||||||
|
|
||||||
V8Debugger::~V8Debugger() {}
|
V8Debugger::~V8Debugger() {}
|
||||||
|
|
||||||
@ -82,6 +83,7 @@ void V8Debugger::disable() {
|
|||||||
m_debuggerScript.Reset();
|
m_debuggerScript.Reset();
|
||||||
m_debuggerContext.Reset();
|
m_debuggerContext.Reset();
|
||||||
allAsyncTasksCanceled();
|
allAsyncTasksCanceled();
|
||||||
|
m_wasmTranslation.Clear();
|
||||||
v8::DebugInterface::SetDebugEventListener(m_isolate, nullptr);
|
v8::DebugInterface::SetDebugEventListener(m_isolate, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -574,12 +576,25 @@ void V8Debugger::handleV8DebugEvent(
|
|||||||
|
|
||||||
V8DebuggerAgentImpl* agent =
|
V8DebuggerAgentImpl* agent =
|
||||||
m_inspector->enabledDebuggerAgentForGroup(getGroupId(eventContext));
|
m_inspector->enabledDebuggerAgentForGroup(getGroupId(eventContext));
|
||||||
if (agent) {
|
if (!agent) return;
|
||||||
v8::HandleScope scope(m_isolate);
|
|
||||||
if (m_ignoreScriptParsedEventsCounter == 0 &&
|
v8::HandleScope scope(m_isolate);
|
||||||
(event == v8::AfterCompile || event == v8::CompileError)) {
|
if (event == v8::AfterCompile || event == v8::CompileError) {
|
||||||
v8::Local<v8::Context> context = debuggerContext();
|
v8::Context::Scope contextScope(debuggerContext());
|
||||||
v8::Context::Scope contextScope(context);
|
// Determine if the script is a wasm script.
|
||||||
|
v8::Local<v8::Value> scriptMirror =
|
||||||
|
callInternalGetterFunction(eventDetails.GetEventData(), "script");
|
||||||
|
DCHECK(scriptMirror->IsObject());
|
||||||
|
v8::Local<v8::Value> scriptWrapper =
|
||||||
|
callInternalGetterFunction(scriptMirror.As<v8::Object>(), "value");
|
||||||
|
DCHECK(scriptWrapper->IsObject());
|
||||||
|
v8::Local<v8::DebugInterface::Script> script =
|
||||||
|
v8::DebugInterface::Script::Wrap(m_isolate,
|
||||||
|
scriptWrapper.As<v8::Object>())
|
||||||
|
.ToLocalChecked();
|
||||||
|
if (script->IsWasm()) {
|
||||||
|
m_wasmTranslation.AddScript(scriptWrapper.As<v8::Object>());
|
||||||
|
} else if (m_ignoreScriptParsedEventsCounter == 0) {
|
||||||
v8::Local<v8::Value> argv[] = {eventDetails.GetEventData()};
|
v8::Local<v8::Value> argv[] = {eventDetails.GetEventData()};
|
||||||
v8::Local<v8::Value> value =
|
v8::Local<v8::Value> value =
|
||||||
callDebuggerMethod("getAfterCompileScript", 1, argv).ToLocalChecked();
|
callDebuggerMethod("getAfterCompileScript", 1, argv).ToLocalChecked();
|
||||||
@ -593,29 +608,28 @@ void V8Debugger::handleV8DebugEvent(
|
|||||||
agent->didParseSource(
|
agent->didParseSource(
|
||||||
wrapUnique(new V8DebuggerScript(m_isolate, script, inLiveEditScope)),
|
wrapUnique(new V8DebuggerScript(m_isolate, script, inLiveEditScope)),
|
||||||
event == v8::AfterCompile);
|
event == v8::AfterCompile);
|
||||||
} else if (event == v8::Exception) {
|
|
||||||
v8::Local<v8::Context> context = debuggerContext();
|
|
||||||
v8::Local<v8::Object> eventData = eventDetails.GetEventData();
|
|
||||||
v8::Local<v8::Value> exception =
|
|
||||||
callInternalGetterFunction(eventData, "exception");
|
|
||||||
v8::Local<v8::Value> promise =
|
|
||||||
callInternalGetterFunction(eventData, "promise");
|
|
||||||
bool isPromiseRejection = !promise.IsEmpty() && promise->IsObject();
|
|
||||||
v8::Local<v8::Value> uncaught =
|
|
||||||
callInternalGetterFunction(eventData, "uncaught");
|
|
||||||
bool isUncaught = uncaught->BooleanValue(context).FromJust();
|
|
||||||
handleProgramBreak(eventContext, eventDetails.GetExecutionState(),
|
|
||||||
exception, v8::Local<v8::Array>(), isPromiseRejection,
|
|
||||||
isUncaught);
|
|
||||||
} else if (event == v8::Break) {
|
|
||||||
v8::Local<v8::Value> argv[] = {eventDetails.GetEventData()};
|
|
||||||
v8::Local<v8::Value> hitBreakpoints =
|
|
||||||
callDebuggerMethod("getBreakpointNumbers", 1, argv).ToLocalChecked();
|
|
||||||
DCHECK(hitBreakpoints->IsArray());
|
|
||||||
handleProgramBreak(eventContext, eventDetails.GetExecutionState(),
|
|
||||||
v8::Local<v8::Value>(),
|
|
||||||
hitBreakpoints.As<v8::Array>());
|
|
||||||
}
|
}
|
||||||
|
} else if (event == v8::Exception) {
|
||||||
|
v8::Local<v8::Context> context = debuggerContext();
|
||||||
|
v8::Local<v8::Object> eventData = eventDetails.GetEventData();
|
||||||
|
v8::Local<v8::Value> exception =
|
||||||
|
callInternalGetterFunction(eventData, "exception");
|
||||||
|
v8::Local<v8::Value> promise =
|
||||||
|
callInternalGetterFunction(eventData, "promise");
|
||||||
|
bool isPromiseRejection = !promise.IsEmpty() && promise->IsObject();
|
||||||
|
v8::Local<v8::Value> uncaught =
|
||||||
|
callInternalGetterFunction(eventData, "uncaught");
|
||||||
|
bool isUncaught = uncaught->BooleanValue(context).FromJust();
|
||||||
|
handleProgramBreak(eventContext, eventDetails.GetExecutionState(),
|
||||||
|
exception, v8::Local<v8::Array>(), isPromiseRejection,
|
||||||
|
isUncaught);
|
||||||
|
} else if (event == v8::Break) {
|
||||||
|
v8::Local<v8::Value> argv[] = {eventDetails.GetEventData()};
|
||||||
|
v8::Local<v8::Value> hitBreakpoints =
|
||||||
|
callDebuggerMethod("getBreakpointNumbers", 1, argv).ToLocalChecked();
|
||||||
|
DCHECK(hitBreakpoints->IsArray());
|
||||||
|
handleProgramBreak(eventContext, eventDetails.GetExecutionState(),
|
||||||
|
v8::Local<v8::Value>(), hitBreakpoints.As<v8::Array>());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
#include "src/inspector/protocol/Forward.h"
|
#include "src/inspector/protocol/Forward.h"
|
||||||
#include "src/inspector/protocol/Runtime.h"
|
#include "src/inspector/protocol/Runtime.h"
|
||||||
#include "src/inspector/v8-debugger-script.h"
|
#include "src/inspector/v8-debugger-script.h"
|
||||||
|
#include "src/inspector/wasm-translation.h"
|
||||||
|
|
||||||
#include "include/v8-inspector.h"
|
#include "include/v8-inspector.h"
|
||||||
|
|
||||||
@ -94,6 +95,8 @@ class V8Debugger {
|
|||||||
|
|
||||||
V8InspectorImpl* inspector() { return m_inspector; }
|
V8InspectorImpl* inspector() { return m_inspector; }
|
||||||
|
|
||||||
|
WasmTranslation* wasmTranslation() { return &m_wasmTranslation; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void compileDebuggerScript();
|
void compileDebuggerScript();
|
||||||
v8::MaybeLocal<v8::Value> callDebuggerMethod(const char* functionName,
|
v8::MaybeLocal<v8::Value> callDebuggerMethod(const char* functionName,
|
||||||
@ -149,6 +152,8 @@ class V8Debugger {
|
|||||||
|
|
||||||
v8::DebugInterface::ExceptionBreakState m_pauseOnExceptionsState;
|
v8::DebugInterface::ExceptionBreakState m_pauseOnExceptionsState;
|
||||||
|
|
||||||
|
WasmTranslation m_wasmTranslation;
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(V8Debugger);
|
DISALLOW_COPY_AND_ASSIGN(V8Debugger);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -260,6 +260,7 @@ void V8InspectorImpl::resetContextGroup(int contextGroupId) {
|
|||||||
SessionMap::iterator session = m_sessions.find(contextGroupId);
|
SessionMap::iterator session = m_sessions.find(contextGroupId);
|
||||||
if (session != m_sessions.end()) session->second->reset();
|
if (session != m_sessions.end()) session->second->reset();
|
||||||
m_contexts.erase(contextGroupId);
|
m_contexts.erase(contextGroupId);
|
||||||
|
m_debugger->wasmTranslation()->Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void V8InspectorImpl::willExecuteScript(v8::Local<v8::Context> context,
|
void V8InspectorImpl::willExecuteScript(v8::Local<v8::Context> context,
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
#include "src/inspector/v8-stack-trace-impl.h"
|
#include "src/inspector/v8-stack-trace-impl.h"
|
||||||
|
|
||||||
#include "src/inspector/string-util.h"
|
#include "src/inspector/string-util.h"
|
||||||
|
#include "src/inspector/v8-debugger-agent-impl.h"
|
||||||
#include "src/inspector/v8-debugger.h"
|
#include "src/inspector/v8-debugger.h"
|
||||||
#include "src/inspector/v8-inspector-impl.h"
|
#include "src/inspector/v8-inspector-impl.h"
|
||||||
|
|
||||||
@ -21,7 +22,9 @@ static const v8::StackTrace::StackTraceOptions stackTraceOptions =
|
|||||||
v8::StackTrace::kScriptId | v8::StackTrace::kScriptNameOrSourceURL |
|
v8::StackTrace::kScriptId | v8::StackTrace::kScriptNameOrSourceURL |
|
||||||
v8::StackTrace::kFunctionName);
|
v8::StackTrace::kFunctionName);
|
||||||
|
|
||||||
V8StackTraceImpl::Frame toFrame(v8::Local<v8::StackFrame> frame) {
|
V8StackTraceImpl::Frame toFrame(v8::Local<v8::StackFrame> frame,
|
||||||
|
WasmTranslation* wasmTranslation,
|
||||||
|
int contextGroupId) {
|
||||||
String16 scriptId = String16::fromInteger(frame->GetScriptId());
|
String16 scriptId = String16::fromInteger(frame->GetScriptId());
|
||||||
String16 sourceName;
|
String16 sourceName;
|
||||||
v8::Local<v8::String> sourceNameValue(frame->GetScriptNameOrSourceURL());
|
v8::Local<v8::String> sourceNameValue(frame->GetScriptNameOrSourceURL());
|
||||||
@ -33,22 +36,30 @@ V8StackTraceImpl::Frame toFrame(v8::Local<v8::StackFrame> frame) {
|
|||||||
if (!functionNameValue.IsEmpty())
|
if (!functionNameValue.IsEmpty())
|
||||||
functionName = toProtocolString(functionNameValue);
|
functionName = toProtocolString(functionNameValue);
|
||||||
|
|
||||||
int sourceLineNumber = frame->GetLineNumber();
|
int sourceLineNumber = frame->GetLineNumber() - 1;
|
||||||
int sourceColumn = frame->GetColumn();
|
int sourceColumn = frame->GetColumn() - 1;
|
||||||
|
// TODO(clemensh): Figure out a way to do this translation only right before
|
||||||
|
// sending the stack trace over wire.
|
||||||
|
if (wasmTranslation)
|
||||||
|
wasmTranslation->TranslateWasmScriptLocationToProtocolLocation(
|
||||||
|
&scriptId, &sourceLineNumber, &sourceColumn, contextGroupId);
|
||||||
return V8StackTraceImpl::Frame(functionName, scriptId, sourceName,
|
return V8StackTraceImpl::Frame(functionName, scriptId, sourceName,
|
||||||
sourceLineNumber, sourceColumn);
|
sourceLineNumber + 1, sourceColumn + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void toFramesVector(v8::Local<v8::StackTrace> stackTrace,
|
void toFramesVector(v8::Local<v8::StackTrace> stackTrace,
|
||||||
std::vector<V8StackTraceImpl::Frame>& frames,
|
std::vector<V8StackTraceImpl::Frame>& frames,
|
||||||
size_t maxStackSize, v8::Isolate* isolate) {
|
size_t maxStackSize, v8::Isolate* isolate,
|
||||||
|
V8Debugger* debugger, int contextGroupId) {
|
||||||
DCHECK(isolate->InContext());
|
DCHECK(isolate->InContext());
|
||||||
int frameCount = stackTrace->GetFrameCount();
|
int frameCount = stackTrace->GetFrameCount();
|
||||||
if (frameCount > static_cast<int>(maxStackSize))
|
if (frameCount > static_cast<int>(maxStackSize))
|
||||||
frameCount = static_cast<int>(maxStackSize);
|
frameCount = static_cast<int>(maxStackSize);
|
||||||
|
WasmTranslation* wasmTranslation =
|
||||||
|
debugger ? debugger->wasmTranslation() : nullptr;
|
||||||
for (int i = 0; i < frameCount; i++) {
|
for (int i = 0; i < frameCount; i++) {
|
||||||
v8::Local<v8::StackFrame> stackFrame = stackTrace->GetFrame(i);
|
v8::Local<v8::StackFrame> stackFrame = stackTrace->GetFrame(i);
|
||||||
frames.push_back(toFrame(stackFrame));
|
frames.push_back(toFrame(stackFrame, wasmTranslation, contextGroupId));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +122,8 @@ std::unique_ptr<V8StackTraceImpl> V8StackTraceImpl::create(
|
|||||||
v8::HandleScope scope(isolate);
|
v8::HandleScope scope(isolate);
|
||||||
std::vector<V8StackTraceImpl::Frame> frames;
|
std::vector<V8StackTraceImpl::Frame> frames;
|
||||||
if (!stackTrace.IsEmpty())
|
if (!stackTrace.IsEmpty())
|
||||||
toFramesVector(stackTrace, frames, maxStackSize, isolate);
|
toFramesVector(stackTrace, frames, maxStackSize, isolate, debugger,
|
||||||
|
contextGroupId);
|
||||||
|
|
||||||
int maxAsyncCallChainDepth = 1;
|
int maxAsyncCallChainDepth = 1;
|
||||||
V8StackTraceImpl* asyncCallChain = nullptr;
|
V8StackTraceImpl* asyncCallChain = nullptr;
|
||||||
|
310
src/inspector/wasm-translation.cc
Normal file
310
src/inspector/wasm-translation.cc
Normal file
@ -0,0 +1,310 @@
|
|||||||
|
// Copyright 2016 the V8 project authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#include "src/inspector/wasm-translation.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include "src/debug/debug-interface.h"
|
||||||
|
#include "src/inspector/protocol/Debugger.h"
|
||||||
|
#include "src/inspector/script-breakpoint.h"
|
||||||
|
#include "src/inspector/string-util.h"
|
||||||
|
#include "src/inspector/v8-debugger-agent-impl.h"
|
||||||
|
#include "src/inspector/v8-debugger-script.h"
|
||||||
|
#include "src/inspector/v8-debugger.h"
|
||||||
|
#include "src/inspector/v8-inspector-impl.h"
|
||||||
|
|
||||||
|
using namespace v8_inspector;
|
||||||
|
using namespace v8;
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
int GetScriptId(Isolate *isolate, Local<Object> script_wrapper) {
|
||||||
|
Local<Value> script_id = script_wrapper
|
||||||
|
->Get(isolate->GetCurrentContext(),
|
||||||
|
toV8StringInternalized(isolate, "id"))
|
||||||
|
.ToLocalChecked();
|
||||||
|
DCHECK(script_id->IsInt32());
|
||||||
|
return script_id->Int32Value(isolate->GetCurrentContext()).FromJust();
|
||||||
|
}
|
||||||
|
|
||||||
|
String16 GetScriptName(Isolate *isolate, Local<Object> script_wrapper) {
|
||||||
|
Local<Value> script_name = script_wrapper
|
||||||
|
->Get(isolate->GetCurrentContext(),
|
||||||
|
toV8StringInternalized(isolate, "name"))
|
||||||
|
.ToLocalChecked();
|
||||||
|
DCHECK(script_name->IsString());
|
||||||
|
return toProtocolString(script_name.As<String>());
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class WasmTranslation::TranslatorImpl {
|
||||||
|
public:
|
||||||
|
struct TransLocation {
|
||||||
|
WasmTranslation *translation;
|
||||||
|
String16 script_id;
|
||||||
|
int line;
|
||||||
|
int column;
|
||||||
|
int context_group_id;
|
||||||
|
TransLocation(WasmTranslation *translation, String16 script_id, int line,
|
||||||
|
int column, int context_group_id)
|
||||||
|
: translation(translation),
|
||||||
|
script_id(script_id),
|
||||||
|
line(line),
|
||||||
|
column(column),
|
||||||
|
context_group_id(context_group_id) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void Translate(TransLocation *loc) = 0;
|
||||||
|
virtual void TranslateBack(TransLocation *loc) = 0;
|
||||||
|
|
||||||
|
class RawTranslator;
|
||||||
|
class DisassemblingTranslator;
|
||||||
|
};
|
||||||
|
|
||||||
|
class WasmTranslation::TranslatorImpl::RawTranslator
|
||||||
|
: public WasmTranslation::TranslatorImpl {
|
||||||
|
public:
|
||||||
|
void Translate(TransLocation *loc) {}
|
||||||
|
void TranslateBack(TransLocation *loc) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
class WasmTranslation::TranslatorImpl::DisassemblingTranslator
|
||||||
|
: public WasmTranslation::TranslatorImpl {
|
||||||
|
using OffsetTable = std::vector<std::tuple<uint32_t, int, int>>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
DisassemblingTranslator(Isolate *isolate, Local<Object> script)
|
||||||
|
: script_(isolate, script) {}
|
||||||
|
|
||||||
|
void Translate(TransLocation *loc) {
|
||||||
|
const OffsetTable &offset_table = GetOffsetTable(loc);
|
||||||
|
DCHECK(!offset_table.empty());
|
||||||
|
uint32_t byte_offset = static_cast<uint32_t>(loc->column);
|
||||||
|
|
||||||
|
// Binary search for the given offset.
|
||||||
|
unsigned left = 0; // inclusive
|
||||||
|
unsigned right = static_cast<unsigned>(offset_table.size()); // exclusive
|
||||||
|
while (right - left > 1) {
|
||||||
|
unsigned mid = (left + right) / 2;
|
||||||
|
if (std::get<0>(offset_table[mid]) <= byte_offset) {
|
||||||
|
left = mid;
|
||||||
|
} else {
|
||||||
|
right = mid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
loc->script_id = GetFakeScriptId(loc);
|
||||||
|
if (std::get<0>(offset_table[left]) == byte_offset) {
|
||||||
|
loc->line = std::get<1>(offset_table[left]);
|
||||||
|
loc->column = std::get<2>(offset_table[left]);
|
||||||
|
} else {
|
||||||
|
loc->line = 0;
|
||||||
|
loc->column = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TranslateBack(TransLocation *loc) {
|
||||||
|
int func_index = GetFunctionIndexFromFakeScriptId(loc->script_id);
|
||||||
|
const OffsetTable *reverse_table = GetReverseTable(func_index);
|
||||||
|
if (!reverse_table) return;
|
||||||
|
DCHECK(!reverse_table->empty());
|
||||||
|
|
||||||
|
// Binary search for the given line and column.
|
||||||
|
unsigned left = 0; // inclusive
|
||||||
|
unsigned right = static_cast<unsigned>(reverse_table->size()); // exclusive
|
||||||
|
while (right - left > 1) {
|
||||||
|
unsigned mid = (left + right) / 2;
|
||||||
|
auto &entry = (*reverse_table)[mid];
|
||||||
|
if (std::get<1>(entry) < loc->line ||
|
||||||
|
(std::get<1>(entry) == loc->line &&
|
||||||
|
std::get<2>(entry) <= loc->column)) {
|
||||||
|
left = mid;
|
||||||
|
} else {
|
||||||
|
right = mid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int found_byte_offset = 0;
|
||||||
|
// If we found an exact match, use it. Otherwise check whether the next
|
||||||
|
// bigger entry is still in the same line. Report that one then.
|
||||||
|
if (std::get<1>((*reverse_table)[left]) == loc->line &&
|
||||||
|
std::get<2>((*reverse_table)[left]) == loc->column) {
|
||||||
|
found_byte_offset = std::get<0>((*reverse_table)[left]);
|
||||||
|
} else if (left + 1 < reverse_table->size() &&
|
||||||
|
std::get<1>((*reverse_table)[left + 1]) == loc->line) {
|
||||||
|
found_byte_offset = std::get<0>((*reverse_table)[left + 1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
v8::Isolate *isolate = loc->translation->isolate_;
|
||||||
|
loc->script_id =
|
||||||
|
String16::fromInteger(GetScriptId(isolate, script_.Get(isolate)));
|
||||||
|
loc->line = func_index;
|
||||||
|
loc->column = found_byte_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
String16 GetFakeScriptUrl(const TransLocation *loc) {
|
||||||
|
v8::Isolate *isolate = loc->translation->isolate_;
|
||||||
|
String16 script_name = GetScriptName(isolate, script_.Get(isolate));
|
||||||
|
return String16::concat("wasm://wasm/", script_name, '/', script_name, '-',
|
||||||
|
String16::fromInteger(loc->line));
|
||||||
|
}
|
||||||
|
|
||||||
|
String16 GetFakeScriptId(const TransLocation *loc) {
|
||||||
|
return String16::concat(loc->script_id, '-',
|
||||||
|
String16::fromInteger(loc->line));
|
||||||
|
}
|
||||||
|
|
||||||
|
int GetFunctionIndexFromFakeScriptId(const String16 &fake_script_id) {
|
||||||
|
size_t last_dash_pos = fake_script_id.reverseFind('-');
|
||||||
|
DCHECK_GT(fake_script_id.length(), last_dash_pos);
|
||||||
|
bool ok = true;
|
||||||
|
int func_index = fake_script_id.substring(last_dash_pos + 1).toInteger(&ok);
|
||||||
|
DCHECK(ok);
|
||||||
|
return func_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
const OffsetTable &GetOffsetTable(const TransLocation *loc) {
|
||||||
|
int func_index = loc->line;
|
||||||
|
auto it = offset_tables_.find(func_index);
|
||||||
|
if (it != offset_tables_.end()) return it->second;
|
||||||
|
|
||||||
|
v8::Isolate *isolate = loc->translation->isolate_;
|
||||||
|
std::pair<std::string, OffsetTable> disassembly =
|
||||||
|
DebugInterface::DisassembleWasmFunction(isolate, script_.Get(isolate),
|
||||||
|
func_index);
|
||||||
|
|
||||||
|
it = offset_tables_
|
||||||
|
.insert(std::make_pair(func_index, std::move(disassembly.second)))
|
||||||
|
.first;
|
||||||
|
|
||||||
|
String16 fake_script_id = GetFakeScriptId(loc);
|
||||||
|
String16 fake_script_url = GetFakeScriptUrl(loc);
|
||||||
|
String16 source(disassembly.first.data(), disassembly.first.length());
|
||||||
|
std::unique_ptr<V8DebuggerScript> fake_script(new V8DebuggerScript(
|
||||||
|
fake_script_id, std::move(fake_script_url), source));
|
||||||
|
|
||||||
|
loc->translation->AddFakeScript(std::move(fake_script), this,
|
||||||
|
loc->context_group_id);
|
||||||
|
|
||||||
|
return it->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
const OffsetTable *GetReverseTable(int func_index) {
|
||||||
|
auto it = reverse_tables_.find(func_index);
|
||||||
|
if (it != reverse_tables_.end()) return &it->second;
|
||||||
|
|
||||||
|
// Find offset table, copy and sort it to get reverse table.
|
||||||
|
it = offset_tables_.find(func_index);
|
||||||
|
if (it == offset_tables_.end()) return nullptr;
|
||||||
|
|
||||||
|
OffsetTable reverse_table = it->second;
|
||||||
|
// Order by line, column, then byte offset.
|
||||||
|
auto cmp = [](std::tuple<uint32_t, int, int> el1,
|
||||||
|
std::tuple<uint32_t, int, int> el2) {
|
||||||
|
if (std::get<1>(el1) != std::get<1>(el2))
|
||||||
|
return std::get<1>(el1) < std::get<1>(el2);
|
||||||
|
if (std::get<2>(el1) != std::get<2>(el2))
|
||||||
|
return std::get<2>(el1) < std::get<2>(el2);
|
||||||
|
return std::get<0>(el1) < std::get<0>(el2);
|
||||||
|
};
|
||||||
|
std::sort(reverse_table.begin(), reverse_table.end(), cmp);
|
||||||
|
|
||||||
|
auto inserted = reverse_tables_.insert(
|
||||||
|
std::make_pair(func_index, std::move(reverse_table)));
|
||||||
|
DCHECK(inserted.second);
|
||||||
|
return &inserted.first->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
Global<Object> script_;
|
||||||
|
|
||||||
|
// We assume to only disassemble a subset of the functions, so store them in a
|
||||||
|
// map instead of an array.
|
||||||
|
std::unordered_map<int, const OffsetTable> offset_tables_;
|
||||||
|
std::unordered_map<int, const OffsetTable> reverse_tables_;
|
||||||
|
};
|
||||||
|
|
||||||
|
WasmTranslation::WasmTranslation(v8::Isolate *isolate, V8Debugger *debugger)
|
||||||
|
: isolate_(isolate), debugger_(debugger), mode_(Disassemble) {}
|
||||||
|
|
||||||
|
WasmTranslation::~WasmTranslation() { Clear(); }
|
||||||
|
|
||||||
|
void WasmTranslation::AddScript(Local<Object> script_wrapper) {
|
||||||
|
int script_id = GetScriptId(isolate_, script_wrapper);
|
||||||
|
DCHECK_EQ(0U, wasm_translators_.count(script_id));
|
||||||
|
std::unique_ptr<TranslatorImpl> impl;
|
||||||
|
switch (mode_) {
|
||||||
|
case Raw:
|
||||||
|
impl.reset(new TranslatorImpl::RawTranslator());
|
||||||
|
break;
|
||||||
|
case Disassemble:
|
||||||
|
impl.reset(new TranslatorImpl::DisassemblingTranslator(isolate_,
|
||||||
|
script_wrapper));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
DCHECK(impl);
|
||||||
|
wasm_translators_.insert(std::make_pair(script_id, std::move(impl)));
|
||||||
|
}
|
||||||
|
|
||||||
|
void WasmTranslation::Clear() {
|
||||||
|
wasm_translators_.clear();
|
||||||
|
fake_scripts_.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Translation "forward" (to artificial scripts).
|
||||||
|
bool WasmTranslation::TranslateWasmScriptLocationToProtocolLocation(
|
||||||
|
String16 *script_id, int *line_number, int *column_number,
|
||||||
|
int context_group_id) {
|
||||||
|
DCHECK(script_id && line_number && column_number);
|
||||||
|
bool ok = true;
|
||||||
|
int script_id_int = script_id->toInteger(&ok);
|
||||||
|
if (!ok) return false;
|
||||||
|
|
||||||
|
auto it = wasm_translators_.find(script_id_int);
|
||||||
|
if (it == wasm_translators_.end()) return false;
|
||||||
|
TranslatorImpl *translator = it->second.get();
|
||||||
|
|
||||||
|
TranslatorImpl::TransLocation trans_loc(this, std::move(*script_id),
|
||||||
|
*line_number, *column_number,
|
||||||
|
context_group_id);
|
||||||
|
translator->Translate(&trans_loc);
|
||||||
|
|
||||||
|
*script_id = std::move(trans_loc.script_id);
|
||||||
|
*line_number = trans_loc.line;
|
||||||
|
*column_number = trans_loc.column;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Translation "backward" (from artificial to real scripts).
|
||||||
|
bool WasmTranslation::TranslateProtocolLocationToWasmScriptLocation(
|
||||||
|
String16 *script_id, int *line_number, int *column_number) {
|
||||||
|
auto it = fake_scripts_.find(*script_id);
|
||||||
|
if (it == fake_scripts_.end()) return false;
|
||||||
|
TranslatorImpl *translator = it->second;
|
||||||
|
|
||||||
|
TranslatorImpl::TransLocation trans_loc(this, std::move(*script_id),
|
||||||
|
*line_number, *column_number, -1);
|
||||||
|
translator->TranslateBack(&trans_loc);
|
||||||
|
|
||||||
|
*script_id = std::move(trans_loc.script_id);
|
||||||
|
*line_number = trans_loc.line;
|
||||||
|
*column_number = trans_loc.column;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WasmTranslation::AddFakeScript(
|
||||||
|
std::unique_ptr<V8DebuggerScript> fake_script, TranslatorImpl *translator,
|
||||||
|
int context_group_id) {
|
||||||
|
bool inserted =
|
||||||
|
fake_scripts_.insert(std::make_pair(fake_script->scriptId(), translator))
|
||||||
|
.second;
|
||||||
|
DCHECK(inserted);
|
||||||
|
USE(inserted);
|
||||||
|
V8DebuggerAgentImpl *agent =
|
||||||
|
debugger_->inspector()->enabledDebuggerAgentForGroup(context_group_id);
|
||||||
|
agent->didParseSource(std::move(fake_script), true);
|
||||||
|
}
|
85
src/inspector/wasm-translation.h
Normal file
85
src/inspector/wasm-translation.h
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
// Copyright 2016 the V8 project authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
#ifndef V8_INSPECTOR_WASMTRANSLATION_H_
|
||||||
|
#define V8_INSPECTOR_WASMTRANSLATION_H_
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
|
#include "include/v8.h"
|
||||||
|
#include "src/base/macros.h"
|
||||||
|
#include "src/inspector/string-16.h"
|
||||||
|
|
||||||
|
namespace v8_inspector {
|
||||||
|
|
||||||
|
// Forward declarations.
|
||||||
|
class V8Debugger;
|
||||||
|
class V8DebuggerScript;
|
||||||
|
struct ScriptBreakpoint;
|
||||||
|
namespace protocol {
|
||||||
|
namespace Debugger {
|
||||||
|
class Location;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class WasmTranslation {
|
||||||
|
public:
|
||||||
|
enum Mode { Raw, Disassemble };
|
||||||
|
|
||||||
|
WasmTranslation(v8::Isolate* isolate, V8Debugger* debugger);
|
||||||
|
~WasmTranslation();
|
||||||
|
|
||||||
|
// Set translation mode.
|
||||||
|
void SetMode(Mode mode) { mode_ = mode; }
|
||||||
|
|
||||||
|
// Make a wasm script known to the translation. Only locations referencing a
|
||||||
|
// registered script will be translated by the Translate functions below.
|
||||||
|
void AddScript(v8::Local<v8::Object> script_wrapper);
|
||||||
|
|
||||||
|
// Clear all registered scripts.
|
||||||
|
void Clear();
|
||||||
|
|
||||||
|
// Translate a location as generated by V8 to a location that should be sent
|
||||||
|
// over protocol.
|
||||||
|
// Does nothing for locations referencing a script which was not registered
|
||||||
|
// before via AddScript.
|
||||||
|
// Line and column are 0-based.
|
||||||
|
// The context group id specifies the context of the script.
|
||||||
|
// If the script was registered and the respective wasm function was not seen
|
||||||
|
// before, a new artificial script representing this function will be created
|
||||||
|
// and made public to the frontend.
|
||||||
|
// Returns true if the location was translated, false otherwise.
|
||||||
|
bool TranslateWasmScriptLocationToProtocolLocation(String16* script_id,
|
||||||
|
int* line_number,
|
||||||
|
int* column_number,
|
||||||
|
int context_group_id);
|
||||||
|
|
||||||
|
// Translate back from protocol locations (potentially referencing artificial
|
||||||
|
// scripts for individual wasm functions) to locations that make sense to V8.
|
||||||
|
// Does nothing if the location was not generated by the translate method
|
||||||
|
// above.
|
||||||
|
// Returns true if the location was translated, false otherwise.
|
||||||
|
bool TranslateProtocolLocationToWasmScriptLocation(String16* script_id,
|
||||||
|
int* line_number,
|
||||||
|
int* column_number);
|
||||||
|
|
||||||
|
private:
|
||||||
|
class TranslatorImpl;
|
||||||
|
friend class TranslatorImpl;
|
||||||
|
|
||||||
|
void AddFakeScript(std::unique_ptr<V8DebuggerScript> fake_script,
|
||||||
|
TranslatorImpl* translator, int context_group_id);
|
||||||
|
|
||||||
|
v8::Isolate* isolate_;
|
||||||
|
V8Debugger* debugger_;
|
||||||
|
std::unordered_map<int, std::unique_ptr<TranslatorImpl>> wasm_translators_;
|
||||||
|
std::unordered_map<String16, TranslatorImpl*> fake_scripts_;
|
||||||
|
Mode mode_;
|
||||||
|
|
||||||
|
DISALLOW_COPY_AND_ASSIGN(WasmTranslation);
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace v8_inspector
|
||||||
|
|
||||||
|
#endif // V8_INSPECTOR_WASMTRANSLATION_H_
|
@ -597,6 +597,15 @@ std::pair<int, int> GetFunctionOffsetAndLength(
|
|||||||
static_cast<int>(func.code_end_offset - func.code_start_offset)};
|
static_cast<int>(func.code_end_offset - func.code_start_offset)};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Vector<const uint8_t> GetFunctionBytes(
|
||||||
|
Handle<WasmCompiledModule> compiled_module, int func_index) {
|
||||||
|
int offset, length;
|
||||||
|
std::tie(offset, length) =
|
||||||
|
GetFunctionOffsetAndLength(compiled_module, func_index);
|
||||||
|
return Vector<const uint8_t>(
|
||||||
|
compiled_module->module_bytes()->GetChars() + offset, length);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
const char* wasm::SectionName(WasmSectionCode code) {
|
const char* wasm::SectionName(WasmSectionCode code) {
|
||||||
@ -1882,6 +1891,26 @@ Handle<Script> wasm::GetScript(Handle<JSObject> instance) {
|
|||||||
return compiled_module->script();
|
return compiled_module->script();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::pair<std::string, std::vector<std::tuple<uint32_t, int, int>>>
|
||||||
|
wasm::DisassembleFunction(Handle<WasmCompiledModule> compiled_module,
|
||||||
|
int func_index) {
|
||||||
|
std::ostringstream disassembly_os;
|
||||||
|
std::vector<std::tuple<uint32_t, int, int>> offset_table;
|
||||||
|
|
||||||
|
Vector<const uint8_t> func_bytes =
|
||||||
|
GetFunctionBytes(compiled_module, func_index);
|
||||||
|
DisallowHeapAllocation no_gc;
|
||||||
|
if (func_bytes.is_empty()) return {};
|
||||||
|
|
||||||
|
AccountingAllocator allocator;
|
||||||
|
bool ok = PrintAst(
|
||||||
|
&allocator, FunctionBodyForTesting(func_bytes.start(), func_bytes.end()),
|
||||||
|
disassembly_os, &offset_table);
|
||||||
|
CHECK(ok);
|
||||||
|
|
||||||
|
return {disassembly_os.str(), std::move(offset_table)};
|
||||||
|
}
|
||||||
|
|
||||||
int wasm::GetAsmWasmSourcePosition(Handle<JSObject> instance, int func_index,
|
int wasm::GetAsmWasmSourcePosition(Handle<JSObject> instance, int func_index,
|
||||||
int byte_offset) {
|
int byte_offset) {
|
||||||
return WasmDebugInfo::GetAsmJsSourcePosition(GetDebugInfo(instance),
|
return WasmDebugInfo::GetAsmJsSourcePosition(GetDebugInfo(instance),
|
||||||
|
@ -387,6 +387,13 @@ bool WasmIsAsmJs(Object* instance, Isolate* isolate);
|
|||||||
// it's of type TYPE_WASM.
|
// it's of type TYPE_WASM.
|
||||||
Handle<Script> GetScript(Handle<JSObject> instance);
|
Handle<Script> GetScript(Handle<JSObject> instance);
|
||||||
|
|
||||||
|
// Compute the disassembly of a wasm function.
|
||||||
|
// Returns the disassembly string and a list of <byte_offset, line, column>
|
||||||
|
// entries, mapping wasm byte offsets to line and column in the disassembly.
|
||||||
|
// The list is guaranteed to be ordered by the byte_offset.
|
||||||
|
std::pair<std::string, std::vector<std::tuple<uint32_t, int, int>>>
|
||||||
|
DisassembleFunction(Handle<WasmCompiledModule> compiled_module, int func_index);
|
||||||
|
|
||||||
// Get the asm.js source position for the given byte offset in the given
|
// Get the asm.js source position for the given byte offset in the given
|
||||||
// function.
|
// function.
|
||||||
int GetAsmWasmSourcePosition(Handle<JSObject> instance, int func_index,
|
int GetAsmWasmSourcePosition(Handle<JSObject> instance, int func_index,
|
||||||
|
@ -2,8 +2,8 @@ Running testFunction with generated WASM bytes...
|
|||||||
Paused on 'debugger;'
|
Paused on 'debugger;'
|
||||||
Number of frames: 5
|
Number of frames: 5
|
||||||
- [0] {"functionName":"call_debugger","function_lineNumber":1,"function_columnNumber":24,"lineNumber":2,"columnNumber":4}
|
- [0] {"functionName":"call_debugger","function_lineNumber":1,"function_columnNumber":24,"lineNumber":2,"columnNumber":4}
|
||||||
- [1] {"functionName":"call_func","lineNumber":1,"columnNumber":1}
|
- [1] {"functionName":"call_func","lineNumber":3,"columnNumber":0}
|
||||||
- [2] {"functionName":"main","lineNumber":2,"columnNumber":1}
|
- [2] {"functionName":"main","lineNumber":4,"columnNumber":2}
|
||||||
- [3] {"functionName":"testFunction","function_lineNumber":0,"function_columnNumber":21,"lineNumber":14,"columnNumber":19}
|
- [3] {"functionName":"testFunction","function_lineNumber":0,"function_columnNumber":21,"lineNumber":14,"columnNumber":19}
|
||||||
- [4] {"functionName":"","function_lineNumber":0,"function_columnNumber":0,"lineNumber":0,"columnNumber":0}
|
- [4] {"functionName":"","function_lineNumber":0,"function_columnNumber":0,"lineNumber":0,"columnNumber":0}
|
||||||
Getting v8-generated stack trace...
|
Getting v8-generated stack trace...
|
||||||
@ -12,7 +12,7 @@ Error: this is your stack trace:
|
|||||||
-- skipped --
|
-- skipped --
|
||||||
at call_debugger (<anonymous>:3:5)
|
at call_debugger (<anonymous>:3:5)
|
||||||
at call_func (<WASM>[1]+1)
|
at call_func (<WASM>[1]+1)
|
||||||
at main (<WASM>[2]+1)
|
at main (<WASM>[2]+3)
|
||||||
at testFunction (<anonymous>:15:20)
|
at testFunction (<anonymous>:15:20)
|
||||||
at <anonymous>:1:1
|
at <anonymous>:1:1
|
||||||
Finished!
|
Finished!
|
||||||
|
@ -11,13 +11,15 @@ var builder = new WasmModuleBuilder();
|
|||||||
|
|
||||||
var imported_idx = builder.addImport("func", kSig_v_v);
|
var imported_idx = builder.addImport("func", kSig_v_v);
|
||||||
|
|
||||||
var call_imported_idx = builder.addFunction("call_func", kSig_v_v)
|
var call_imported_idx = builder.addFunction('call_func', kSig_v_v)
|
||||||
.addBody([kExprCallFunction, imported_idx])
|
.addBody([kExprCallFunction, imported_idx])
|
||||||
.index;
|
.index;
|
||||||
|
|
||||||
builder.addFunction("main", kSig_v_v)
|
// Open a block in order to make the positions more interesting...
|
||||||
.addBody([kExprCallFunction, call_imported_idx])
|
builder.addFunction('main', kSig_v_v)
|
||||||
.exportAs("main");
|
.addBody(
|
||||||
|
[kExprBlock, kAstStmt, kExprCallFunction, call_imported_idx, kExprEnd])
|
||||||
|
.exportAs('main');
|
||||||
|
|
||||||
var module_bytes = builder.toArray();
|
var module_bytes = builder.toArray();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user