blob: 332756ef7e0fa474718a83be4f68364dad2fb0fd [file]
/*
* Copyright (C) 2026 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 "src/trace_processor/shell/shell_utils.h"
#include <cinttypes>
#include <cstdio>
#include <string>
#include "perfetto/base/build_config.h"
#include "perfetto/base/logging.h"
#include "perfetto/base/status.h"
#include "perfetto/ext/base/file_utils.h"
#include "perfetto/ext/base/scoped_file.h"
#include "perfetto/ext/base/status_macros.h"
#include "perfetto/ext/base/string_utils.h"
#include "perfetto/trace_processor/basic_types.h"
#include "perfetto/trace_processor/trace_processor.h"
#if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN)
#include <unistd.h>
#else
#include <io.h>
#define ftruncate _chsize
#endif
namespace perfetto::trace_processor {
bool StderrSupportsColors() {
#if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) && \
!PERFETTO_BUILDFLAG(PERFETTO_OS_WASM) && \
!PERFETTO_BUILDFLAG(PERFETTO_CHROMIUM_BUILD)
static const bool use_colors = isatty(STDERR_FILENO);
return use_colors;
#else
return false;
#endif
}
namespace {
base::Status PrintStatsSection(TraceProcessor* tp,
const char* header,
const char* color,
const char* query) {
const bool colors = StderrSupportsColors();
auto it = tp->ExecuteQuery(query);
bool first = true;
while (it.Next()) {
if (first) {
base::StackString<256> line(" %s%s%s", colors ? color : "", header,
colors ? "\x1b[0m" : "");
fprintf(stderr, "%s\n", line.c_str());
first = false;
}
// Columns: name, idx, value, description.
const char* name = it.Get(0).string_value;
SqlValue idx = it.Get(1);
int64_t value = it.Get(2).long_value;
const char* description = it.Get(3).string_value;
if (idx.type == SqlValue::Type::kNull) {
base::StackString<512> line(" %s: %" PRIi64, name, value);
fprintf(stderr, "%s", line.c_str());
} else {
base::StackString<512> line(" %s[%" PRIi64 "]: %" PRIi64, name,
idx.long_value, value);
fprintf(stderr, "%s", line.c_str());
}
if (description && description[0] != '\0') {
base::StackString<512> desc(" | %s", description);
fprintf(stderr, "%s", desc.c_str());
}
fprintf(stderr, "\n");
}
if (!first) {
fprintf(stderr, "\n");
}
base::Status status = it.Status();
if (!status.ok()) {
return base::ErrStatus("Error while iterating stats (%s)",
status.c_message());
}
return base::OkStatus();
}
} // namespace
base::Status PrintStats(TraceProcessor* tp) {
// Quick check: are there any issues at all?
auto check = tp->ExecuteQuery(
"SELECT 1 FROM stats "
"WHERE severity IN ('error', 'data_loss') AND value > 0 LIMIT 1");
bool has_issues = check.Next();
{
base::Status s = check.Status();
if (!s.ok())
return s;
}
if (!has_issues)
return base::OkStatus();
const bool colors = StderrSupportsColors();
base::StackString<64> title("\n%sTrace health issues:%s\n",
colors ? "\x1b[1;33m" : "",
colors ? "\x1b[0m" : "");
fprintf(stderr, "\n%s\n", title.c_str());
RETURN_IF_ERROR(PrintStatsSection(
tp, "Trace errors", "\x1b[1;31m",
"SELECT name, idx, value, description FROM stats "
"WHERE severity = 'error' AND source = 'trace' AND value > 0 "
"ORDER BY name, idx"));
RETURN_IF_ERROR(PrintStatsSection(
tp, "Import errors", "\x1b[1;31m",
"SELECT name, idx, value, description FROM stats "
"WHERE severity = 'error' AND source = 'analysis' AND value > 0 "
"ORDER BY name, idx"));
RETURN_IF_ERROR(
PrintStatsSection(tp, "Data losses", "\x1b[1;33m",
"SELECT name, idx, value, description FROM stats "
"WHERE severity = 'data_loss' AND value > 0 "
"ORDER BY name, idx"));
return base::OkStatus();
}
base::Status ExportTraceToDatabase(TraceProcessor* trace_processor,
const std::string& output_name) {
PERFETTO_CHECK(output_name.find('\'') == std::string::npos);
{
base::ScopedFile fd(base::OpenFile(output_name, O_CREAT | O_RDWR, 0600));
if (!fd)
return base::ErrStatus("Failed to create file: %s", output_name.c_str());
int res = ftruncate(fd.get(), 0);
PERFETTO_CHECK(res == 0);
}
std::string attach_sql =
"ATTACH DATABASE '" + output_name + "' AS perfetto_export";
auto attach_it = trace_processor->ExecuteQuery(attach_sql);
bool attach_has_more = attach_it.Next();
PERFETTO_DCHECK(!attach_has_more);
RETURN_IF_ERROR(attach_it.Status());
// Export real and virtual tables.
auto tables_it =
trace_processor->ExecuteQuery("SELECT name FROM perfetto_tables");
while (tables_it.Next()) {
std::string table_name = tables_it.Get(0).string_value;
PERFETTO_CHECK(!base::Contains(table_name, '\''));
std::string export_sql = "CREATE TABLE perfetto_export." + table_name +
" AS SELECT * FROM " + table_name;
auto export_it = trace_processor->ExecuteQuery(export_sql);
bool export_has_more = export_it.Next();
PERFETTO_DCHECK(!export_has_more);
RETURN_IF_ERROR(export_it.Status());
}
RETURN_IF_ERROR(tables_it.Status());
// Export views.
auto views_it = trace_processor->ExecuteQuery(
"SELECT sql FROM sqlite_master WHERE type='view'");
while (views_it.Next()) {
std::string sql = views_it.Get(0).string_value;
// View statements are of the form "CREATE VIEW name AS stmt". We need to
// rewrite name to point to the exported db.
const std::string kPrefix = "CREATE VIEW ";
PERFETTO_CHECK(sql.find(kPrefix) == 0);
sql = sql.substr(0, kPrefix.size()) + "perfetto_export." +
sql.substr(kPrefix.size());
auto export_it = trace_processor->ExecuteQuery(sql);
bool export_has_more = export_it.Next();
PERFETTO_DCHECK(!export_has_more);
RETURN_IF_ERROR(export_it.Status());
}
RETURN_IF_ERROR(views_it.Status());
auto detach_it =
trace_processor->ExecuteQuery("DETACH DATABASE perfetto_export");
bool detach_has_more = attach_it.Next();
PERFETTO_DCHECK(!detach_has_more);
return detach_it.Status();
}
} // namespace perfetto::trace_processor