blob: d7492b742beec34a06771a7406b9e9fcce334e6f [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/scrub_process_stats.h"
#include <string>
#include "perfetto/base/status.h"
#include "perfetto/protozero/field.h"
#include "perfetto/protozero/scattered_heap_buffer.h"
#include "src/trace_processor/util/status_macros.h"
#include "src/trace_redaction/proto_util.h"
#include "src/trace_redaction/trace_redaction_framework.h"
#include "protos/perfetto/trace/ps/process_stats.pbzero.h"
namespace perfetto::trace_redaction {
base::Status ScrubProcessStats::Transform(const Context& context,
std::string* packet) const {
if (!context.package_uid.has_value()) {
return base::ErrStatus("FilterProcessStats: missing package uid.");
}
if (!context.timeline) {
return base::ErrStatus("FilterProcessStats: missing timeline.");
}
protozero::ProtoDecoder packet_decoder(*packet);
// Very few packets will have process stats. It's best to avoid
// reserialization whenever possible.
if (!packet_decoder
.FindField(protos::pbzero::TracePacket::kProcessStatsFieldNumber)
.valid()) {
return base::OkStatus();
}
protozero::HeapBuffered<protos::pbzero::TracePacket> message;
// Not all packets will have a top-level timestamp, but for process stats, the
// timestamp is located at the trace packet.
auto time_field = packet_decoder.FindField(
protos::pbzero::TracePacket::kTimestampFieldNumber);
PERFETTO_DCHECK(time_field.valid());
auto ts = time_field.as_uint64();
for (auto field = packet_decoder.ReadField(); field.valid();
field = packet_decoder.ReadField()) {
if (field.id() == protos::pbzero::TracePacket::kProcessStatsFieldNumber) {
RETURN_IF_ERROR(OnProcessStats(context, ts, field.as_bytes(),
message->set_process_stats()));
} else {
proto_util::AppendField(field, message.get());
}
}
packet->assign(message.SerializeAsString());
return base::OkStatus();
}
base::Status ScrubProcessStats::OnProcessStats(
const Context& context,
uint64_t ts,
protozero::ConstBytes bytes,
protos::pbzero::ProcessStats* message) const {
protozero::ProtoDecoder decoder(bytes);
for (auto field = decoder.ReadField(); field.valid();
field = decoder.ReadField()) {
if (field.id() == protos::pbzero::ProcessStats::kProcessesFieldNumber) {
RETURN_IF_ERROR(OnProcess(context, ts, field, message));
} else {
proto_util::AppendField(field, message);
}
}
return base::OkStatus();
}
base::Status ScrubProcessStats::OnProcess(
const Context& context,
uint64_t ts,
protozero::Field field,
protos::pbzero::ProcessStats* message) const {
PERFETTO_DCHECK(field.id() ==
protos::pbzero::ProcessStats::kProcessesFieldNumber);
protozero::ProtoDecoder decoder(field.as_bytes());
auto pid =
decoder.FindField(protos::pbzero::ProcessStats::Process::kPidFieldNumber);
PERFETTO_DCHECK(pid.valid());
PERFETTO_DCHECK(filter_);
if (filter_->Includes(context, ts, pid.as_int32())) {
proto_util::AppendField(field, message);
}
return base::OkStatus();
}
} // namespace perfetto::trace_redaction