blob: 1d23f7723c6a935b552b67419cc0ec08b965e957 [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_TRACED_PROBES_PROBES_PRODUCER_H_
#define SRC_TRACED_PROBES_PROBES_PRODUCER_H_
#include <map>
#include <memory>
#include <utility>
#include "perfetto/base/task_runner.h"
#include "perfetto/base/watchdog.h"
#include "perfetto/tracing/core/producer.h"
#include "perfetto/tracing/core/service.h"
#include "perfetto/tracing/core/trace_writer.h"
#include "src/traced/probes/filesystem/inode_file_data_source.h"
#include "src/traced/probes/ftrace/ftrace_controller.h"
#include "src/traced/probes/process_stats_data_source.h"
#include "perfetto/trace/filesystem/inode_file_map.pbzero.h"
namespace perfetto {
const uint64_t kLRUInodeCacheSize = 1000;
class ProbesProducer : public Producer {
public:
ProbesProducer();
~ProbesProducer() override;
// Producer Impl:
void OnConnect() override;
void OnDisconnect() override;
void CreateDataSourceInstance(DataSourceInstanceID,
const DataSourceConfig&) override;
void TearDownDataSourceInstance(DataSourceInstanceID) override;
void OnTracingSetup() override;
void Flush(FlushRequestID,
const DataSourceInstanceID* data_source_ids,
size_t num_data_sources) override;
// Our Impl
void ConnectWithRetries(const char* socket_name,
base::TaskRunner* task_runner);
bool CreateFtraceDataSourceInstance(TracingSessionID session_id,
DataSourceInstanceID id,
const DataSourceConfig& config);
void CreateProcessStatsDataSourceInstance(TracingSessionID session_id,
DataSourceInstanceID id,
const DataSourceConfig& config);
void CreateInodeFileDataSourceInstance(TracingSessionID session_id,
DataSourceInstanceID id,
DataSourceConfig config);
void OnMetadata(const FtraceMetadata& metadata);
private:
using FtraceBundleHandle =
protozero::MessageHandle<protos::pbzero::FtraceEventBundle>;
using FtraceStatsHandle =
protozero::MessageHandle<protos::pbzero::FtraceStats>;
class SinkDelegate : public FtraceSink::Delegate {
public:
SinkDelegate(TracingSessionID,
base::TaskRunner*,
std::unique_ptr<TraceWriter>);
~SinkDelegate() override;
TracingSessionID session_id() const { return session_id_; }
void Flush();
// FtraceDelegateImpl
FtraceBundleHandle GetBundleForCpu(size_t cpu) override;
void OnBundleComplete(size_t cpu,
FtraceBundleHandle bundle,
const FtraceMetadata& metadata) override;
void OnCreate(FtraceSink*) override;
void WriteStats();
void set_sink(std::unique_ptr<FtraceSink> sink) { sink_ = std::move(sink); }
void set_ps_source(base::WeakPtr<ProcessStatsDataSource> ptr) {
ps_source_ = std::move(ptr);
}
const base::WeakPtr<ProcessStatsDataSource>& ps_source() const {
return ps_source_;
}
void set_file_source(base::WeakPtr<InodeFileDataSource> ptr) {
file_source_ = std::move(ptr);
}
const base::WeakPtr<InodeFileDataSource>& file_source() const {
return file_source_;
}
private:
const TracingSessionID session_id_;
base::TaskRunner* task_runner_;
std::unique_ptr<FtraceSink> sink_ = nullptr;
std::unique_ptr<TraceWriter> writer_;
FtraceStats stats_before_ = {};
base::WeakPtr<ProcessStatsDataSource> ps_source_;
base::WeakPtr<InodeFileDataSource> file_source_;
// Keep this after the TraceWriter because TracePackets must not outlive
// their originating writer.
TraceWriter::TracePacketHandle trace_packet_;
// Keep this last.
base::WeakPtrFactory<SinkDelegate> weak_factory_;
};
enum State {
kNotStarted = 0,
kNotConnected,
kConnecting,
kConnected,
};
ProbesProducer(const ProbesProducer&) = delete;
ProbesProducer& operator=(const ProbesProducer&) = delete;
void Connect();
void Restart();
void ResetConnectionBackoff();
void IncreaseConnectionBackoff();
void AddWatchdogsTimer(DataSourceInstanceID id,
const DataSourceConfig& source_config);
State state_ = kNotStarted;
base::TaskRunner* task_runner_ = nullptr;
std::unique_ptr<Service::ProducerEndpoint> endpoint_ = nullptr;
std::unique_ptr<FtraceController> ftrace_ = nullptr;
bool ftrace_creation_failed_ = false;
uint32_t connection_backoff_ms_ = 0;
const char* socket_name_ = nullptr;
std::set<DataSourceInstanceID> failed_sources_;
std::map<DataSourceInstanceID, std::unique_ptr<ProcessStatsDataSource>>
process_stats_sources_;
std::map<DataSourceInstanceID, std::unique_ptr<SinkDelegate>> delegates_;
std::map<DataSourceInstanceID, base::Watchdog::Timer> watchdogs_;
std::map<DataSourceInstanceID, std::unique_ptr<InodeFileDataSource>>
file_map_sources_;
LRUInodeCache cache_{kLRUInodeCacheSize};
std::map<BlockDeviceID, std::unordered_map<Inode, InodeMapValue>>
system_inodes_;
};
} // namespace perfetto
#endif // SRC_TRACED_PROBES_PROBES_PRODUCER_H_