blob: d3c3e5d128ddd36c60b9ef8f454155b828497b58 [file] [log] [blame]
// Copyright 2017 The Cobalt Authors. All Rights Reserved.
//
// 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 "cobalt/h5vcc/h5vcc_crash_log.h"
#include <map>
#include <memory>
#include <string>
#include <vector>
#include "base/atomicops.h"
#include "base/memory/singleton.h"
#include "base/synchronization/lock.h"
#include "starboard/extension/crash_handler.h"
#include "starboard/system.h"
namespace cobalt {
namespace h5vcc {
bool H5vccCrashLog::SetString(const std::string& key,
const std::string& value) {
auto crash_handler_extension =
static_cast<const CobaltExtensionCrashHandlerApi*>(
SbSystemGetExtension(kCobaltExtensionCrashHandlerName));
if (crash_handler_extension && crash_handler_extension->version >= 2) {
return crash_handler_extension->SetString(key.c_str(), value.c_str());
}
// The platform has not implemented a version of the CrashHandler Cobalt
// Extension appropriate for this use case.
return false;
}
void H5vccCrashLog::TriggerCrash(H5vccCrashType intent) {
if (intent == kH5vccCrashTypeNullDereference) {
*(reinterpret_cast<volatile char*>(0)) = 0;
}
if (intent == kH5vccCrashTypeIllegalInstruction) {
#if SB_IS(ARCH_ARM) || SB_IS(ARCH_ARM64)
__asm(".word 0xf7f0a000\n");
#elif !SB_IS(ARCH_X64) // inline asm not allowed on 64bit MSVC
__asm("ud2");
#endif
}
if (intent == kH5vccCrashTypeDebugger) {
SbSystemBreakIntoDebugger();
}
if (intent == kH5vccCrashTypeOutOfMemory) {
void* p = nullptr;
posix_memalign(&p, 128, SIZE_MAX);
}
}
bool H5vccCrashLog::Register(const std::string& name,
const std::string& description,
WatchdogState watchdog_state,
int64_t time_interval_milliseconds,
int64_t time_wait_milliseconds,
WatchdogReplace watchdog_replace) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) {
base::ApplicationState monitor_state;
switch (watchdog_state) {
case kWatchdogStateStarted:
monitor_state = base::kApplicationStateStarted;
break;
case kWatchdogStateBlurred:
monitor_state = base::kApplicationStateBlurred;
break;
case kWatchdogStateConcealed:
monitor_state = base::kApplicationStateConcealed;
break;
case kWatchdogStateFrozen:
monitor_state = base::kApplicationStateFrozen;
break;
default:
monitor_state = base::kApplicationStateStarted;
}
watchdog::Replace replace;
switch (watchdog_replace) {
case kWatchdogReplaceNone:
replace = watchdog::NONE;
break;
case kWatchdogReplacePing:
replace = watchdog::PING;
break;
case kWatchdogReplaceAll:
replace = watchdog::ALL;
break;
default:
replace = watchdog::NONE;
}
return watchdog->Register(name, description, monitor_state,
time_interval_milliseconds * 1000,
time_wait_milliseconds * 1000, replace);
}
return false;
}
bool H5vccCrashLog::Unregister(const std::string& name) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) return watchdog->Unregister(name);
return false;
}
bool H5vccCrashLog::Ping(const std::string& name,
const std::string& ping_info) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) return watchdog->Ping(name, ping_info);
return false;
}
std::string H5vccCrashLog::GetWatchdogViolations(
const script::Sequence<std::string>& clients) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) {
// If not clients name is given, return all clients' data.
if (clients.size() == 0) {
return watchdog->GetWatchdogViolations();
}
std::vector<std::string> client_names;
for (script::Sequence<std::string>::size_type i = 0; i < clients.size();
++i) {
client_names.push_back(clients.at(i).c_str());
}
return watchdog->GetWatchdogViolations(client_names);
}
return "";
}
script::Sequence<std::string> H5vccCrashLog::GetWatchdogViolationClients() {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
script::Sequence<std::string> client_names;
if (watchdog) {
std::vector<std::string> client_string_names =
watchdog->GetWatchdogViolationClientNames();
for (std::size_t i = 0; i < client_string_names.size(); ++i) {
client_names.push_back(client_string_names[i]);
}
}
return client_names;
}
bool H5vccCrashLog::GetPersistentSettingWatchdogEnable() {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) return watchdog->GetPersistentSettingWatchdogEnable();
return true;
}
void H5vccCrashLog::SetPersistentSettingWatchdogEnable(bool enable_watchdog) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) watchdog->SetPersistentSettingWatchdogEnable(enable_watchdog);
}
bool H5vccCrashLog::GetPersistentSettingWatchdogCrash() {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) return watchdog->GetPersistentSettingWatchdogCrash();
return false;
}
void H5vccCrashLog::SetPersistentSettingWatchdogCrash(bool can_trigger_crash) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) watchdog->SetPersistentSettingWatchdogCrash(can_trigger_crash);
}
bool H5vccCrashLog::LogEvent(const std::string& event) {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (!watchdog) {
return false;
}
return watchdog->LogEvent(event);
}
script::Sequence<std::string> H5vccCrashLog::GetLogTrace() {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
script::Sequence<std::string> sequence;
if (watchdog) {
std::vector<std::string> logTrace = watchdog->GetLogTrace();
for (std::size_t i = 0; i < logTrace.size(); ++i) {
sequence.push_back(logTrace[i]);
}
}
return sequence;
}
void H5vccCrashLog::ClearLog() {
watchdog::Watchdog* watchdog = watchdog::Watchdog::GetInstance();
if (watchdog) {
watchdog->ClearLog();
}
}
} // namespace h5vcc
} // namespace cobalt