| /* |
| * 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 "perfetto/base/logging.h" |
| #include "perfetto/base/status.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/redact_sched_events.h" |
| #include "src/trace_redaction/scrub_process_stats.h" |
| #include "src/trace_redaction/trace_redaction_framework.h" |
| #include "src/trace_redaction/trace_redactor.h" |
| #include "src/trace_redaction/verify_integrity.h" |
| |
| namespace perfetto::trace_redaction { |
| |
| // Builds and runs a trace redactor. |
| static base::Status Main(std::string_view input, |
| std::string_view output, |
| std::string_view package_name) { |
| TraceRedactor redactor; |
| |
| // 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. |
| 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>(); |
| } |
| |
| Context context; |
| context.package_name = package_name; |
| |
| return redactor.Redact(input, output, &context); |
| } |
| |
| } // namespace perfetto::trace_redaction |
| |
| int main(int argc, char** argv) { |
| constexpr int kSuccess = 0; |
| constexpr int kFailure = 1; |
| constexpr int kInvalidArgs = 2; |
| |
| if (argc != 4) { |
| PERFETTO_ELOG( |
| "Invalid arguments: %s <input file> <output file> <package name>", |
| argv[0]); |
| return kInvalidArgs; |
| } |
| |
| auto result = perfetto::trace_redaction::Main(argv[1], argv[2], argv[3]); |
| |
| if (result.ok()) { |
| return kSuccess; |
| } |
| |
| PERFETTO_ELOG("Unexpected error: %s", result.c_message()); |
| return kFailure; |
| } |