blob: 8f35739abbba7f9e6dfa675e4a92d0e7a8d500dc [file] [log] [blame]
/*
* Copyright (C) 2024 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_redaction/trace_redactor.h"
#include <cstddef>
#include <string>
#include <string_view>
#include "perfetto/base/status.h"
#include "perfetto/ext/base/file_utils.h"
#include "perfetto/ext/base/scoped_file.h"
#include "perfetto/ext/base/scoped_mmap.h"
#include "perfetto/protozero/scattered_heap_buffer.h"
#include "perfetto/trace_processor/trace_blob.h"
#include "perfetto/trace_processor/trace_blob_view.h"
#include "src/trace_processor/util/status_macros.h"
#include "src/trace_redaction/broadphase_packet_filter.h"
#include "src/trace_redaction/collect_frame_cookies.h"
#include "src/trace_redaction/collect_system_info.h"
#include "src/trace_redaction/collect_timeline_events.h"
#include "src/trace_redaction/find_package_uid.h"
#include "src/trace_redaction/merge_threads.h"
#include "src/trace_redaction/populate_allow_lists.h"
#include "src/trace_redaction/prune_package_list.h"
#include "src/trace_redaction/redact_ftrace_events.h"
#include "src/trace_redaction/redact_process_events.h"
#include "src/trace_redaction/redact_process_trees.h"
#include "src/trace_redaction/scrub_process_stats.h"
#include "src/trace_redaction/trace_redaction_framework.h"
#include "src/trace_redaction/verify_integrity.h"
#include "protos/perfetto/trace/trace.pbzero.h"
namespace perfetto::trace_redaction {
using Trace = protos::pbzero::Trace;
using TracePacket = protos::pbzero::TracePacket;
TraceRedactor::TraceRedactor() = default;
TraceRedactor::~TraceRedactor() = default;
base::Status TraceRedactor::Redact(std::string_view source_filename,
std::string_view dest_filename,
Context* context) const {
const std::string source_filename_str(source_filename);
base::ScopedMmap mapped =
base::ReadMmapWholeFile(source_filename_str.c_str());
if (!mapped.IsValid()) {
return base::ErrStatus("TraceRedactor: failed to map pages for trace (%s)",
source_filename_str.c_str());
}
trace_processor::TraceBlobView whole_view(
trace_processor::TraceBlob::FromMmap(std::move(mapped)));
RETURN_IF_ERROR(Collect(context, whole_view));
for (const auto& builder : builders_) {
RETURN_IF_ERROR(builder->Build(context));
}
return Transform(*context, whole_view, std::string(dest_filename));
}
base::Status TraceRedactor::Collect(
Context* context,
const trace_processor::TraceBlobView& view) const {
for (const auto& collector : collectors_) {
RETURN_IF_ERROR(collector->Begin(context));
}
const Trace::Decoder trace_decoder(view.data(), view.length());
for (auto packet_it = trace_decoder.packet(); packet_it; ++packet_it) {
const TracePacket::Decoder packet(packet_it->as_bytes());
for (auto& collector : collectors_) {
RETURN_IF_ERROR(collector->Collect(packet, context));
}
}
for (const auto& collector : collectors_) {
RETURN_IF_ERROR(collector->End(context));
}
return base::OkStatus();
}
base::Status TraceRedactor::Transform(
const Context& context,
const trace_processor::TraceBlobView& view,
const std::string& dest_file) const {
std::ignore = context;
const auto dest_fd = base::OpenFile(dest_file, O_RDWR | O_CREAT, 0666);
if (dest_fd.get() == -1) {
return base::ErrStatus(
"Failed to open destination file; can't write redacted trace.");
}
const Trace::Decoder trace_decoder(view.data(), view.length());
for (auto packet_it = trace_decoder.packet(); packet_it; ++packet_it) {
auto packet = packet_it->as_std_string();
for (const auto& transformer : transformers_) {
// If the packet has been cleared, it means a tranformation has removed it
// from the trace. Stop processing it. This saves transforms from having
// to check and handle empty packets.
if (packet.empty()) {
break;
}
RETURN_IF_ERROR(transformer->Transform(context, &packet));
}
// The packet has been removed from the trace. Don't write an empty packet
// to disk.
if (packet.empty()) {
continue;
}
protozero::HeapBuffered<protos::pbzero::Trace> serializer;
serializer->add_packet()->AppendRawProtoBytes(packet.data(), packet.size());
packet.assign(serializer.SerializeAsString());
if (const auto exported_data =
base::WriteAll(dest_fd.get(), packet.data(), packet.size());
exported_data <= 0) {
return base::ErrStatus(
"TraceRedactor: failed to write redacted trace to disk");
}
}
return base::OkStatus();
}
std::unique_ptr<TraceRedactor> TraceRedactor::CreateInstance(
const Config& config) {
auto redactor = std::make_unique<TraceRedactor>();
// VerifyIntegrity breaks the CollectPrimitive pattern. Instead of writing to
// the context, its job is to read trace packets and return errors if any
// packet does not look "correct". This primitive is added first in an effort
// to detect and react to bad input before other collectors run.
if (config.verify) {
redactor->emplace_collect<VerifyIntegrity>();
}
// Add all collectors.
redactor->emplace_collect<FindPackageUid>();
redactor->emplace_collect<CollectTimelineEvents>();
redactor->emplace_collect<CollectFrameCookies>();
redactor->emplace_collect<CollectSystemInfo>();
// Add all builders.
redactor->emplace_build<ReduceFrameCookies>();
redactor->emplace_build<BuildSyntheticThreads>();
{
// In order for BroadphasePacketFilter to work, something needs to populate
// the masks (i.e. PopulateAllowlists).
redactor->emplace_build<PopulateAllowlists>();
redactor->emplace_transform<BroadphasePacketFilter>();
}
{
auto* primitive = redactor->emplace_transform<RedactFtraceEvents>();
primitive->emplace_ftrace_filter<FilterRss>();
primitive->emplace_post_filter_modifier<DoNothing>();
}
{
auto* primitive = redactor->emplace_transform<RedactFtraceEvents>();
primitive->emplace_ftrace_filter<FilterFtraceUsingSuspendResume>();
primitive->emplace_post_filter_modifier<DoNothing>();
}
{
// Remove all frame timeline events that don't belong to the target package.
redactor->emplace_transform<FilterFrameEvents>();
}
redactor->emplace_transform<PrunePackageList>();
// Process stats includes per-process information, such as:
//
// processes {
// pid: 1
// vm_size_kb: 11716992
// vm_rss_kb: 5396
// rss_anon_kb: 2896
// rss_file_kb: 1728
// rss_shmem_kb: 772
// vm_swap_kb: 4236
// vm_locked_kb: 0
// vm_hwm_kb: 6720
// oom_score_adj: -1000
// }
//
// Use the ConnectedToPackage primitive to ensure only the target package has
// stats in the trace.
{
auto* primitive = redactor->emplace_transform<ScrubProcessStats>();
primitive->emplace_filter<ConnectedToPackage>();
}
// Redacts all switch and waking events. This should use the same modifier and
// filter as the process events (see below).
{
auto* primitive = redactor->emplace_transform<RedactSchedEvents>();
primitive->emplace_modifier<ClearComms>();
primitive->emplace_waking_filter<ConnectedToPackage>();
}
// Redacts all new task, rename task, process free events. This should use the
// same modifier and filter as the schedule events (see above).
{
auto* primitive = redactor->emplace_transform<RedactProcessEvents>();
primitive->emplace_modifier<ClearComms>();
primitive->emplace_filter<ConnectedToPackage>();
}
// Merge Threads (part 1): Remove all waking events that connected to the
// target package. Change the pids not connected to the target package.
{
auto* primitive = redactor->emplace_transform<RedactSchedEvents>();
primitive->emplace_modifier<MergeThreadsPids>();
primitive->emplace_waking_filter<ConnectedToPackage>();
}
// Merge Threads (part 2): Drop all process events not belonging to the
// target package. No modification is needed.
{
auto* primitive = redactor->emplace_transform<RedactProcessEvents>();
primitive->emplace_modifier<DoNothing>();
primitive->emplace_filter<ConnectedToPackage>();
}
// Merge Threads (part 3): Replace ftrace event's pid (not the task's pid)
// for all pids not connected to the target package.
{
auto* primitive = redactor->emplace_transform<RedactFtraceEvents>();
primitive->emplace_post_filter_modifier<MergeThreadsPids>();
primitive->emplace_ftrace_filter<AllowAll>();
}
// Configure the primitive to remove processes and threads that don't belong
// to the target package and adds a process and threads for the synth thread
// group and threads.
{
auto* primitive = redactor->emplace_transform<RedactProcessTrees>();
primitive->emplace_modifier<ProcessTreeCreateSynthThreads>();
primitive->emplace_filter<ConnectedToPackage>();
}
return redactor;
}
} // namespace perfetto::trace_redaction