| /* |
| * Copyright (C) 2017 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_TRACED_PROBES_FTRACE_FTRACE_CONTROLLER_H_ |
| #define SRC_TRACED_PROBES_FTRACE_FTRACE_CONTROLLER_H_ |
| |
| #include <stdint.h> |
| #include <unistd.h> |
| |
| #include <map> |
| #include <memory> |
| #include <set> |
| #include <string> |
| |
| #include "perfetto/base/task_runner.h" |
| #include "perfetto/ext/base/weak_ptr.h" |
| #include "perfetto/ext/tracing/core/basic_types.h" |
| #include "src/kallsyms/lazy_kernel_symbolizer.h" |
| #include "src/traced/probes/ftrace/cpu_reader.h" |
| #include "src/traced/probes/ftrace/ftrace_config_utils.h" |
| |
| namespace perfetto { |
| |
| class FtraceConfigMuxer; |
| class FtraceDataSource; |
| class FtraceProcfs; |
| class LazyKernelSymbolizer; |
| class ProtoTranslationTable; |
| struct FtraceStats; |
| |
| // Method of last resort to reset ftrace state. |
| bool HardResetFtraceState(); |
| |
| // Stores the a snapshot of the timestamps from ftrace's trace clock |
| // and CLOCK_BOOTITME. |
| // |
| // This is used when the "boot" (i.e. CLOCK_BOOTITME) is not available |
| // for timestamping trace events (on Android O- and 3.x Linux kernels). |
| // Trace processor can use this data to sync clocks just as it would |
| // with ClockSnapshot packets. |
| struct FtraceClockSnapshot { |
| // The timestamp according to the ftrace clock. |
| int64_t ftrace_clock_ts = 0; |
| |
| // The timestamp according to CLOCK_BOOTTIME. |
| int64_t boot_clock_ts = 0; |
| }; |
| |
| // Utility class for controlling ftrace. |
| class FtraceController { |
| public: |
| class Observer { |
| public: |
| virtual ~Observer(); |
| virtual void OnFtraceDataWrittenIntoDataSourceBuffers() = 0; |
| }; |
| |
| // The passed Observer must outlive the returned FtraceController instance. |
| static std::unique_ptr<FtraceController> Create(base::TaskRunner*, Observer*); |
| virtual ~FtraceController(); |
| |
| bool AddDataSource(FtraceDataSource*) PERFETTO_WARN_UNUSED_RESULT; |
| bool StartDataSource(FtraceDataSource*); |
| void RemoveDataSource(FtraceDataSource*); |
| |
| // Force a read of the ftrace buffers. Will call OnFtraceFlushComplete() on |
| // all started data sources. |
| void Flush(FlushRequestID); |
| |
| void DumpFtraceStats(FtraceDataSource*, FtraceStats*); |
| |
| base::WeakPtr<FtraceController> GetWeakPtr() { |
| return weak_factory_.GetWeakPtr(); |
| } |
| |
| // public for testing |
| static std::optional<std::string> AbsolutePathForInstance( |
| const std::string& tracefs_root, |
| const std::string& raw_cfg_name); |
| |
| protected: |
| // Everything protected/virtual for testing: |
| |
| FtraceController(std::unique_ptr<FtraceProcfs>, |
| std::unique_ptr<ProtoTranslationTable>, |
| std::unique_ptr<FtraceConfigMuxer>, |
| base::TaskRunner*, |
| Observer*); |
| |
| struct FtraceInstanceState { |
| FtraceInstanceState(std::unique_ptr<FtraceProcfs>, |
| std::unique_ptr<ProtoTranslationTable>, |
| std::unique_ptr<FtraceConfigMuxer>); |
| |
| std::unique_ptr<FtraceProcfs> ftrace_procfs; |
| std::unique_ptr<ProtoTranslationTable> table; |
| std::unique_ptr<FtraceConfigMuxer> ftrace_config_muxer; |
| std::vector<CpuReader> cpu_readers; // empty if no started data sources |
| std::set<FtraceDataSource*> started_data_sources; |
| bool buffer_watches_posted = false; |
| }; |
| |
| FtraceInstanceState* GetInstance(const std::string& instance_name); |
| // TODO(rsavitski): figure out a better testing shim. |
| virtual std::unique_ptr<FtraceInstanceState> CreateSecondaryInstance( |
| const std::string& instance_name); |
| |
| virtual uint64_t NowMs() const; |
| |
| private: |
| friend class TestFtraceController; |
| enum class PollSupport { kUntested, kSupported, kUnsupported }; |
| |
| FtraceController(const FtraceController&) = delete; |
| FtraceController& operator=(const FtraceController&) = delete; |
| |
| // Periodic task that reads all per-cpu ftrace buffers. Global across tracefs |
| // instances. |
| void ReadTick(int generation); |
| bool ReadPassForInstance(FtraceInstanceState* instance); |
| uint32_t GetTickPeriodMs(); |
| // Optional: additional reads based on buffer capacity. Per tracefs instance. |
| void UpdateBufferWatermarkWatches(FtraceInstanceState* instance, |
| const std::string& instance_name); |
| void OnBufferPastWatermark(std::string instance_name, |
| size_t cpu, |
| bool repoll_watermark); |
| void RemoveBufferWatermarkWatches(FtraceInstanceState* instance); |
| PollSupport VerifyKernelSupportForBufferWatermark(); |
| |
| void FlushForInstance(FtraceInstanceState* instance); |
| |
| void StartIfNeeded(FtraceInstanceState* instance, |
| const std::string& instance_name); |
| void StopIfNeeded(FtraceInstanceState* instance); |
| |
| FtraceInstanceState* GetOrCreateInstance(const std::string& instance_name); |
| void DestroyIfUnusedSeconaryInstance(FtraceInstanceState* instance); |
| |
| size_t GetStartedDataSourcesCount(); |
| void MaybeSnapshotFtraceClock(); // valid only for primary_ tracefs instance |
| |
| template <typename F /* void(FtraceInstanceState*) */> |
| void ForEachInstance(F fn); |
| |
| base::TaskRunner* const task_runner_; |
| Observer* const observer_; |
| CpuReader::ParsingBuffers parsing_mem_; |
| LazyKernelSymbolizer symbolizer_; |
| FtraceConfigId next_cfg_id_ = 1; |
| int tick_generation_ = 0; |
| bool retain_ksyms_on_stop_ = false; |
| PollSupport buffer_watermark_support_ = PollSupport::kUntested; |
| std::set<FtraceDataSource*> data_sources_; |
| // Default tracefs instance (normally /sys/kernel/tracing) is valid for as |
| // long as the controller is valid. |
| // Secondary instances (i.e. /sys/kernel/tracing/instances/...) are created |
| // and destroyed as necessary between AddDataSource and RemoveDataSource: |
| FtraceInstanceState primary_; |
| std::map<std::string, std::unique_ptr<FtraceInstanceState>> |
| secondary_instances_; |
| |
| // Additional state for snapshotting non-boot ftrace clock, specific to the |
| // primary_ instance: |
| base::ScopedFile cpu_zero_stats_fd_; |
| FtraceClockSnapshot ftrace_clock_snapshot_; |
| |
| base::WeakPtrFactory<FtraceController> weak_factory_; // Keep last. |
| }; |
| |
| } // namespace perfetto |
| |
| #endif // SRC_TRACED_PROBES_FTRACE_FTRACE_CONTROLLER_H_ |