|  | // Copyright 2014 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/v8-debugger-script.h" | 
|  |  | 
|  | #include "src/base/memory.h" | 
|  | #include "src/inspector/inspected-context.h" | 
|  | #include "src/inspector/protocol/Debugger.h" | 
|  | #include "src/inspector/string-util.h" | 
|  | #include "src/inspector/v8-debugger-agent-impl.h" | 
|  | #include "src/inspector/v8-inspector-impl.h" | 
|  |  | 
|  | namespace v8_inspector { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char kGlobalDebuggerScriptHandleLabel[] = "DevTools debugger"; | 
|  |  | 
|  | // Hash algorithm for substrings is described in "Über die Komplexität der | 
|  | // Multiplikation in | 
|  | // eingeschränkten Branchingprogrammmodellen" by Woelfe. | 
|  | // http://opendatastructures.org/versions/edition-0.1d/ods-java/node33.html#SECTION00832000000000000000 | 
|  | String16 calculateHash(v8::Isolate* isolate, v8::Local<v8::String> source) { | 
|  | static uint64_t prime[] = {0x3FB75161, 0xAB1F4E4F, 0x82675BC5, 0xCD924D35, | 
|  | 0x81ABE279}; | 
|  | static uint64_t random[] = {0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, | 
|  | 0xC3D2E1F0}; | 
|  | static uint32_t randomOdd[] = {0xB4663807, 0xCC322BF5, 0xD4F91BBD, 0xA7BEA11D, | 
|  | 0x8F462907}; | 
|  |  | 
|  | uint64_t hashes[] = {0, 0, 0, 0, 0}; | 
|  | uint64_t zi[] = {1, 1, 1, 1, 1}; | 
|  |  | 
|  | const size_t hashesSize = arraysize(hashes); | 
|  |  | 
|  | size_t current = 0; | 
|  |  | 
|  | std::unique_ptr<UChar[]> buffer(new UChar[source->Length()]); | 
|  | int written = source->Write( | 
|  | isolate, reinterpret_cast<uint16_t*>(buffer.get()), 0, source->Length()); | 
|  |  | 
|  | const uint32_t* data = nullptr; | 
|  | size_t sizeInBytes = sizeof(UChar) * written; | 
|  | data = reinterpret_cast<const uint32_t*>(buffer.get()); | 
|  | for (size_t i = 0; i < sizeInBytes / 4; ++i) { | 
|  | uint32_t d = v8::base::ReadUnalignedValue<uint32_t>( | 
|  | reinterpret_cast<v8::internal::Address>(data + i)); | 
|  | #if V8_TARGET_LITTLE_ENDIAN | 
|  | uint32_t v = d; | 
|  | #else | 
|  | uint32_t v = (d << 16) | (d >> 16); | 
|  | #endif | 
|  | uint64_t xi = v * randomOdd[current] & 0x7FFFFFFF; | 
|  | hashes[current] = (hashes[current] + zi[current] * xi) % prime[current]; | 
|  | zi[current] = (zi[current] * random[current]) % prime[current]; | 
|  | current = current == hashesSize - 1 ? 0 : current + 1; | 
|  | } | 
|  | if (sizeInBytes % 4) { | 
|  | uint32_t v = 0; | 
|  | const uint8_t* data_8b = reinterpret_cast<const uint8_t*>(data); | 
|  | for (size_t i = sizeInBytes - sizeInBytes % 4; i < sizeInBytes; ++i) { | 
|  | v <<= 8; | 
|  | #if V8_TARGET_LITTLE_ENDIAN | 
|  | v |= data_8b[i]; | 
|  | #else | 
|  | if (i % 2) { | 
|  | v |= data_8b[i - 1]; | 
|  | } else { | 
|  | v |= data_8b[i + 1]; | 
|  | } | 
|  | #endif | 
|  | } | 
|  | uint64_t xi = v * randomOdd[current] & 0x7FFFFFFF; | 
|  | hashes[current] = (hashes[current] + zi[current] * xi) % prime[current]; | 
|  | zi[current] = (zi[current] * random[current]) % prime[current]; | 
|  | current = current == hashesSize - 1 ? 0 : current + 1; | 
|  | } | 
|  |  | 
|  | for (size_t i = 0; i < hashesSize; ++i) | 
|  | hashes[i] = (hashes[i] + zi[i] * (prime[i] - 1)) % prime[i]; | 
|  |  | 
|  | String16Builder hash; | 
|  | for (size_t i = 0; i < hashesSize; ++i) | 
|  | hash.appendUnsignedAsHex(static_cast<uint32_t>(hashes[i])); | 
|  | return hash.toString(); | 
|  | } | 
|  |  | 
|  | class ActualScript : public V8DebuggerScript { | 
|  | friend class V8DebuggerScript; | 
|  |  | 
|  | public: | 
|  | ActualScript(v8::Isolate* isolate, v8::Local<v8::debug::Script> script, | 
|  | bool isLiveEdit, V8DebuggerAgentImpl* agent, | 
|  | V8InspectorClient* client) | 
|  | : V8DebuggerScript(isolate, String16::fromInteger(script->Id()), | 
|  | GetScriptURL(isolate, script, client), | 
|  | GetScriptName(isolate, script, client)), | 
|  | m_agent(agent), | 
|  | m_isLiveEdit(isLiveEdit) { | 
|  | Initialize(script); | 
|  | } | 
|  |  | 
|  | bool isLiveEdit() const override { return m_isLiveEdit; } | 
|  | bool isModule() const override { return m_isModule; } | 
|  |  | 
|  | String16 source(size_t pos, size_t len) const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | v8::Local<v8::String> v8Source; | 
|  | if (!script()->Source().ToLocal(&v8Source)) return String16(); | 
|  | if (pos >= static_cast<size_t>(v8Source->Length())) return String16(); | 
|  | size_t substringLength = | 
|  | std::min(len, static_cast<size_t>(v8Source->Length()) - pos); | 
|  | std::unique_ptr<UChar[]> buffer(new UChar[substringLength]); | 
|  | v8Source->Write(m_isolate, reinterpret_cast<uint16_t*>(buffer.get()), | 
|  | static_cast<int>(pos), static_cast<int>(substringLength)); | 
|  | return String16(buffer.get(), substringLength); | 
|  | } | 
|  | v8::Maybe<v8::MemorySpan<const uint8_t>> wasmBytecode() const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | auto script = this->script(); | 
|  | if (!script->IsWasm()) return v8::Nothing<v8::MemorySpan<const uint8_t>>(); | 
|  | return v8::Just(v8::debug::WasmScript::Cast(*script)->Bytecode()); | 
|  | } | 
|  | Language getLanguage() const override { return m_language; } | 
|  | v8::Maybe<v8::debug::WasmScript::DebugSymbolsType> getDebugSymbolsType() | 
|  | const override { | 
|  | auto script = this->script(); | 
|  | if (!script->IsWasm()) | 
|  | return v8::Nothing<v8::debug::WasmScript::DebugSymbolsType>(); | 
|  | return v8::Just(v8::debug::WasmScript::Cast(*script)->GetDebugSymbolType()); | 
|  | } | 
|  | v8::Maybe<String16> getExternalDebugSymbolsURL() const override { | 
|  | auto script = this->script(); | 
|  | if (!script->IsWasm()) return v8::Nothing<String16>(); | 
|  | v8::MemorySpan<const char> external_url = | 
|  | v8::debug::WasmScript::Cast(*script)->ExternalSymbolsURL(); | 
|  | if (external_url.size() == 0) return v8::Nothing<String16>(); | 
|  | return v8::Just(String16(external_url.data(), external_url.size())); | 
|  | } | 
|  | int startLine() const override { return m_startLine; } | 
|  | int startColumn() const override { return m_startColumn; } | 
|  | int endLine() const override { return m_endLine; } | 
|  | int endColumn() const override { return m_endColumn; } | 
|  | int codeOffset() const override { | 
|  | auto script = this->script(); | 
|  | if (!script->IsWasm()) return 0; | 
|  | return v8::debug::WasmScript::Cast(*script)->CodeOffset(); | 
|  | } | 
|  | bool isSourceLoadedLazily() const override { return false; } | 
|  | int length() const override { | 
|  | auto script = this->script(); | 
|  | if (script->IsWasm()) { | 
|  | return static_cast<int>( | 
|  | v8::debug::WasmScript::Cast(*script)->Bytecode().size()); | 
|  | } | 
|  | v8::HandleScope scope(m_isolate); | 
|  | v8::Local<v8::String> v8Source; | 
|  | return script->Source().ToLocal(&v8Source) ? v8Source->Length() : 0; | 
|  | } | 
|  |  | 
|  | const String16& sourceMappingURL() const override { | 
|  | return m_sourceMappingURL; | 
|  | } | 
|  |  | 
|  | void setSourceMappingURL(const String16& sourceMappingURL) override { | 
|  | m_sourceMappingURL = sourceMappingURL; | 
|  | } | 
|  |  | 
|  | void setSource(const String16& newSource, bool preview, | 
|  | v8::debug::LiveEditResult* result) override { | 
|  | v8::EscapableHandleScope scope(m_isolate); | 
|  | v8::Local<v8::String> v8Source = toV8String(m_isolate, newSource); | 
|  | if (!m_script.Get(m_isolate)->SetScriptSource(v8Source, preview, result)) { | 
|  | result->message = scope.Escape(result->message); | 
|  | return; | 
|  | } | 
|  | // NOP if preview or unchanged source (diffs.empty() in PatchScript) | 
|  | if (preview || result->script.IsEmpty()) return; | 
|  |  | 
|  | m_hash = String16(); | 
|  | Initialize(scope.Escape(result->script)); | 
|  | } | 
|  |  | 
|  | bool getPossibleBreakpoints( | 
|  | const v8::debug::Location& start, const v8::debug::Location& end, | 
|  | bool restrictToFunction, | 
|  | std::vector<v8::debug::BreakLocation>* locations) override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | v8::Local<v8::debug::Script> script = m_script.Get(m_isolate); | 
|  | std::vector<v8::debug::BreakLocation> allLocations; | 
|  | if (!script->GetPossibleBreakpoints(start, end, restrictToFunction, | 
|  | &allLocations)) { | 
|  | return false; | 
|  | } | 
|  | if (!allLocations.size()) return true; | 
|  | v8::debug::BreakLocation current = allLocations[0]; | 
|  | for (size_t i = 1; i < allLocations.size(); ++i) { | 
|  | if (allLocations[i].GetLineNumber() == current.GetLineNumber() && | 
|  | allLocations[i].GetColumnNumber() == current.GetColumnNumber()) { | 
|  | if (allLocations[i].type() != v8::debug::kCommonBreakLocation) { | 
|  | DCHECK(allLocations[i].type() == v8::debug::kCallBreakLocation || | 
|  | allLocations[i].type() == v8::debug::kReturnBreakLocation); | 
|  | // debugger can returns more then one break location at the same | 
|  | // source location, e.g. foo() - in this case there are two break | 
|  | // locations before foo: for statement and for function call, we can | 
|  | // merge them for inspector and report only one with call type. | 
|  | current = allLocations[i]; | 
|  | } | 
|  | } else { | 
|  | // we assume that returned break locations are sorted. | 
|  | DCHECK( | 
|  | allLocations[i].GetLineNumber() > current.GetLineNumber() || | 
|  | (allLocations[i].GetColumnNumber() >= current.GetColumnNumber() && | 
|  | allLocations[i].GetLineNumber() == current.GetLineNumber())); | 
|  | locations->push_back(current); | 
|  | current = allLocations[i]; | 
|  | } | 
|  | } | 
|  | locations->push_back(current); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void resetBlackboxedStateCache() override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | v8::debug::ResetBlackboxedStateCache(m_isolate, m_script.Get(m_isolate)); | 
|  | } | 
|  |  | 
|  | int offset(int lineNumber, int columnNumber) const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | return m_script.Get(m_isolate)->GetSourceOffset( | 
|  | v8::debug::Location(lineNumber, columnNumber)); | 
|  | } | 
|  |  | 
|  | v8::debug::Location location(int offset) const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | return m_script.Get(m_isolate)->GetSourceLocation(offset); | 
|  | } | 
|  |  | 
|  | bool setBreakpoint(const String16& condition, v8::debug::Location* location, | 
|  | int* id) const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | return script()->SetBreakpoint(toV8String(m_isolate, condition), location, | 
|  | id); | 
|  | } | 
|  |  | 
|  | bool setBreakpointOnRun(int* id) const override { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | return script()->SetBreakpointOnScriptEntry(id); | 
|  | } | 
|  |  | 
|  | const String16& hash() const override { | 
|  | if (!m_hash.isEmpty()) return m_hash; | 
|  | v8::HandleScope scope(m_isolate); | 
|  | v8::Local<v8::String> v8Source; | 
|  | if (script()->Source().ToLocal(&v8Source)) { | 
|  | m_hash = calculateHash(m_isolate, v8Source); | 
|  | } | 
|  | DCHECK(!m_hash.isEmpty()); | 
|  | return m_hash; | 
|  | } | 
|  |  | 
|  | private: | 
|  | static String16 GetScriptURL(v8::Isolate* isolate, | 
|  | v8::Local<v8::debug::Script> script, | 
|  | V8InspectorClient* client) { | 
|  | v8::Local<v8::String> sourceURL; | 
|  | if (script->SourceURL().ToLocal(&sourceURL) && sourceURL->Length() > 0) | 
|  | return toProtocolString(isolate, sourceURL); | 
|  | return GetScriptName(isolate, script, client); | 
|  | } | 
|  |  | 
|  | static String16 GetScriptName(v8::Isolate* isolate, | 
|  | v8::Local<v8::debug::Script> script, | 
|  | V8InspectorClient* client) { | 
|  | v8::Local<v8::String> v8Name; | 
|  | if (script->Name().ToLocal(&v8Name) && v8Name->Length() > 0) { | 
|  | String16 name = toProtocolString(isolate, v8Name); | 
|  | std::unique_ptr<StringBuffer> url = | 
|  | client->resourceNameToUrl(toStringView(name)); | 
|  | return url ? toString16(url->string()) : name; | 
|  | } | 
|  | return String16(); | 
|  | } | 
|  |  | 
|  | v8::Local<v8::debug::Script> script() const override { | 
|  | return m_script.Get(m_isolate); | 
|  | } | 
|  |  | 
|  | void Initialize(v8::Local<v8::debug::Script> script) { | 
|  | v8::Local<v8::String> tmp; | 
|  | m_hasSourceURLComment = | 
|  | script->SourceURL().ToLocal(&tmp) && tmp->Length() > 0; | 
|  | if (script->SourceMappingURL().ToLocal(&tmp)) | 
|  | m_sourceMappingURL = toProtocolString(m_isolate, tmp); | 
|  | m_startLine = script->LineOffset(); | 
|  | m_startColumn = script->ColumnOffset(); | 
|  | std::vector<int> lineEnds = script->LineEnds(); | 
|  | if (lineEnds.size()) { | 
|  | int source_length = lineEnds[lineEnds.size() - 1]; | 
|  | m_endLine = static_cast<int>(lineEnds.size()) + m_startLine - 1; | 
|  | if (lineEnds.size() > 1) { | 
|  | m_endColumn = source_length - lineEnds[lineEnds.size() - 2] - 1; | 
|  | } else { | 
|  | m_endColumn = source_length + m_startColumn; | 
|  | } | 
|  | } else if (script->IsWasm()) { | 
|  | DCHECK_EQ(0, m_startLine); | 
|  | DCHECK_EQ(0, m_startColumn); | 
|  | m_endLine = 0; | 
|  | m_endColumn = static_cast<int>( | 
|  | v8::debug::WasmScript::Cast(*script)->Bytecode().size()); | 
|  | } else { | 
|  | m_endLine = m_startLine; | 
|  | m_endColumn = m_startColumn; | 
|  | } | 
|  |  | 
|  | USE(script->ContextId().To(&m_executionContextId)); | 
|  | if (script->IsWasm()) { | 
|  | m_language = V8DebuggerScript::Language::WebAssembly; | 
|  | } else { | 
|  | m_language = V8DebuggerScript::Language::JavaScript; | 
|  | } | 
|  |  | 
|  | m_isModule = script->IsModule(); | 
|  |  | 
|  | m_script.Reset(m_isolate, script); | 
|  | m_script.AnnotateStrongRetainer(kGlobalDebuggerScriptHandleLabel); | 
|  | } | 
|  |  | 
|  | void MakeWeak() override { | 
|  | m_script.SetWeak( | 
|  | this, | 
|  | [](const v8::WeakCallbackInfo<ActualScript>& data) { | 
|  | data.GetParameter()->WeakCallback(); | 
|  | }, | 
|  | v8::WeakCallbackType::kFinalizer); | 
|  | } | 
|  |  | 
|  | void WeakCallback() { | 
|  | m_script.ClearWeak(); | 
|  | m_agent->ScriptCollected(this); | 
|  | } | 
|  |  | 
|  | V8DebuggerAgentImpl* m_agent; | 
|  | String16 m_sourceMappingURL; | 
|  | Language m_language; | 
|  | bool m_isLiveEdit = false; | 
|  | bool m_isModule = false; | 
|  | mutable String16 m_hash; | 
|  | int m_startLine = 0; | 
|  | int m_startColumn = 0; | 
|  | int m_endLine = 0; | 
|  | int m_endColumn = 0; | 
|  | v8::Global<v8::debug::Script> m_script; | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | std::unique_ptr<V8DebuggerScript> V8DebuggerScript::Create( | 
|  | v8::Isolate* isolate, v8::Local<v8::debug::Script> scriptObj, | 
|  | bool isLiveEdit, V8DebuggerAgentImpl* agent, V8InspectorClient* client) { | 
|  | return std::make_unique<ActualScript>(isolate, scriptObj, isLiveEdit, agent, | 
|  | client); | 
|  | } | 
|  |  | 
|  | V8DebuggerScript::V8DebuggerScript(v8::Isolate* isolate, String16 id, | 
|  | String16 url, String16 embedderName) | 
|  | : m_id(std::move(id)), | 
|  | m_url(std::move(url)), | 
|  | m_isolate(isolate), | 
|  | m_embedderName(embedderName) {} | 
|  |  | 
|  | V8DebuggerScript::~V8DebuggerScript() = default; | 
|  |  | 
|  | void V8DebuggerScript::setSourceURL(const String16& sourceURL) { | 
|  | if (sourceURL.length() > 0) { | 
|  | m_hasSourceURLComment = true; | 
|  | m_url = sourceURL; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool V8DebuggerScript::setBreakpoint(const String16& condition, | 
|  | v8::debug::Location* loc, int* id) const { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | return script()->SetBreakpoint(toV8String(m_isolate, condition), loc, id); | 
|  | } | 
|  |  | 
|  | void V8DebuggerScript::removeWasmBreakpoint(int id) { | 
|  | v8::HandleScope scope(m_isolate); | 
|  | script()->RemoveWasmBreakpoint(id); | 
|  | } | 
|  |  | 
|  | }  // namespace v8_inspector |