mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-24 02:03:06 -05:00
79fa9765ca
We now use AK::Error and AK::ErrorOr<T> in both kernel and userspace! This was a slightly tedious refactoring that took a long time, so it's not unlikely that some bugs crept in. Nevertheless, it does pass basic functionality testing, and it's just real nice to finally see the same pattern in all contexts. :^)
99 lines
3 KiB
C++
99 lines
3 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/StringView.h>
|
|
#include <Kernel/Arch/x86/InterruptDisabler.h>
|
|
#include <Kernel/DoubleBuffer.h>
|
|
|
|
namespace Kernel {
|
|
|
|
inline void DoubleBuffer::compute_lockfree_metadata()
|
|
{
|
|
InterruptDisabler disabler;
|
|
m_empty = m_read_buffer_index >= m_read_buffer->size && m_write_buffer->size == 0;
|
|
m_space_for_writing = m_capacity - m_write_buffer->size;
|
|
}
|
|
|
|
ErrorOr<NonnullOwnPtr<DoubleBuffer>> DoubleBuffer::try_create(size_t capacity)
|
|
{
|
|
auto storage = TRY(KBuffer::try_create_with_size(capacity * 2, Memory::Region::Access::ReadWrite, "DoubleBuffer"));
|
|
return adopt_nonnull_own_or_enomem(new (nothrow) DoubleBuffer(capacity, move(storage)));
|
|
}
|
|
|
|
DoubleBuffer::DoubleBuffer(size_t capacity, NonnullOwnPtr<KBuffer> storage)
|
|
: m_write_buffer(&m_buffer1)
|
|
, m_read_buffer(&m_buffer2)
|
|
, m_storage(move(storage))
|
|
, m_capacity(capacity)
|
|
{
|
|
m_buffer1.data = m_storage->data();
|
|
m_buffer1.size = 0;
|
|
m_buffer2.data = m_storage->data() + capacity;
|
|
m_buffer2.size = 0;
|
|
m_space_for_writing = capacity;
|
|
}
|
|
|
|
void DoubleBuffer::flip()
|
|
{
|
|
VERIFY(m_read_buffer_index == m_read_buffer->size);
|
|
swap(m_read_buffer, m_write_buffer);
|
|
m_write_buffer->size = 0;
|
|
m_read_buffer_index = 0;
|
|
compute_lockfree_metadata();
|
|
}
|
|
|
|
ErrorOr<size_t> DoubleBuffer::write(const UserOrKernelBuffer& data, size_t size)
|
|
{
|
|
if (!size)
|
|
return 0;
|
|
MutexLocker locker(m_lock);
|
|
size_t bytes_to_write = min(size, m_space_for_writing);
|
|
u8* write_ptr = m_write_buffer->data + m_write_buffer->size;
|
|
TRY(data.read(write_ptr, bytes_to_write));
|
|
m_write_buffer->size += bytes_to_write;
|
|
compute_lockfree_metadata();
|
|
if (m_unblock_callback && !m_empty)
|
|
m_unblock_callback();
|
|
return bytes_to_write;
|
|
}
|
|
|
|
ErrorOr<size_t> DoubleBuffer::read(UserOrKernelBuffer& data, size_t size)
|
|
{
|
|
if (!size)
|
|
return 0;
|
|
MutexLocker locker(m_lock);
|
|
if (m_read_buffer_index >= m_read_buffer->size && m_write_buffer->size != 0)
|
|
flip();
|
|
if (m_read_buffer_index >= m_read_buffer->size)
|
|
return 0;
|
|
size_t nread = min(m_read_buffer->size - m_read_buffer_index, size);
|
|
TRY(data.write(m_read_buffer->data + m_read_buffer_index, nread));
|
|
m_read_buffer_index += nread;
|
|
compute_lockfree_metadata();
|
|
if (m_unblock_callback && m_space_for_writing > 0)
|
|
m_unblock_callback();
|
|
return nread;
|
|
}
|
|
|
|
ErrorOr<size_t> DoubleBuffer::peek(UserOrKernelBuffer& data, size_t size)
|
|
{
|
|
if (!size)
|
|
return 0;
|
|
MutexLocker locker(m_lock);
|
|
if (m_read_buffer_index >= m_read_buffer->size && m_write_buffer->size != 0) {
|
|
flip();
|
|
}
|
|
if (m_read_buffer_index >= m_read_buffer->size)
|
|
return 0;
|
|
size_t nread = min(m_read_buffer->size - m_read_buffer_index, size);
|
|
TRY(data.write(m_read_buffer->data + m_read_buffer_index, nread));
|
|
compute_lockfree_metadata();
|
|
if (m_unblock_callback && m_space_for_writing > 0)
|
|
m_unblock_callback();
|
|
return nread;
|
|
}
|
|
|
|
}
|