/*
* 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.
*/
#include "src/traced/probes/filesystem/file_scanner.h"
#include <dirent.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include "src/traced/probes/filesystem/inode_file_data_source.h"
namespace perfetto {
namespace {
std::string JoinPaths(const std::string& one, const std::string& other) {
std::string result;
result.reserve(one.size() + other.size() + 1);
result += one;
if (!result.empty() && result.back() != '/')
result += '/';
result += other;
return result;
}
} // namespace
FileScanner::FileScanner(std::vector<std::string> root_directories,
Delegate* delegate,
uint32_t scan_interval_ms,
uint32_t scan_steps)
: delegate_(delegate),
scan_interval_ms_(scan_interval_ms),
scan_steps_(scan_steps),
queue_(std::move(root_directories)),
weak_factory_(this) {}
FileScanner::FileScanner(std::vector<std::string> root_directories,
Delegate* delegate)
: FileScanner(std::move(root_directories),
delegate,
0 /* scan_interval_ms */,
0 /* scan_steps */) {}
void FileScanner::Scan() {
while (!Done())
Step();
delegate_->OnInodeScanDone();
}
void FileScanner::Scan(base::TaskRunner* task_runner) {
PERFETTO_DCHECK(scan_interval_ms_ && scan_steps_);
Steps(scan_steps_);
if (Done())
return delegate_->OnInodeScanDone();
auto weak_this = weak_factory_.GetWeakPtr();
task_runner->PostDelayedTask(
[weak_this, task_runner] {
if (!weak_this)
return;
weak_this->Scan(task_runner);
},
scan_interval_ms_);
}
void FileScanner::NextDirectory() {
std::string directory = std::move(queue_.back());
queue_.pop_back();
current_dir_handle_.reset(opendir(directory.c_str()));
if (!current_dir_handle_) {
PERFETTO_DPLOG("opendir %s", directory.c_str());
current_directory_.clear();
return;
}
current_directory_ = std::move(directory);
struct stat buf;
if (fstat(dirfd(current_dir_handle_.get()), &buf) != 0) {
PERFETTO_DPLOG("fstat %s", current_directory_.c_str());
current_dir_handle_.reset();
current_directory_.clear();
return;
}
if (S_ISLNK(buf.st_mode)) {
current_dir_handle_.reset();
current_directory_.clear();
return;
}
current_block_device_id_ = buf.st_dev;
}
void FileScanner::Step() {
if (!current_dir_handle_) {
if (queue_.empty())
return;
NextDirectory();
}
if (!current_dir_handle_)
return;
struct dirent* entry = readdir(current_dir_handle_.get());
if (entry == nullptr) {
current_dir_handle_.reset();
return;
}
std::string filename = entry->d_name;
if (filename == "." || filename == "..")
return;
std::string filepath = JoinPaths(current_directory_, filename);
protos::pbzero::InodeFileMap_Entry_Type type =
protos::pbzero::InodeFileMap_Entry_Type_UNKNOWN;
// Readdir and stat not guaranteed to have directory info for all systems
if (entry->d_type == DT_DIR) {
// Continue iterating through files if current entry is a directory
queue_.emplace_back(filepath);
type = protos::pbzero::InodeFileMap_Entry_Type_DIRECTORY;
} else if (entry->d_type == DT_REG) {
type = protos::pbzero::InodeFileMap_Entry_Type_FILE;
}
if (!delegate_->OnInodeFound(current_block_device_id_, entry->d_ino, filepath,
type)) {
queue_.clear();
current_dir_handle_.reset();
}
}
void FileScanner::Steps(uint32_t n) {
for (uint32_t i = 0; i < n && !Done(); ++i)
Step();
}
bool FileScanner::Done() {
return !current_dir_handle_ && queue_.empty();
}
FileScanner::Delegate::~Delegate() = default;
} // namespace perfetto