blob: 62cc4d7cdce379da0b285746071fb1f3584041d4 [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_FILESYSTEM_INODE_FILE_DATA_SOURCE_H_
#define SRC_TRACED_PROBES_FILESYSTEM_INODE_FILE_DATA_SOURCE_H_
#include <sys/stat.h>
#include <sys/types.h>
#include <map>
#include <memory>
#include <set>
#include <string>
#include <unordered_map>
#include "perfetto/base/flat_set.h"
#include "perfetto/base/task_runner.h"
#include "perfetto/ext/base/weak_ptr.h"
#include "perfetto/ext/traced/data_source_types.h"
#include "perfetto/ext/tracing/core/basic_types.h"
#include "perfetto/ext/tracing/core/trace_writer.h"
#include "perfetto/tracing/core/data_source_config.h"
#include "src/traced/probes/filesystem/file_scanner.h"
#include "src/traced/probes/filesystem/fs_mount.h"
#include "src/traced/probes/filesystem/lru_inode_cache.h"
#include "src/traced/probes/probes_data_source.h"
#include "protos/perfetto/trace/filesystem/inode_file_map.pbzero.h"
namespace perfetto {
using InodeFileMap = protos::pbzero::InodeFileMap;
class TraceWriter;
// Creates block_device_map for /system partition
void CreateStaticDeviceToInodeMap(
const std::string& root_directory,
std::map<BlockDeviceID, std::unordered_map<Inode, InodeMapValue>>*
static_file_map);
class InodeFileDataSource : public ProbesDataSource,
public FileScanner::Delegate {
public:
static const ProbesDataSource::Descriptor descriptor;
InodeFileDataSource(
const DataSourceConfig&,
base::TaskRunner*,
TracingSessionID,
std::map<BlockDeviceID, std::unordered_map<Inode, InodeMapValue>>*
static_file_map,
LRUInodeCache* cache,
std::unique_ptr<TraceWriter> writer);
~InodeFileDataSource() override;
base::WeakPtr<InodeFileDataSource> GetWeakPtr() const;
// Called when Inodes are seen in the FtraceEventBundle
void OnInodes(const base::FlatSet<InodeBlockPair>& inodes);
// Search in /system partition and add inodes to InodeFileMap proto if found
void AddInodesFromStaticMap(BlockDeviceID block_device_id,
std::set<Inode>* inode_numbers);
// Search in LRUInodeCache and add inodes to InodeFileMap if found
void AddInodesFromLRUCache(BlockDeviceID block_device_id,
std::set<Inode>* inode_numbers);
virtual void FillInodeEntry(InodeFileMap* destination,
Inode inode_number,
const InodeMapValue& inode_map_value);
// ProbesDataSource implementation.
void Start() override;
void Flush(FlushRequestID, std::function<void()> callback) override;
protected:
std::multimap<BlockDeviceID, std::string> mount_points_;
private:
InodeFileMap* AddToCurrentTracePacket(BlockDeviceID block_device_id);
void ResetTracePacket();
void FindMissingInodes();
// Callbacks for dynamic filesystem scan.
bool OnInodeFound(BlockDeviceID block_device_id,
Inode inode_number,
const std::string& path,
InodeFileMap_Entry_Type type) override;
void OnInodeScanDone() override;
void AddRootsForBlockDevice(BlockDeviceID block_device_id,
std::vector<std::string>* roots);
void RemoveFromNextMissingInodes(BlockDeviceID block_device_id,
Inode inode_number);
std::set<std::string> scan_mount_points_;
std::map<std::string, std::vector<std::string>> mount_point_mapping_;
base::TaskRunner* task_runner_;
std::map<BlockDeviceID, std::unordered_map<Inode, InodeMapValue>>*
static_file_map_;
LRUInodeCache* cache_;
std::unique_ptr<TraceWriter> writer_;
std::map<BlockDeviceID, std::set<Inode>> missing_inodes_;
std::map<BlockDeviceID, std::set<Inode>> next_missing_inodes_;
std::set<BlockDeviceID> seen_block_devices_;
BlockDeviceID current_block_device_id_;
TraceWriter::TracePacketHandle current_trace_packet_;
InodeFileMap* current_file_map_;
bool has_current_trace_packet_ = false;
bool scan_running_ = false;
bool do_not_scan_ = false;
uint32_t scan_interval_ms_ = 0;
uint32_t scan_delay_ms_ = 0;
uint32_t scan_batch_size_ = 0;
std::unique_ptr<FileScanner> file_scanner_;
base::WeakPtrFactory<InodeFileDataSource> weak_factory_; // Keep last.
};
} // namespace perfetto
#endif // SRC_TRACED_PROBES_FILESYSTEM_INODE_FILE_DATA_SOURCE_H_