1 Star 0 Fork 177

jinjiawei/arkcompiler_toolchain

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
inspector.cpp 13.57 KB
一键复制 编辑 原始数据 按行查看 历史
石闽南 提交于 2024-07-30 09:19 . Fix alarm
/*
* Copyright (c) 2021-2022 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 "inspector.h"
#include <chrono>
#include <shared_mutex>
#if defined(OHOS_PLATFORM)
#include <syscall.h>
#endif
#include <thread>
#if defined(OHOS_PLATFORM)
#include <unistd.h>
#endif
#include <unordered_map>
#include "common/log_wrapper.h"
#include "library_loader.h"
#if defined(IOS_PLATFORM)
#include "tooling/debugger_service.h"
#endif
#if defined(ENABLE_FFRT_INTERFACES)
#include "ffrt.h"
#endif
namespace OHOS::ArkCompiler::Toolchain {
namespace {
enum DispatchStatus : int32_t {
UNKNOWN = 0,
DISPATCHING,
DISPATCHED
};
using InitializeDebugger = void(*)(void*, const std::function<void(const void*, const std::string&)>&);
using UninitializeDebugger = void(*)(void*);
using WaitForDebugger = void(*)(void*);
using OnMessage = void(*)(void*, std::string&&);
using ProcessMessage = void(*)(void*);
using GetDispatchStatus = int32_t(*)(void*);
OnMessage g_onMessage = nullptr;
InitializeDebugger g_initializeDebugger = nullptr;
UninitializeDebugger g_uninitializeDebugger = nullptr;
WaitForDebugger g_waitForDebugger = nullptr;
ProcessMessage g_processMessage = nullptr;
GetDispatchStatus g_getDispatchStatus = nullptr;
std::atomic<bool> g_hasArkFuncsInited = false;
std::unordered_map<const void*, Inspector*> g_inspectors;
std::unordered_map<int, std::pair<void*, const DebuggerPostTask>> g_debuggerInfo;
std::shared_mutex g_mutex;
#if !defined(IOS_PLATFORM)
thread_local void* g_handle = nullptr;
#endif
thread_local void* g_vm = nullptr;
#if !defined(IOS_PLATFORM)
#if defined(WINDOWS_PLATFORM)
constexpr char ARK_DEBUGGER_SHARED_LIB[] = "libark_ecma_debugger.dll";
#elif defined(MAC_PLATFORM)
constexpr char ARK_DEBUGGER_SHARED_LIB[] = "libark_ecma_debugger.dylib";
#else
constexpr char ARK_DEBUGGER_SHARED_LIB[] = "libark_ecma_debugger.so";
#endif
#endif
void* HandleClient(void* const server)
{
LOGI("HandleClient");
if (server == nullptr) {
LOGE("HandleClient server nullptr");
return nullptr;
}
#if defined(IOS_PLATFORM) || defined(MAC_PLATFORM)
pthread_setname_np("OS_DebugThread");
#else
pthread_setname_np(pthread_self(), "OS_DebugThread");
#endif
static_cast<WsServer*>(server)->RunServer();
return nullptr;
}
#if !defined(IOS_PLATFORM)
bool LoadArkDebuggerLibrary()
{
if (g_handle != nullptr) {
LOGI("LoadArkDebuggerLibrary, handle has already loaded");
return true;
}
g_handle = Load(ARK_DEBUGGER_SHARED_LIB);
if (g_handle == nullptr) {
LOGE("LoadArkDebuggerLibrary, handle load failed");
return false;
}
return true;
}
void* GetArkDynFunction(const char* symbol)
{
return ResolveSymbol(g_handle, symbol);
}
#endif
void SendReply(const void* vm, const std::string& message)
{
std::shared_lock<std::shared_mutex> lock(g_mutex);
auto iter = g_inspectors.find(vm);
if (iter != g_inspectors.end() && iter->second != nullptr &&
iter->second->websocketServer_ != nullptr) {
iter->second->websocketServer_->SendReply(message);
}
}
void ResetServiceLocked(void *vm, bool isCloseHandle)
{
auto iter = g_inspectors.find(vm);
if (iter != g_inspectors.end() && iter->second != nullptr &&
iter->second->websocketServer_ != nullptr) {
iter->second->websocketServer_->StopServer();
delete iter->second;
iter->second = nullptr;
g_inspectors.erase(iter);
}
#if !defined(IOS_PLATFORM)
if (g_handle != nullptr && isCloseHandle) {
CloseHandle(g_handle);
g_handle = nullptr;
}
#endif
}
bool InitializeInspector(
void* vm, const DebuggerPostTask& debuggerPostTask, const DebugInfo& debugInfo, int tidForSocketPair = 0)
{
std::unique_lock<std::shared_mutex> lock(g_mutex);
Inspector *newInspector = nullptr;
auto iter = g_inspectors.find(vm);
if (iter != g_inspectors.end()) {
newInspector = g_inspectors[vm];
} else {
newInspector = new Inspector();
if (!g_inspectors.emplace(vm, newInspector).second) {
delete newInspector;
return false;
}
}
newInspector->tidForSocketPair_ = tidForSocketPair;
newInspector->tid_ = pthread_self();
newInspector->vm_ = vm;
newInspector->debuggerPostTask_ = debuggerPostTask;
newInspector->websocketServer_ = std::make_unique<WsServer>(debugInfo,
std::bind(&Inspector::OnMessage, newInspector, std::placeholders::_1));
pthread_t tid;
if (pthread_create(&tid, nullptr, &HandleClient, static_cast<void *>(
newInspector->websocketServer_.get())) != 0) {
LOGE("Create inspector thread failed");
return false;
}
newInspector->websocketServer_->tid_ = tid;
return true;
}
#if !defined(IOS_PLATFORM)
bool InitializeArkFunctionsOthers()
{
g_initializeDebugger = reinterpret_cast<InitializeDebugger>(
GetArkDynFunction("InitializeDebugger"));
if (g_initializeDebugger == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
g_uninitializeDebugger = reinterpret_cast<UninitializeDebugger>(
GetArkDynFunction("UninitializeDebugger"));
if (g_uninitializeDebugger == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
g_waitForDebugger = reinterpret_cast<WaitForDebugger>(
GetArkDynFunction("WaitForDebugger"));
if (g_waitForDebugger == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
g_onMessage = reinterpret_cast<OnMessage>(
GetArkDynFunction("OnMessage"));
if (g_onMessage == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
g_getDispatchStatus = reinterpret_cast<GetDispatchStatus>(
GetArkDynFunction("GetDispatchStatus"));
if (g_getDispatchStatus == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
g_processMessage = reinterpret_cast<ProcessMessage>(
GetArkDynFunction("ProcessMessage"));
if (g_processMessage == nullptr) {
ResetServiceLocked(g_vm, true);
return false;
}
return true;
}
#else
bool InitializeArkFunctionsIOS()
{
using namespace panda::ecmascript;
g_initializeDebugger = reinterpret_cast<InitializeDebugger>(&tooling::InitializeDebugger);
g_uninitializeDebugger = reinterpret_cast<UninitializeDebugger>(&tooling::UninitializeDebugger);
g_waitForDebugger = reinterpret_cast<WaitForDebugger>(&tooling::WaitForDebugger);
g_onMessage = reinterpret_cast<OnMessage>(&tooling::OnMessage);
g_getDispatchStatus = reinterpret_cast<GetDispatchStatus>(&tooling::GetDispatchStatus);
g_processMessage = reinterpret_cast<ProcessMessage>(&tooling::ProcessMessage);
return true;
}
#endif
bool InitializeArkFunctions()
{
// no need to initialize again in case of multi-instance
if (g_hasArkFuncsInited) {
return true;
}
std::unique_lock<std::shared_mutex> lock(g_mutex);
if (g_hasArkFuncsInited) {
return true;
}
#if !defined(IOS_PLATFORM)
if (!InitializeArkFunctionsOthers()) {
return false;
}
#else
InitializeArkFunctionsIOS()
#endif
g_hasArkFuncsInited = true;
return true;
}
} // namespace
void Inspector::OnMessage(std::string&& msg)
{
g_onMessage(vm_, std::move(msg));
// message will be processed soon if the debugger thread is in running or waiting status
if (g_getDispatchStatus(vm_) != DispatchStatus::UNKNOWN) {
return;
}
std::this_thread::sleep_for(std::chrono::microseconds(DELAY_CHECK_DISPATCH_STATUS));
if (g_getDispatchStatus(vm_) != DispatchStatus::UNKNOWN) {
return;
}
// the debugger thread maybe in idle status, so try to post a task to wake it up
if (debuggerPostTask_ != nullptr) {
if (tidForSocketPair_ == 0) {
debuggerPostTask_([tid = tid_, vm = vm_] {
if (tid != pthread_self()) {
LOGE("Task not in debugger thread");
return;
}
g_processMessage(vm);
});
} else {
#if defined(OHOS_PLATFORM)
debuggerPostTask_([tid = tidForSocketPair_, vm = vm_, this] {
uint64_t threadOrTaskId = GetThreadOrTaskId();
if (tid != static_cast<pid_t>(threadOrTaskId)) {
LOGE("Task not in debugger thread for socketpair");
return;
}
g_processMessage(vm);
});
#endif // defined(OHOS_PLATFORM)
}
} else {
LOGW("No debuggerPostTask provided");
}
}
#if defined(OHOS_PLATFORM)
uint64_t Inspector::GetThreadOrTaskId()
{
uint64_t threadOrTaskId = static_cast<uint64_t>(getproctid());
#if defined(ENABLE_FFRT_INTERFACES)
threadOrTaskId = ffrt_this_task_get_id();
if (threadOrTaskId == 0) {
threadOrTaskId = static_cast<uint64_t>(getproctid());
}
#endif // defined(ENABLE_FFRT_INTERFACES)
return threadOrTaskId;
}
#endif // defined(OHOS_PLATFORM)
const DebuggerPostTask &GetDebuggerPostTask(int tid)
{
std::shared_lock<std::shared_mutex> lock(g_mutex);
if (g_debuggerInfo.find(tid) == g_debuggerInfo.end()) {
static DebuggerPostTask tempTask;
return tempTask;
}
return g_debuggerInfo[tid].second;
}
void *GetEcmaVM(int tid)
{
std::shared_lock<std::shared_mutex> lock(g_mutex);
if (g_debuggerInfo.find(tid) == g_debuggerInfo.end()) {
return nullptr;
}
return g_debuggerInfo[tid].first;
}
bool InitializeDebuggerForSocketpair(void* vm)
{
#if !defined(IOS_PLATFORM)
if (!LoadArkDebuggerLibrary()) {
return false;
}
#endif
if (!InitializeArkFunctions()) {
LOGE("Initialize ark functions failed");
return false;
}
g_initializeDebugger(vm, std::bind(&SendReply, vm, std::placeholders::_2));
return true;
}
// for ohos platform.
bool StartDebugForSocketpair(int tid, int socketfd)
{
LOGI("StartDebugForSocketpair, tid = %{private}d, socketfd = %{private}d", tid, socketfd);
void* vm = GetEcmaVM(tid);
if (vm == nullptr) {
LOGD("VM has already been destroyed");
return false;
}
g_vm = vm;
if (!InitializeDebuggerForSocketpair(vm)) {
return false;
}
const DebuggerPostTask &debuggerPostTask = GetDebuggerPostTask(tid);
DebugInfo debugInfo = {socketfd};
if (!InitializeInspector(vm, debuggerPostTask, debugInfo, tid)) {
LOGE("Initialize inspector failed");
return false;
}
return true;
}
// for cross-platform, previewer and old process of StartDebugger.
bool StartDebug(const std::string& componentName, void* vm, bool isDebugMode,
int32_t instanceId, const DebuggerPostTask& debuggerPostTask, int port)
{
LOGI("StartDebug, componentName = %{private}s, isDebugMode = %{private}d, instanceId = %{private}d",
componentName.c_str(), isDebugMode, instanceId);
g_vm = vm;
#if !defined(IOS_PLATFORM)
if (!LoadArkDebuggerLibrary()) {
return false;
}
#endif
if (!InitializeArkFunctions()) {
LOGE("Initialize ark functions failed");
return false;
}
g_initializeDebugger(vm, std::bind(&SendReply, vm, std::placeholders::_2));
int startDebugInOldProcess = -2; // start debug in old process.
DebugInfo debugInfo = {startDebugInOldProcess, componentName, instanceId, port};
if (!InitializeInspector(vm, debuggerPostTask, debugInfo)) {
LOGE("Initialize inspector failed");
return false;
}
if (isDebugMode && port > 0) {
LOGI("Wait for debugger for previewer");
g_waitForDebugger(vm);
}
return true;
}
void WaitForDebugger(void* vm)
{
LOGI("WaitForDebugger");
g_waitForDebugger(vm);
}
void StopDebug(void* vm)
{
LOGI("StopDebug start, vm is %{private}p", vm);
std::unique_lock<std::shared_mutex> lock(g_mutex);
auto iter = g_inspectors.find(vm);
if (iter == g_inspectors.end() || iter->second == nullptr) {
return;
}
#ifdef PANDA_TARGET_MACOS
uint32_t tid = static_cast<uint32_t>(reinterpret_cast<uintptr_t>(g_inspectors[vm]->tid_));
#else
uint32_t tid = g_inspectors[vm]->tid_;
#endif
auto debuggerInfo = g_debuggerInfo.find(tid);
if (debuggerInfo != g_debuggerInfo.end()) {
g_debuggerInfo.erase(debuggerInfo);
}
g_uninitializeDebugger(vm);
ResetServiceLocked(vm, true);
LOGI("StopDebug end");
}
void StopOldDebug(int tid, const std::string& componentName)
{
LOGI("StopDebug start, componentName = %{private}s, tid = %{private}d", componentName.c_str(), tid);
void* vm = GetEcmaVM(tid);
if (vm == nullptr) {
return;
}
std::unique_lock<std::shared_mutex> lock(g_mutex);
auto iter = g_inspectors.find(vm);
if (iter == g_inspectors.end() || iter->second == nullptr) {
return;
}
ResetServiceLocked(vm, false);
LOGI("StopDebug end");
}
// for socketpair process.
void StoreDebuggerInfo(int tid, void* vm, const DebuggerPostTask& debuggerPostTask)
{
std::unique_lock<std::shared_mutex> lock(g_mutex);
if (g_debuggerInfo.find(tid) == g_debuggerInfo.end()) {
g_debuggerInfo.emplace(tid, std::make_pair(vm, debuggerPostTask));
}
}
} // namespace OHOS::ArkCompiler::Toolchain
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C++
1
https://gitee.com/wwzm-jjw/arkcompiler_toolchain.git
git@gitee.com:wwzm-jjw/arkcompiler_toolchain.git
wwzm-jjw
arkcompiler_toolchain
arkcompiler_toolchain
master

搜索帮助