blob: b2c579138ff854544c3bc0f002f2eeae23f1f74a [file] [log] [blame]
/*
* Copyright (C) 2018 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.
*/
#ifndef SRC_TRACE_PROCESSOR_SCHED_TRACKER_H_
#define SRC_TRACE_PROCESSOR_SCHED_TRACKER_H_
#include <array>
#include "perfetto/base/string_view.h"
#include "perfetto/base/utils.h"
#include "src/trace_processor/trace_storage.h"
namespace perfetto {
namespace trace_processor {
class TraceProcessorContext;
// This class takes sched events from the trace and processes them to store
// as sched slices.
class SchedTracker {
public:
explicit SchedTracker(TraceProcessorContext*);
SchedTracker(const SchedTracker&) = delete;
SchedTracker& operator=(const SchedTracker&) = delete;
virtual ~SchedTracker();
struct SchedSwitchEvent {
uint64_t timestamp = 0;
uint32_t prev_pid = 0;
uint32_t prev_state = 0;
uint32_t next_pid = 0;
bool valid() const { return timestamp != 0; }
};
// A Counter is a trace event that has a value attached to a timestamp.
// These include CPU frequency ftrace events and systrace trace_marker
// counter events.
struct Counter {
uint64_t timestamp = 0;
double value = 0;
};
// Used as the key in |prev_counters_| to find the previous counter with the
// same ref and name_id.
struct CounterKey {
uint64_t ref; // cpu, utid, ...
StringId name_id; // "cpufreq"
bool operator==(const CounterKey& other) const {
return (ref == other.ref && name_id == other.name_id);
}
struct Hasher {
size_t operator()(const CounterKey& c) const {
size_t const h1(std::hash<uint64_t>{}(c.ref));
size_t const h2(std::hash<size_t>{}(c.name_id));
return h1 ^ (h2 << 1);
}
};
};
// This method is called when a sched switch event is seen in the trace.
virtual void PushSchedSwitch(uint32_t cpu,
uint64_t timestamp,
uint32_t prev_pid,
uint32_t prev_state,
base::StringView prev_comm,
uint32_t next_pid);
// This method is called when a cpu freq event is seen in the trace.
// TODO(taylori): Move to a more appropriate class or rename class.
virtual void PushCounter(uint64_t timestamp,
double value,
StringId name_id,
uint64_t ref,
RefType ref_type);
private:
// Store the previous sched event to calculate the duration before storing it.
std::array<SchedSwitchEvent, base::kMaxCpus> last_sched_per_cpu_;
// Store the previous counter event to calculate the duration and value delta
// before storing it in trace storage.
std::unordered_map<CounterKey, Counter, CounterKey::Hasher> prev_counters_;
// Timestamp of the previous event. Used to discard events arriving out
// of order.
uint64_t prev_timestamp_ = 0;
StringId const idle_string_id_;
TraceProcessorContext* const context_;
};
} // namespace trace_processor
} // namespace perfetto
#endif // SRC_TRACE_PROCESSOR_SCHED_TRACKER_H_