blob: 28405d10d41e1ebb3a8fb3a8e84384ae897cbb26 [file] [log] [blame]
/*
* Copyright (C) 2019 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/importers/proto/android_probes_module.h"
#include "perfetto/base/build_config.h"
#include "perfetto/protozero/scattered_heap_buffer.h"
#include "src/trace_processor/importers/proto/android_probes_parser.h"
#include "src/trace_processor/timestamped_trace_piece.h"
#include "src/trace_processor/trace_sorter.h"
#include "protos/perfetto/config/trace_config.pbzero.h"
#include "protos/perfetto/trace/power/power_rails.pbzero.h"
#include "protos/perfetto/trace/trace_packet.pbzero.h"
namespace perfetto {
namespace trace_processor {
using perfetto::protos::pbzero::TracePacket;
AndroidProbesModule::AndroidProbesModule(TraceProcessorContext* context)
: parser_(context), context_(context) {
RegisterForField(TracePacket::kBatteryFieldNumber, context);
RegisterForField(TracePacket::kPowerRailsFieldNumber, context);
RegisterForField(TracePacket::kAndroidLogFieldNumber, context);
RegisterForField(TracePacket::kPackagesListFieldNumber, context);
RegisterForField(TracePacket::kInitialDisplayStateFieldNumber, context);
}
ModuleResult AndroidProbesModule::TokenizePacket(
const protos::pbzero::TracePacket_Decoder&,
TraceBlobView* packet,
int64_t packet_timestamp,
PacketSequenceState* state,
uint32_t field_id) {
if (field_id != TracePacket::kPowerRailsFieldNumber)
return ModuleResult::Ignored();
// Power rails are similar to ftrace in that they have many events, each with
// their own timestamp, packed inside a single TracePacket. This means that,
// similar to ftrace, we need to unpack them and individually sort them.
// However, as these events are not perf sensitive, it's not worth adding
// a lot of machinery to shepherd these events through the sorting queues
// in a special way. Therefore, we just forge new packets and sort them as if
// they came from the underlying trace.
protos::pbzero::TracePacket::Decoder decoder(packet->data(),
packet->length());
auto power_rails = decoder.power_rails();
protos::pbzero::PowerRails::Decoder evt(power_rails.data, power_rails.size);
// Break out the rail descriptor into its own packet with the same timestamp
// as the packet itself.
if (evt.has_rail_descriptor()) {
protozero::HeapBuffered<protos::pbzero::TracePacket> desc_packet;
desc_packet->set_timestamp(static_cast<uint64_t>(packet_timestamp));
auto* rails = desc_packet->set_power_rails();
for (auto it = evt.rail_descriptor(); it; ++it) {
protozero::ConstBytes desc = *it;
rails->add_rail_descriptor()->AppendRawProtoBytes(desc.data, desc.size);
}
std::vector<uint8_t> vec = desc_packet.SerializeAsArray();
std::unique_ptr<uint8_t[]> buffer(new uint8_t[vec.size()]);
memcpy(buffer.get(), vec.data(), vec.size());
context_->sorter->PushTracePacket(
packet_timestamp, state,
TraceBlobView(std::move(buffer), 0, vec.size()));
}
// For each energy data message, turn it into its own trace packet
// making sure its timestamp is consistent between the packet level and
// the EnergyData level.
for (auto it = evt.energy_data(); it; ++it) {
protozero::ConstBytes bytes = *it;
protos::pbzero::PowerRails_EnergyData_Decoder data(bytes.data, bytes.size);
int64_t actual_ts =
data.has_timestamp_ms()
? static_cast<int64_t>(data.timestamp_ms()) * 1000000
: packet_timestamp;
protozero::HeapBuffered<protos::pbzero::TracePacket> data_packet;
data_packet->set_timestamp(static_cast<uint64_t>(actual_ts));
auto* energy = data_packet->set_power_rails()->add_energy_data();
energy->set_energy(data.energy());
energy->set_index(data.index());
energy->set_timestamp_ms(static_cast<uint64_t>(actual_ts / 1000000));
std::vector<uint8_t> vec = data_packet.SerializeAsArray();
std::unique_ptr<uint8_t[]> buffer(new uint8_t[vec.size()]);
memcpy(buffer.get(), vec.data(), vec.size());
context_->sorter->PushTracePacket(
actual_ts, state, TraceBlobView(std::move(buffer), 0, vec.size()));
}
return ModuleResult::Handled();
}
void AndroidProbesModule::ParsePacket(const TracePacket::Decoder& decoder,
const TimestampedTracePiece& ttp,
uint32_t field_id) {
switch (field_id) {
case TracePacket::kBatteryFieldNumber:
parser_.ParseBatteryCounters(ttp.timestamp, decoder.battery());
return;
case TracePacket::kPowerRailsFieldNumber:
parser_.ParsePowerRails(ttp.timestamp, decoder.power_rails());
return;
case TracePacket::kAndroidLogFieldNumber:
parser_.ParseAndroidLogPacket(decoder.android_log());
return;
case TracePacket::kPackagesListFieldNumber:
parser_.ParseAndroidPackagesList(decoder.packages_list());
return;
case TracePacket::kInitialDisplayStateFieldNumber:
parser_.ParseInitialDisplayState(ttp.timestamp,
decoder.initial_display_state());
return;
}
}
void AndroidProbesModule::ParseTraceConfig(
const protos::pbzero::TraceConfig::Decoder& decoder) {
if (decoder.has_statsd_metadata()) {
parser_.ParseStatsdMetadata(decoder.statsd_metadata());
}
}
} // namespace trace_processor
} // namespace perfetto