blob: 8b616c7e8defdf74ec7def01f49b98fdafceef2b [file] [log] [blame]
// Copyright 2020 The Fuchsia 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/developer/forensics/exceptions/handler/main.h"
#include <lib/async-loop/cpp/loop.h>
#include <lib/async-loop/default.h>
#include <lib/component/incoming/cpp/protocol.h>
#include <lib/fidl/cpp/binding.h>
#include <lib/sys/cpp/service_directory.h>
#include <lib/syslog/cpp/log_settings.h>
#include <lib/syslog/cpp/macros.h>
#include <lib/zx/time.h>
#include <zircon/errors.h>
#include <zircon/processargs.h>
#include <memory>
#include <string>
#include "src/developer/forensics/exceptions/constants.h"
#include "src/developer/forensics/exceptions/handler/crash_reporter.h"
#include "src/lib/fxl/strings/substitute.h"
namespace forensics {
namespace exceptions {
namespace handler {
namespace {
std::string ExtractHandlerIndex(const std::string& process_name) {
// The process name should be of the form "exception_handler_001".
auto first_num = process_name.find_last_of('_');
FX_CHECK(first_num != std::string::npos);
FX_CHECK((++first_num) != std::string::npos);
return process_name.substr(first_num);
}
// Returns nullptr if there's an error connecting to required protocols.
std::unique_ptr<WakeLease> CreateWakeLease(async_dispatcher_t* dispatcher,
const std::string& handler_index) {
zx::result sag_client_end = component::Connect<fuchsia_power_system::ActivityGovernor>();
if (!sag_client_end.is_ok()) {
FX_LOGS(ERROR)
<< "Synchronous error when connecting to the |fuchsia_power_system::ActivityGovernor| protocol: "
<< sag_client_end.status_string();
return nullptr;
}
const std::string lease_name = fxl::Substitute("exceptions-element-$0", handler_index);
return std::make_unique<WakeLease>(dispatcher, lease_name, std::move(sag_client_end).value());
}
} // namespace
int main(const std::string& process_name, const std::string& suspend_enabled_flag) {
async::Loop loop(&kAsyncLoopConfigAttachToCurrentThread);
using forensics::exceptions::kComponentLookupTimeout;
using Binding = fidl::Binding<forensics::exceptions::handler::CrashReporter,
std::unique_ptr<fuchsia::exception::internal::CrashReporter>>;
const std::string handler_index = ExtractHandlerIndex(process_name);
fuchsia_logging::LogSettingsBuilder()
.WithTags({"forensics", "exception", handler_index})
.BuildAndInitialize();
// We receive a channel that we interpret as a fuchsia.exception.internal.CrashReporter
// connection.
zx::channel channel(zx_take_startup_handle(PA_HND(PA_USER0, 0)));
if (!channel.is_valid()) {
FX_LOGS(FATAL) << "Received invalid channel";
return EXIT_FAILURE;
}
fidl::ClientEnd<fuchsia_driver_crash::CrashIntrospect> driver_crash_introspect_client_end;
{
zx::result driver_crash_introspect_result =
component::Connect<fuchsia_driver_crash::CrashIntrospect>();
if (driver_crash_introspect_result.is_error()) {
FX_LOGS(WARNING) << "Failed to connect to driver crash introspect channel.";
} else {
driver_crash_introspect_client_end = std::move(driver_crash_introspect_result).value();
}
}
std::unique_ptr<WakeLease> wake_lease = suspend_enabled_flag == kSuspendEnabledFlag
? CreateWakeLease(loop.dispatcher(), handler_index)
: nullptr;
auto crash_reporter = std::make_unique<forensics::exceptions::handler::CrashReporter>(
loop.dispatcher(), sys::ServiceDirectory::CreateFromNamespace(), kComponentLookupTimeout,
std::move(wake_lease), std::move(driver_crash_introspect_client_end));
Binding crash_reporter_binding(std::move(crash_reporter), std::move(channel), loop.dispatcher());
crash_reporter_binding.set_error_handler([&loop](const zx_status_t status) {
FX_PLOGS(WARNING, status) << "Lost connection to client";
loop.Shutdown();
});
loop.Run();
crash_reporter_binding.Close(ZX_OK);
return EXIT_SUCCESS;
}
} // namespace handler
} // namespace exceptions
} // namespace forensics