10 #include <botan/internal/proc_walk.h>
11 #include <botan/secmem.h>
14 #ifndef _POSIX_C_SOURCE
15 #define _POSIX_C_SOURCE 199309
18 #include <sys/types.h>
28 class Directory_Walker :
public File_Descriptor_Source
31 explicit Directory_Walker(
const std::string& root) :
32 m_cur_dir(std::make_pair<DIR*, std::string>(
nullptr,
""))
34 if(DIR* root_dir = ::opendir(root.c_str()))
35 m_cur_dir = std::make_pair(root_dir, root);
44 int next_fd()
override;
46 std::pair<struct dirent*, std::string> get_next_dirent();
52 std::pair<struct dirent*, std::string> Directory_Walker::get_next_dirent()
56 if(
struct dirent* dir = ::readdir(
m_cur_dir.first))
57 return std::make_pair(dir,
m_cur_dir.second);
60 m_cur_dir = std::make_pair<DIR*, std::string>(
nullptr,
"");
64 const std::string next_dir_name =
m_dirlist[0];
67 if(DIR* next_dir = ::opendir(next_dir_name.c_str()))
68 m_cur_dir = std::make_pair(next_dir, next_dir_name);
72 return std::make_pair<struct dirent*, std::string>(
nullptr,
"");
75 int Directory_Walker::next_fd()
79 std::pair<struct dirent*, std::string> entry = get_next_dirent();
84 const std::string filename = entry.first->d_name;
86 if(filename ==
"." || filename ==
"..")
89 const std::string full_path = entry.second +
"/" + filename;
92 if(::lstat(full_path.c_str(), &stat_buf) == -1)
95 if(S_ISDIR(stat_buf.st_mode))
99 else if(S_ISREG(stat_buf.st_mode) && (stat_buf.st_mode & S_IROTH))
101 int fd = ::open(full_path.c_str(), O_RDONLY |
O_NOCTTY);
115 const size_t MAX_FILES_READ_PER_POLL = 2048;
117 lock_guard_type<mutex_type> lock(m_mutex);
120 m_dir.reset(
new Directory_Walker(m_path));
126 for(
size_t i = 0; i != MAX_FILES_READ_PER_POLL; ++i)
128 int fd = m_dir->next_fd();
137 ssize_t got = ::read(fd, m_buf.data(), m_buf.size());
142 rng.
add_entropy(m_buf.data(),
static_cast<size_t>(got));
virtual void add_entropy(const uint8_t input[], size_t length)=0
std::deque< std::string > m_dirlist
std::pair< DIR *, std::string > m_cur_dir
size_t poll(RandomNumberGenerator &rng) override