blob: f37d2090d087c6a4dd4edfddebf66c169c81eb96 [file] [log] [blame]
/*
* Copyright (C) 2021 The Android Open Source Project
*
* 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 <stdio.h>
#include <fstream>
#include <string>
#include <android-base/file.h>
#include <android-base/logging.h>
#include <android-base/strings.h>
#include <gflags/gflags.h>
#include "common/libs/fs/shared_fd.h"
#include "common/libs/fs/shared_select.h"
#include "common/libs/utils/environment.h"
#include "common/libs/utils/files.h"
#include "common/libs/utils/known_paths.h"
#include "common/libs/utils/subprocess.h"
#include "common/libs/utils/tee_logging.h"
#include "host/libs/config/cuttlefish_config.h"
#include "ziparchive/zip_writer.h"
DEFINE_string(output, "host_bugreport.zip", "Where to write the output");
namespace cuttlefish {
namespace {
void SaveFile(ZipWriter& writer, const std::string& zip_path,
const std::string& file_path) {
writer.StartEntry(zip_path, ZipWriter::kCompress | ZipWriter::kAlign32);
std::fstream file(file_path, std::fstream::in | std::fstream::binary);
do {
char data[1024 * 10];
file.read(data, sizeof(data));
writer.WriteBytes(data, file.gcount());
} while (file);
writer.FinishEntry();
if (file.bad()) {
LOG(ERROR) << "Error in logging " << file_path << " to " << zip_path;
}
}
void AddNetsimdLogs(ZipWriter& writer) {
// The temp directory name depends on whether the `USER` environment variable
// is defined.
// https://source.corp.google.com/h/googleplex-android/platform/superproject/main/+/main:tools/netsim/rust/common/src/system/mod.rs;l=37-57;drc=360ddb57df49472a40275b125bb56af2a65395c7
std::string user = StringFromEnv("USER", "");
std::string dir = user.empty()
? TempDir() + "/android/netsimd"
: fmt::format("{}/android-{}/netsimd", TempDir(), user);
if (!DirectoryExists(dir)) {
LOG(INFO) << "netsimd logs directory: `" << dir << "` does not exist.";
return;
}
auto names = DirectoryContents(dir);
if (!names.ok()) {
LOG(ERROR) << "Cannot read from netsimd directory `" << dir
<< "`: " << names.error().FormatForEnv(/* color = */ false);
return;
}
for (const auto& name : names.value()) {
SaveFile(writer, "netsimd/" + name, dir + "/" + name);
}
}
Result<void> CreateDeviceBugreport(
const CuttlefishConfig::InstanceSpecific& ins, const std::string& out_dir) {
std::string adb_bin_path = HostBinaryPath("adb");
CF_EXPECT(FileExists(adb_bin_path),
"adb binary not found at: " << adb_bin_path);
Command connect_cmd("timeout");
connect_cmd.SetWorkingDirectory(
"/"); // Use a deterministic working directory
connect_cmd.AddParameter("30s")
.AddParameter(adb_bin_path)
.AddParameter("connect")
.AddParameter(ins.adb_ip_and_port());
CF_EXPECT_EQ(connect_cmd.Start().Wait(), 0, "adb connect failed");
Command wait_for_device_cmd("timeout");
wait_for_device_cmd.SetWorkingDirectory(
"/"); // Use a deterministic working directory
wait_for_device_cmd.AddParameter("30s")
.AddParameter(adb_bin_path)
.AddParameter("-s")
.AddParameter(ins.adb_ip_and_port())
.AddParameter("wait-for-device");
CF_EXPECT_EQ(wait_for_device_cmd.Start().Wait(), 0,
"adb wait-for-device failed");
Command bugreport_cmd("timeout");
bugreport_cmd.SetWorkingDirectory(
"/"); // Use a deterministic working directory
bugreport_cmd.AddParameter("300s")
.AddParameter(adb_bin_path)
.AddParameter("-s")
.AddParameter(ins.adb_ip_and_port())
.AddParameter("bugreport")
.AddParameter(out_dir);
CF_EXPECT_EQ(bugreport_cmd.Start().Wait(), 0, "adb bugreport failed");
return {};
}
Result<void> CvdHostBugreportMain(int argc, char** argv) {
::android::base::InitLogging(argv, android::base::StderrLogger);
google::ParseCommandLineFlags(&argc, &argv, true);
std::string log_filename = TempDir() + "/cvd_hbr.log.XXXXXX";
{
auto fd = SharedFD::Mkstemp(&log_filename);
CF_EXPECT(fd->IsOpen(), "Unable to create log file: " << fd->StrError());
android::base::SetLogger(TeeLogger({
{ConsoleSeverity(), SharedFD::Dup(2), MetadataLevel::ONLY_MESSAGE},
{LogFileSeverity(), fd, MetadataLevel::FULL},
}));
}
auto config = CuttlefishConfig::Get();
CHECK(config) << "Unable to find the config";
auto out_path = FLAGS_output.c_str();
std::unique_ptr<FILE, decltype(&fclose)> out(fopen(out_path, "wbe"), &fclose);
ZipWriter writer(out.get());
auto save = [&writer, config](const std::string& path) {
SaveFile(writer, "cuttlefish_assembly/" + path, config->AssemblyPath(path));
};
save("assemble_cvd.log");
save("cuttlefish_config.json");
for (const auto& instance : config->Instances()) {
auto save = [&writer, instance](const std::string& path) {
const auto& zip_name = instance.instance_name() + "/" + path;
const auto& file_name = instance.PerInstancePath(path.c_str());
SaveFile(writer, zip_name, file_name);
};
save("cuttlefish_config.json");
save("disk_config.txt");
if (DirectoryExists(instance.PerInstancePath("logs"))) {
auto result = DirectoryContents(instance.PerInstancePath("logs"));
if (result.ok()) {
for (const auto& log : result.value()) {
save("logs/" + log);
}
} else {
LOG(ERROR) << "Cannot read from logs directory: "
<< result.error().FormatForEnv(/* color = */ false);
}
} else {
save("kernel.log");
save("launcher.log");
save("logcat");
save("metrics.log");
}
{
auto result = DirectoryContents(instance.PerInstancePath("tombstones"));
if (result.ok()) {
for (const auto& tombstone : result.value()) {
save("tombstones/" + tombstone);
}
} else {
LOG(ERROR) << "Cannot read from tombstones directory: "
<< result.error().FormatForEnv(/* color = */ false);
}
}
{
auto result = DirectoryContents(instance.PerInstancePath("recording"));
if (result.ok()) {
for (const auto& recording : result.value()) {
save("recording/" + recording);
}
} else {
LOG(ERROR) << "Cannot read from recording directory: "
<< result.error().FormatForEnv(/* color = */ false);
}
}
{
// TODO(b/359657254) Create the `adb bugreport` asynchronously.
std::string device_br_dir = TempDir() + "/cvd_dbrXXXXXX";
CF_EXPECTF(mkdtemp(device_br_dir.data()) != nullptr,
"mkdtemp failed: '{}'", strerror(errno));
auto result = CreateDeviceBugreport(instance, device_br_dir);
if (result.ok()) {
auto names = DirectoryContents(device_br_dir);
if (names.ok()) {
for (const auto& name : names.value()) {
std::string filename = device_br_dir + "/" + name;
SaveFile(writer, android::base::Basename(filename), filename);
}
} else {
LOG(ERROR) << "Cannot read from device bugreport directory: "
<< names.error().FormatForEnv(/* color = */ false);
}
} else {
LOG(ERROR) << "Failed to create device bugreport: "
<< result.error().FormatForEnv(/* color = */ false);
}
static_cast<void>(RecursivelyRemoveDirectory(device_br_dir));
}
}
AddNetsimdLogs(writer);
LOG(INFO) << "Building cvd bugreport completed";
SaveFile(writer, "cvd_bugreport_builder.log", log_filename);
writer.Finish();
LOG(INFO) << "Saved to \"" << FLAGS_output << "\"";
if (!RemoveFile(log_filename)) {
LOG(INFO) << "Failed to remove host bug report log file: " << log_filename;
}
return {};
}
} // namespace
} // namespace cuttlefish
int main(int argc, char** argv) {
auto result = cuttlefish::CvdHostBugreportMain(argc, argv);
CHECK(result.ok()) << result.error().FormatForEnv();
return 0;
}