mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-23 09:51:57 -05:00
65b420f996
These instances were detected by searching for files that include AK/Memory.h, but don't match the regex: \\b(fast_u32_copy|fast_u32_fill|secure_zero|timing_safe_compare)\\b This regex is pessimistic, so there might be more files that don't actually use any memory function. In theory, one might use LibCPP to detect things like this automatically, but let's do this one step after another.
173 lines
4.9 KiB
C++
173 lines
4.9 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2021, sin-ack <sin-ack@protonmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <Kernel/FileSystem/Inode.h>
|
|
#include <Kernel/FileSystem/InodeWatcher.h>
|
|
#include <Kernel/Process.h>
|
|
|
|
namespace Kernel {
|
|
|
|
ErrorOr<NonnullLockRefPtr<InodeWatcher>> InodeWatcher::try_create()
|
|
{
|
|
return adopt_nonnull_lock_ref_or_enomem(new (nothrow) InodeWatcher);
|
|
}
|
|
|
|
InodeWatcher::~InodeWatcher()
|
|
{
|
|
(void)close();
|
|
}
|
|
|
|
bool InodeWatcher::can_read(OpenFileDescription const&, u64) const
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
return !m_queue.is_empty();
|
|
}
|
|
|
|
ErrorOr<size_t> InodeWatcher::read(OpenFileDescription&, u64, UserOrKernelBuffer& buffer, size_t buffer_size)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
if (m_queue.is_empty())
|
|
// can_read will catch the blocking case.
|
|
return EAGAIN;
|
|
|
|
auto event = m_queue.dequeue();
|
|
|
|
size_t bytes_to_write = sizeof(InodeWatcherEvent);
|
|
if (event.path)
|
|
bytes_to_write += event.path->length() + 1;
|
|
|
|
if (buffer_size < bytes_to_write)
|
|
return EINVAL;
|
|
|
|
auto result = buffer.write_buffered<MAXIMUM_EVENT_SIZE>(bytes_to_write, [&](Bytes bytes) {
|
|
size_t offset = 0;
|
|
|
|
memcpy(bytes.offset(offset), &event.wd, sizeof(InodeWatcherEvent::watch_descriptor));
|
|
offset += sizeof(InodeWatcherEvent::watch_descriptor);
|
|
memcpy(bytes.offset(offset), &event.type, sizeof(InodeWatcherEvent::type));
|
|
offset += sizeof(InodeWatcherEvent::type);
|
|
|
|
if (event.path) {
|
|
size_t name_length = event.path->length() + 1;
|
|
memcpy(bytes.offset(offset), &name_length, sizeof(InodeWatcherEvent::name_length));
|
|
offset += sizeof(InodeWatcherEvent::name_length);
|
|
memcpy(bytes.offset(offset), event.path->characters(), name_length);
|
|
} else {
|
|
memset(bytes.offset(offset), 0, sizeof(InodeWatcherEvent::name_length));
|
|
}
|
|
|
|
return bytes.size();
|
|
});
|
|
evaluate_block_conditions();
|
|
return result;
|
|
}
|
|
|
|
ErrorOr<void> InodeWatcher::close()
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
|
|
for (auto& entry : m_wd_to_watches) {
|
|
auto& inode = const_cast<Inode&>(entry.value->inode);
|
|
inode.unregister_watcher({}, *this);
|
|
}
|
|
|
|
m_wd_to_watches.clear();
|
|
m_inode_to_watches.clear();
|
|
return {};
|
|
}
|
|
|
|
ErrorOr<NonnullOwnPtr<KString>> InodeWatcher::pseudo_path(OpenFileDescription const&) const
|
|
{
|
|
return KString::formatted("InodeWatcher:({})", m_wd_to_watches.size());
|
|
}
|
|
|
|
void InodeWatcher::notify_inode_event(Badge<Inode>, InodeIdentifier inode_id, InodeWatcherEvent::Type event_type, StringView name)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
|
|
auto it = m_inode_to_watches.find(inode_id);
|
|
if (it == m_inode_to_watches.end())
|
|
return;
|
|
|
|
auto& watcher = *it->value;
|
|
if (!(watcher.event_mask & static_cast<unsigned>(event_type)))
|
|
return;
|
|
|
|
OwnPtr<KString> path;
|
|
if (!name.is_null())
|
|
path = KString::try_create(name).release_value_but_fixme_should_propagate_errors();
|
|
m_queue.enqueue({ watcher.wd, event_type, move(path) });
|
|
evaluate_block_conditions();
|
|
}
|
|
|
|
ErrorOr<int> InodeWatcher::register_inode(Inode& inode, unsigned event_mask)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
|
|
if (m_inode_to_watches.find(inode.identifier()) != m_inode_to_watches.end())
|
|
return EEXIST;
|
|
|
|
int wd;
|
|
do {
|
|
wd = m_wd_counter.value();
|
|
|
|
m_wd_counter++;
|
|
if (m_wd_counter.has_overflow())
|
|
m_wd_counter = 1;
|
|
} while (m_wd_to_watches.find(wd) != m_wd_to_watches.end());
|
|
|
|
auto description = TRY(WatchDescription::create(wd, inode, event_mask));
|
|
|
|
TRY(m_inode_to_watches.try_set(inode.identifier(), description.ptr()));
|
|
auto set_result = m_wd_to_watches.try_set(wd, move(description));
|
|
if (set_result.is_error()) {
|
|
m_inode_to_watches.remove(inode.identifier());
|
|
return set_result.release_error();
|
|
}
|
|
|
|
auto register_result = inode.register_watcher({}, *this);
|
|
if (register_result.is_error()) {
|
|
m_inode_to_watches.remove(inode.identifier());
|
|
m_wd_to_watches.remove(wd);
|
|
return register_result.release_error();
|
|
}
|
|
|
|
return wd;
|
|
}
|
|
|
|
ErrorOr<void> InodeWatcher::unregister_by_wd(int wd)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
|
|
auto it = m_wd_to_watches.find(wd);
|
|
if (it == m_wd_to_watches.end())
|
|
return ENOENT;
|
|
|
|
auto& inode = it->value->inode;
|
|
inode.unregister_watcher({}, *this);
|
|
|
|
m_inode_to_watches.remove(inode.identifier());
|
|
m_wd_to_watches.remove(it);
|
|
|
|
return {};
|
|
}
|
|
|
|
void InodeWatcher::unregister_by_inode(Badge<Inode>, InodeIdentifier identifier)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
|
|
auto it = m_inode_to_watches.find(identifier);
|
|
if (it == m_inode_to_watches.end())
|
|
return;
|
|
|
|
// NOTE: no need to call unregister_watcher here, the Inode calls us.
|
|
|
|
m_inode_to_watches.remove(identifier);
|
|
m_wd_to_watches.remove(it->value->wd);
|
|
}
|
|
|
|
}
|