blob: 6c5ab9140e17bdb121d29d42b187174fdedbf02d [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/reduce_threads_in_process_trees.h"
#include <string>
#include "perfetto/base/status.h"
#include "perfetto/protozero/field.h"
#include "perfetto/protozero/scattered_heap_buffer.h"
#include "src/trace_redaction/proto_util.h"
#include "src/trace_redaction/trace_redaction_framework.h"
#include "protos/perfetto/trace/ps/process_tree.pbzero.h"
#include "protos/perfetto/trace/trace_packet.pbzero.h"
namespace perfetto::trace_redaction {
namespace {
bool ShouldCopyProcessToProcessTree(const ProcessThreadTimeline& timeline,
uint64_t package,
uint64_t ts,
protozero::ConstBytes src) {
protos::pbzero::ProcessTree::Process::Decoder decoder(src);
return decoder.has_pid() &&
timeline.PidConnectsToUid(ts, decoder.pid(), package);
}
bool ShoudCopyThreadToProcessTree(const ProcessThreadTimeline& timeline,
uint64_t package,
uint64_t ts,
protozero::ConstBytes src) {
protos::pbzero::ProcessTree::Thread::Decoder decoder(src);
return decoder.has_tid() &&
timeline.PidConnectsToUid(ts, decoder.tid(), package);
}
void CopyProcessTree(const ProcessThreadTimeline& timeline,
uint64_t package,
uint64_t ts,
protozero::ConstBytes src,
protos::pbzero::ProcessTree* dest) {
protozero::ProtoDecoder decoder(src);
for (auto it = decoder.ReadField(); it; it = decoder.ReadField()) {
switch (it.id()) {
case protos::pbzero::ProcessTree::kProcessesFieldNumber:
if (ShouldCopyProcessToProcessTree(timeline, package, ts,
it.as_bytes())) {
proto_util::AppendField(it, dest);
}
break;
case protos::pbzero::ProcessTree::kThreadsFieldNumber:
if (ShoudCopyThreadToProcessTree(timeline, package, ts,
it.as_bytes())) {
proto_util::AppendField(it, dest);
}
break;
default:
proto_util::AppendField(it, dest);
break;
}
}
}
} // namespace
base::Status ReduceThreadsInProcessTrees::Transform(const Context& context,
std::string* packet) const {
PERFETTO_DCHECK(packet);
if (!context.package_uid.has_value()) {
return base::ErrStatus("RedactProcessTrees: missing package uid.");
}
if (!context.timeline) {
return base::ErrStatus("RedactProcessTrees: missing timeline.");
}
protozero::ProtoDecoder decoder(*packet);
// This has been verified by the verify primitive.
auto timestamp =
decoder.FindField(protos::pbzero::TracePacket::kTimestampFieldNumber);
protozero::HeapBuffered<protos::pbzero::TracePacket> message;
for (auto it = decoder.ReadField(); it.valid(); it = decoder.ReadField()) {
if (it.id() == protos::pbzero::TracePacket::kProcessTreeFieldNumber) {
CopyProcessTree(*context.timeline, *context.package_uid,
timestamp.as_uint64(), it.as_bytes(),
message->set_process_tree());
} else {
proto_util::AppendField(it, message.get());
}
}
packet->assign(message.SerializeAsString());
return base::OkStatus();
}
} // namespace perfetto::trace_redaction