blob: 6499770c2518b6de84735a5156e4352697a62e70 [file] [log] [blame]
// Copyright 2019 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 <signal.h>
#include <stdlib.h>
#include <fstream>
#include <optional>
#include <string>
#include <thread>
#include <vector>
#include "tools/fidlcat/command_line_options.h"
#include "tools/fidlcat/lib/interception_workflow.h"
// TODO: Look into this. Removing the hack that led to this (in
// debug_ipc/helper/message_loop.h) seems to work, except it breaks SDK builds
// on CQ in a way I can't repro locally.
#undef __TA_REQUIRES
#include "lib/fidl/cpp/message.h"
#include "src/developer/debug/zxdb/console/command_utils.h"
#include "tools/fidlcat/lib/library_loader.h"
#include "tools/fidlcat/lib/message_decoder.h"
#include "tools/fidlcat/lib/zx_channel_params.h"
namespace fidlcat {
static bool called_onexit_once_ = false;
static std::atomic<InterceptionWorkflow*> workflow_;
static void OnExit(int signum, siginfo_t* info, void* ptr) {
if (called_onexit_once_) {
// Exit immediately.
#if defined(__APPLE__)
_Exit(1);
#else
_exit(1);
#endif
} else {
// Maybe detach cleanly here, if we can.
FXL_LOG(INFO) << "Shutting down...";
called_onexit_once_ = true;
workflow_.load()->Shutdown();
}
}
void CatchSigterm() {
static struct sigaction action;
memset(&action, 0, sizeof(action));
action.sa_sigaction = OnExit;
action.sa_flags = SA_SIGINFO;
sigaction(SIGINT, &action, NULL);
}
// Add the startup actions to the loop: connect, attach to pid, set breakpoints.
void EnqueueStartup(
InterceptionWorkflow& workflow, LibraryLoader& loader,
std::map<std::tuple<zx_handle_t, uint64_t>, Direction>* handle_directions,
const CommandLineOptions& options, const DisplayOptions& display_options,
std::vector<std::string>& params) {
uint64_t process_koid = ULLONG_MAX;
if (options.remote_pid) {
const std::string& pid_str = *options.remote_pid;
process_koid = strtoull(pid_str.c_str(), nullptr, 10);
// There is no process 0, and if there were, we probably wouldn't be able to
// talk with it.
if (process_koid == 0) {
fprintf(stderr, "Invalid pid %s\n", pid_str.c_str());
exit(1);
}
}
// Currently, we only display messages for one process. process_koid is
// correctly set when we attach to a process but is ULLONG_MAX when we create
// the process.
workflow.SetZxChannelWriteCallback(
[loader = &loader, handle_directions, &display_options, process_koid](
const zxdb::Err& err, const ZxChannelParams& params) {
if (!err.ok()) {
FXL_LOG(INFO) << "Unable to decode zx_channel_write params: "
<< err.msg();
return;
}
fidl::BytePart bytes(params.GetBytes().get(), params.GetNumBytes(),
params.GetNumBytes());
fidl::HandlePart handles(params.GetHandles().get(),
params.GetNumHandles(),
params.GetNumHandles());
fidl::Message message(std::move(bytes), std::move(handles));
DecodeMessage(loader, handle_directions, display_options, process_koid,
params.GetHandle(), message, /*read=*/false, std::cout);
});
workflow.SetZxChannelReadCallback(
[loader = &loader, handle_directions, &display_options, process_koid](
const zxdb::Err& err, const ZxChannelParams& params) {
if (!err.ok()) {
FXL_LOG(INFO) << "Unable to decode zx_channel_read params: "
<< err.msg();
return;
}
fidl::BytePart bytes(params.GetBytes().get(), params.GetNumBytes(),
params.GetNumBytes());
fidl::HandlePart handles(params.GetHandles().get(),
params.GetNumHandles(),
params.GetNumHandles());
fidl::Message message(std::move(bytes), std::move(handles));
DecodeMessage(loader, handle_directions, display_options, process_koid,
params.GetHandle(), message, /*read=*/true, std::cout);
});
std::string host;
uint16_t port;
zxdb::Err parse_err = zxdb::ParseHostPort(*(options.connect), &host, &port);
if (!parse_err.ok()) {
FXL_LOG(FATAL) << "Could not parse host/port pair: " << parse_err.msg();
}
auto set_breakpoints = [&workflow, process_koid](const zxdb::Err& err) {
workflow.SetBreakpoints(process_koid);
};
auto attach = [&workflow, process_koid, filter = options.filter, params,
set_breakpoints =
std::move(set_breakpoints)](const zxdb::Err& err) {
if (!err.ok()) {
FXL_LOG(FATAL) << "Unable to connect: " << err.msg();
return;
}
FXL_LOG(INFO) << "Connected!";
if (process_koid != ULLONG_MAX) {
workflow.Attach(process_koid, set_breakpoints);
} else if (!filter.empty()) {
workflow.Filter(filter, set_breakpoints);
} else {
workflow.Launch(params, set_breakpoints);
}
};
auto connect = [&workflow, attach = std::move(attach), host, port]() {
FXL_LOG(INFO) << "Connecting to port " << port << " on " << host << "...";
workflow.Connect(host, port, attach);
};
debug_ipc::MessageLoop::Current()->PostTask(FROM_HERE, connect);
}
int ConsoleMain(int argc, const char* argv[]) {
CommandLineOptions options;
DisplayOptions display_options;
std::vector<std::string> params;
cmdline::Status status =
ParseCommandLine(argc, argv, &options, &display_options, &params);
if (status.has_error()) {
fprintf(stderr, "%s\n", status.error_message().c_str());
return 1;
}
InterceptionWorkflow workflow;
workflow.Initialize(options.symbol_paths);
std::vector<std::unique_ptr<std::istream>> paths;
std::vector<std::string> bad_paths;
ExpandFidlPathsFromOptions(options.fidl_ir_paths, paths, bad_paths);
if (paths.size() == 0) {
std::string error = "No FIDL IR paths provided.";
if (bad_paths.size() != 0) {
error.append(" File(s) not found: [ ");
for (auto& s : bad_paths) {
error.append(s);
error.append(" ");
}
error.append("]");
}
FXL_LOG(INFO) << error;
}
fidlcat::LibraryReadError loader_err;
LibraryLoader loader(paths, &loader_err);
if (loader_err.value != fidlcat::LibraryReadError::kOk) {
FXL_LOG(ERROR) << "Failed to read libraries";
return 1;
}
std::map<std::tuple<zx_handle_t, uint64_t>, Direction> handle_directions;
EnqueueStartup(workflow, loader, &handle_directions, options, display_options,
params);
// TODO: When the attached koid terminates normally, we should exit and call
// QuitNow() on the MessageLoop.
workflow_.store(&workflow);
CatchSigterm();
workflow.Go();
return 0;
}
} // namespace fidlcat
int main(int argc, const char* argv[]) { fidlcat::ConsoleMain(argc, argv); }