mirror of
https://gitee.com/openharmony/arkcompiler_toolchain
synced 2024-11-23 15:40:03 +00:00
39c6c158f5
Add threadState switch for debugger Issue:https://gitee.com/openharmony/arkcompiler_ets_runtime/issues/IA699D?from=project-issue Signed-off-by: lukai <lukai25@huawei.com> Change-Id: I14188917a1de97c248b6ba19d5395aa1ae1d1f5a
136 lines
4.1 KiB
C++
136 lines
4.1 KiB
C++
/*
|
|
* Copyright (c) 2021 Huawei Device Co., Ltd.
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "protocol_handler.h"
|
|
|
|
#include "agent/debugger_impl.h"
|
|
|
|
namespace panda::ecmascript::tooling {
|
|
void ProtocolHandler::WaitForDebugger()
|
|
{
|
|
waitingForDebugger_ = true;
|
|
ProcessCommand();
|
|
}
|
|
|
|
void ProtocolHandler::RunIfWaitingForDebugger()
|
|
{
|
|
waitingForDebugger_ = false;
|
|
}
|
|
|
|
void ProtocolHandler::DispatchCommand(std::string &&msg)
|
|
{
|
|
LOG_DEBUGGER(DEBUG) << "ProtocolHandler::DispatchCommand: " << msg;
|
|
std::unique_lock<std::mutex> queueLock(requestLock_);
|
|
requestQueue_.push(std::move(msg));
|
|
requestQueueCond_.notify_one();
|
|
}
|
|
|
|
// called after DispatchCommand
|
|
int32_t ProtocolHandler::GetDispatchStatus()
|
|
{
|
|
if (isDispatchingMessage_ || waitingForDebugger_) {
|
|
return DispatchStatus::DISPATCHING;
|
|
}
|
|
std::unique_lock<std::mutex> queueLock(requestLock_);
|
|
if (requestQueue_.empty()) {
|
|
return DispatchStatus::DISPATCHED;
|
|
}
|
|
return DispatchStatus::UNKNOWN;
|
|
}
|
|
|
|
void ProtocolHandler::ProcessCommand()
|
|
{
|
|
std::queue<std::string> dispatchingQueue;
|
|
do {
|
|
DebuggerApi::DebuggerNativeScope nativeScope(vm_);
|
|
{
|
|
std::unique_lock<std::mutex> queueLock(requestLock_);
|
|
if (requestQueue_.empty()) {
|
|
if (!waitingForDebugger_) {
|
|
return;
|
|
}
|
|
requestQueueCond_.wait(queueLock);
|
|
}
|
|
requestQueue_.swap(dispatchingQueue);
|
|
}
|
|
|
|
isDispatchingMessage_ = true;
|
|
{
|
|
DebuggerApi::DebuggerManagedScope managedScope(vm_);
|
|
while (!dispatchingQueue.empty()) {
|
|
std::string msg = std::move(dispatchingQueue.front());
|
|
dispatchingQueue.pop();
|
|
|
|
[[maybe_unused]] LocalScope scope(vm_);
|
|
auto exception = DebuggerApi::GetAndClearException(vm_);
|
|
dispatcher_.Dispatch(DispatchRequest(msg));
|
|
DebuggerApi::SetException(vm_, exception);
|
|
}
|
|
}
|
|
isDispatchingMessage_ = false;
|
|
} while (true);
|
|
}
|
|
|
|
void ProtocolHandler::SendResponse(const DispatchRequest &request, const DispatchResponse &response,
|
|
const PtBaseReturns &result)
|
|
{
|
|
LOG_DEBUGGER(INFO) << "ProtocolHandler::SendResponse: "
|
|
<< (response.IsOk() ? "success" : "failed: " + response.GetMessage());
|
|
|
|
std::unique_ptr<PtJson> reply = PtJson::CreateObject();
|
|
reply->Add("id", request.GetCallId());
|
|
std::unique_ptr<PtJson> resultObj;
|
|
if (response.IsOk()) {
|
|
resultObj = result.ToJson();
|
|
} else {
|
|
resultObj = CreateErrorReply(response);
|
|
}
|
|
reply->Add("result", resultObj);
|
|
SendReply(*reply);
|
|
reply->ReleaseRoot();
|
|
}
|
|
|
|
void ProtocolHandler::SendNotification(const PtBaseEvents &events)
|
|
{
|
|
LOG_DEBUGGER(DEBUG) << "ProtocolHandler::SendNotification: " << events.GetName();
|
|
std::unique_ptr<PtJson> reply = events.ToJson();
|
|
SendReply(*reply);
|
|
reply->ReleaseRoot();
|
|
}
|
|
|
|
void ProtocolHandler::SendReply(const PtJson &reply)
|
|
{
|
|
std::string str = reply.Stringify();
|
|
if (str.empty()) {
|
|
LOG_DEBUGGER(ERROR) << "ProtocolHandler::SendReply: json stringify error";
|
|
return;
|
|
}
|
|
|
|
callback_(reinterpret_cast<const void *>(vm_), str);
|
|
}
|
|
|
|
std::unique_ptr<PtJson> ProtocolHandler::CreateErrorReply(const DispatchResponse &response)
|
|
{
|
|
std::unique_ptr<PtJson> result = PtJson::CreateObject();
|
|
|
|
if (!response.IsOk()) {
|
|
result->Add("code", static_cast<int32_t>(response.GetError()));
|
|
result->Add("message", response.GetMessage().c_str());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
} // namespace panda::ecmascript::tooling
|