arkcompiler_toolchain/tooling/dispatcher.cpp

181 lines
5.6 KiB
C++
Raw Normal View History

/*
* 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 "dispatcher.h"
#include "agent/debugger_impl.h"
#include "agent/runtime_impl.h"
#ifdef ECMASCRIPT_SUPPORT_HEAPPROFILER
#include "agent/heapprofiler_impl.h"
#endif
#ifdef ECMASCRIPT_SUPPORT_CPUPROFILER
#include "agent/profiler_impl.h"
#endif
#include "agent/tracing_impl.h"
#include "protocol_channel.h"
namespace panda::ecmascript::tooling {
DispatchRequest::DispatchRequest(const std::string &message)
{
std::unique_ptr<PtJson> json = PtJson::Parse(message);
if (json == nullptr) {
code_ = RequestCode::JSON_PARSE_ERROR;
LOG_DEBUGGER(ERROR) << "json parse error";
return;
}
if (!json->IsObject()) {
code_ = RequestCode::PARAMS_FORMAT_ERROR;
LOG_DEBUGGER(ERROR) << "json parse format error";
json->ReleaseRoot();
return;
}
Result ret;
int32_t callId;
ret = json->GetInt("id", &callId);
if (ret != Result::SUCCESS) {
code_ = RequestCode::PARSE_ID_ERROR;
LOG_DEBUGGER(ERROR) << "parse id error";
return;
}
callId_ = callId;
std::string wholeMethod;
ret = json->GetString("method", &wholeMethod);
if (ret != Result::SUCCESS) {
code_ = RequestCode::PARSE_METHOD_ERROR;
LOG_DEBUGGER(ERROR) << "parse method error";
return;
}
std::string::size_type length = wholeMethod.length();
std::string::size_type indexPoint;
indexPoint = wholeMethod.find_first_of('.', 0);
if (indexPoint == std::string::npos || indexPoint == 0 || indexPoint == length - 1) {
code_ = RequestCode::METHOD_FORMAT_ERROR;
LOG_DEBUGGER(ERROR) << "method format error: " << wholeMethod;
return;
}
domain_ = wholeMethod.substr(0, indexPoint);
method_ = wholeMethod.substr(indexPoint + 1, length);
LOG_DEBUGGER(DEBUG) << "id: " << callId_ << ", domain: " << domain_ << ", method: " << method_;
std::unique_ptr<PtJson> params;
ret = json->GetObject("params", &params);
if (ret == Result::NOT_EXIST) {
return;
}
if (ret == Result::TYPE_ERROR) {
code_ = RequestCode::PARAMS_FORMAT_ERROR;
LOG_DEBUGGER(ERROR) << "params format error";
return;
}
params_ = std::move(params);
}
DispatchRequest::~DispatchRequest()
{
params_->ReleaseRoot();
}
DispatchResponse DispatchResponse::Create(ResponseCode code, const std::string &msg)
{
DispatchResponse response;
response.code_ = code;
response.errorMsg_ = msg;
return response;
}
DispatchResponse DispatchResponse::Create(std::optional<std::string> error)
{
DispatchResponse response;
if (error.has_value()) {
response.code_ = ResponseCode::NOK;
response.errorMsg_ = error.value();
}
return response;
}
DispatchResponse DispatchResponse::Ok()
{
return DispatchResponse();
}
DispatchResponse DispatchResponse::Fail(const std::string &message)
{
DispatchResponse response;
response.code_ = ResponseCode::NOK;
response.errorMsg_ = message;
return response;
}
void DispatcherBase::SendResponse(const DispatchRequest &request, const DispatchResponse &response,
const PtBaseReturns &result)
{
if (channel_ != nullptr) {
channel_->SendResponse(request, response, result);
}
}
Dispatcher::Dispatcher(const EcmaVM *vm, ProtocolChannel *channel)
{
// profiler
#ifdef ECMASCRIPT_SUPPORT_CPUPROFILER
auto profiler = std::make_unique<ProfilerImpl>(vm, channel);
dispatchers_["Profiler"] =
std::make_unique<ProfilerImpl::DispatcherImpl>(channel, std::move(profiler));
#endif
#ifdef ECMASCRIPT_SUPPORT_HEAPPROFILER
auto heapProfiler = std::make_unique<HeapProfilerImpl>(vm, channel);
dispatchers_["HeapProfiler"] =
std::make_unique<HeapProfilerImpl::DispatcherImpl>(channel, std::move(heapProfiler));
#endif
auto tracing = std::make_unique<TracingImpl>(vm, channel);
dispatchers_["Tracing"] =
std::make_unique<TracingImpl::DispatcherImpl>(channel, std::move(tracing));
// debugger
auto runtime = std::make_unique<RuntimeImpl>(vm, channel);
auto debugger = std::make_unique<DebuggerImpl>(vm, channel, runtime.get());
dispatchers_["Runtime"] =
std::make_unique<RuntimeImpl::DispatcherImpl>(channel, std::move(runtime));
dispatchers_["Debugger"] =
std::make_unique<DebuggerImpl::DispatcherImpl>(channel, std::move(debugger));
}
void Dispatcher::Dispatch(const DispatchRequest &request)
{
if (!request.IsValid()) {
LOG_DEBUGGER(ERROR) << "Unknown request";
return;
}
const std::string &domain = request.GetDomain();
auto dispatcher = dispatchers_.find(domain);
if (dispatcher != dispatchers_.end()) {
dispatcher->second->Dispatch(request);
} else {
if (domain == "Test") {
if (request.GetMethod() == "fail") {
LOG_DEBUGGER(FATAL) << "Test fail";
UNREACHABLE();
}
LOG_DEBUGGER(INFO) << "Test success";
} else {
LOG_DEBUGGER(ERROR) << "unknown domain: " << domain;
}
}
}
} // namespace panda::ecmascript::tooling