blob: ed1b09bb3a6551e61c158ceb95a2b05f2aabbdb5 [file] [log] [blame]
// Copyright 2018 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/debug/zxdb/console/console_main.h"
#include <lib/cmdline/args_parser.h>
#include <lib/fit/defer.h>
#include <cstdlib>
#include "src/developer/debug/shared/logging/logging.h"
#include "src/developer/debug/shared/message_loop_poll.h"
#include "src/developer/debug/zxdb/client/job.h"
#include "src/developer/debug/zxdb/client/session.h"
#include "src/developer/debug/zxdb/client/setting_schema_definition.h"
#include "src/developer/debug/zxdb/common/curl.h"
#include "src/developer/debug/zxdb/common/version.h"
#include "src/developer/debug/zxdb/console/actions.h"
#include "src/developer/debug/zxdb/console/analytics.h"
#include "src/developer/debug/zxdb/console/command_line_options.h"
#include "src/developer/debug/zxdb/console/console_impl.h"
#include "src/developer/debug/zxdb/console/output_buffer.h"
#include "src/developer/debug/zxdb/console/verbs.h"
#include "src/developer/debug/zxdb/debug_adapter/server.h"
#include "src/lib/fxl/strings/string_printf.h"
namespace zxdb {
namespace {
// Loads any actions specified on the command line into the vector.
Err SetupActions(const CommandLineOptions& options, std::vector<Action>* actions) {
if (options.core) {
if (options.connect || options.run) {
return Err("--core can't be used with commands to connect or run.");
}
std::string cmd = VerbToString(Verb::kOpenDump) + " " + *options.core;
actions->push_back(Action("Open Dump", [cmd](const Action&, const Session&, Console* console) {
console->ProcessInputLine(cmd.c_str(), ActionFlow::PostActionCallback, false);
}));
}
if (options.connect) {
std::string cmd = "connect " + *options.connect;
actions->push_back(Action("Connect", [cmd](const Action&, const Session&, Console* console) {
console->ProcessInputLine(cmd.c_str(), ActionFlow::PostActionCallback, false);
}));
}
if (options.unix_connect) {
std::string cmd = "connect -u " + *options.unix_connect;
actions->push_back(Action("Connect", [cmd](const Action&, const Session&, Console* console) {
console->ProcessInputLine(cmd.c_str(), ActionFlow::PostActionCallback, false);
}));
}
if (options.run) {
std::string cmd = "run " + *options.run;
actions->push_back(Action("Run", [cmd](const Action&, const Session&, Console* console) {
console->ProcessInputLine(cmd.c_str(), ActionFlow::PostActionCallback, false);
}));
}
if (options.script_file) {
Err err = ScriptFileToActions(*options.script_file, actions);
if (err.has_error())
return err;
}
for (const auto& filter : options.filter) {
std::string cmd = "filter attach " + filter;
actions->push_back(Action("Filter", [cmd](const Action&, const Session&, Console* console) {
console->ProcessInputLine(cmd.c_str(), ActionFlow::PostActionCallback, false);
}));
}
return Err();
}
void InitConsole(zxdb::Console& console) {
console.Init();
// Help text.
OutputBuffer help;
help.Append(Syntax::kWarning, "👉 ");
help.Append(Syntax::kComment, "To get started, try \"status\" or \"help\".");
console.Output(help);
}
void ScheduleActions(zxdb::Session& session, zxdb::Console& console,
std::vector<zxdb::Action> actions) {
auto callback = [&](zxdb::Err err) {
std::string msg;
if (!err.has_error()) {
msg = "All actions were executed successfully.";
} else if (err.type() == zxdb::ErrType::kCanceled) {
msg = "Action processing was cancelled.";
} else {
msg = fxl::StringPrintf("Error executing actions: %s", err.msg().c_str());
}
// Go into interactive mode.
InitConsole(console);
};
// This will add the actions to the MessageLoop and oversee that all the actions run or the flow
// is interrupted if one of them fails. Actions run on a singleton ActionFlow instance.
zxdb::ActionFlow& flow = zxdb::ActionFlow::Singleton();
flow.ScheduleActions(std::move(actions), &session, &console, callback);
}
void SetupCommandLineOptions(const CommandLineOptions& options, Session* session) {
auto& system_settings = session->system().settings();
if (options.symbol_cache) {
system_settings.SetString(ClientSettings::System::kSymbolCache, *options.symbol_cache);
}
if (!options.symbol_index_files.empty()) {
system_settings.SetList(ClientSettings::System::kSymbolIndexFiles, options.symbol_index_files);
}
if (!options.symbol_servers.empty()) {
system_settings.SetList(ClientSettings::System::kSymbolServers, options.symbol_servers);
}
if (!options.symbol_paths.empty()) {
system_settings.SetList(ClientSettings::System::kSymbolPaths, options.symbol_paths);
}
if (!options.build_id_dirs.empty()) {
system_settings.SetList(ClientSettings::System::kBuildIdDirs, options.build_id_dirs);
}
if (!options.ids_txts.empty()) {
system_settings.SetList(ClientSettings::System::kIdsTxts, options.ids_txts);
}
if (!options.build_dirs.empty()) {
system_settings.SetList(ClientSettings::Target::kBuildDirs, options.build_dirs);
}
}
} // namespace
int ConsoleMain(int argc, const char* argv[]) {
using ::analytics::core_dev_tools::EarlyProcessAnalyticsOptions;
Curl::GlobalInit();
auto deferred_cleanup_curl = fit::defer(Curl::GlobalCleanup);
auto deferred_cleanup_analytics = fit::defer(Analytics::CleanUp);
CommandLineOptions options;
std::vector<std::string> params;
cmdline::Status status = ParseCommandLine(argc, argv, &options, &params);
if (status.has_error()) {
fprintf(stderr, "%s", status.error_message().c_str());
return 1;
}
if (options.requested_version) {
printf("Version: %s\n", kBuildVersion);
return 0;
}
if (EarlyProcessAnalyticsOptions<Analytics>(options.analytics, options.analytics_show)) {
return 0;
}
std::vector<zxdb::Action> actions;
Err err = SetupActions(options, &actions);
if (err.has_error()) {
fprintf(stderr, "%s\n", err.msg().c_str());
return 1;
}
debug::MessageLoopPoll loop;
std::string error_message;
if (!loop.Init(&error_message)) {
fprintf(stderr, "%s", error_message.c_str());
return 1;
}
// This scope forces all the objects to be destroyed before the Cleanup() call which will mark the
// message loop as not-current.
{
Session session;
Analytics::Init(session, options.analytics);
Analytics::IfEnabledSendInvokeEvent(&session);
debug::SetLogCategories({debug::LogCategory::kAll});
if (options.debug_mode) {
session.system().settings().SetBool(ClientSettings::System::kDebugMode, true);
}
if (options.no_auto_attach_limbo) {
session.system().settings().SetBool(ClientSettings::System::kAutoAttachLimbo, false);
}
std::unique_ptr<DebugAdapterServer> debug_adapter;
if (options.enable_debug_adapter) {
int port = options.debug_adapter_port;
debug_adapter = std::make_unique<DebugAdapterServer>(&session, port);
err = debug_adapter->Init();
if (err.has_error()) {
fprintf(stderr, "Failed to initialize debug adapter: %s\n", err.msg().c_str());
loop.Cleanup();
return EXIT_FAILURE;
}
}
ConsoleImpl console(&session);
if (options.quit_agent_on_quit) {
session.system().settings().SetBool(ClientSettings::System::kQuitAgentOnExit, true);
}
SetupCommandLineOptions(options, &session);
if (!actions.empty()) {
ScheduleActions(session, console, std::move(actions));
} else {
// Interactive mode is the default mode.
InitConsole(console);
}
loop.Run();
}
loop.Cleanup();
return 0;
}
} // namespace zxdb