mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-12 07:44:52 +00:00
133 lines
4.3 KiB
C++
133 lines
4.3 KiB
C++
/*
|
|
* Copyright (c) Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
#include "htclow_mux_ring_buffer.hpp"
|
|
|
|
namespace ams::htclow::mux {
|
|
|
|
void RingBuffer::Initialize(void *buffer, size_t buffer_size) {
|
|
/* Validate pre-conditions. */
|
|
AMS_ASSERT(m_buffer == nullptr);
|
|
AMS_ASSERT(m_read_only_buffer == nullptr);
|
|
|
|
/* Set our fields. */
|
|
m_buffer = buffer;
|
|
m_buffer_size = buffer_size;
|
|
m_is_read_only = false;
|
|
}
|
|
|
|
void RingBuffer::InitializeForReadOnly(const void *buffer, size_t buffer_size) {
|
|
/* Validate pre-conditions. */
|
|
AMS_ASSERT(m_buffer == nullptr);
|
|
AMS_ASSERT(m_read_only_buffer == nullptr);
|
|
|
|
/* Set our fields. */
|
|
m_read_only_buffer = const_cast<void *>(buffer);
|
|
m_buffer_size = buffer_size;
|
|
m_data_size = buffer_size;
|
|
m_is_read_only = true;
|
|
}
|
|
|
|
void RingBuffer::Clear() {
|
|
m_data_size = 0;
|
|
m_offset = 0;
|
|
m_can_discard = false;
|
|
}
|
|
|
|
Result RingBuffer::Read(void *dst, size_t size) {
|
|
/* Copy the data. */
|
|
R_TRY(this->Copy(dst, size));
|
|
|
|
/* Discard. */
|
|
R_TRY(this->Discard(size));
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result RingBuffer::Write(const void *data, size_t size) {
|
|
/* Validate pre-conditions. */
|
|
AMS_ASSERT(!m_is_read_only);
|
|
|
|
/* Check that our buffer can hold the data. */
|
|
R_UNLESS(m_buffer != nullptr, htclow::ResultChannelBufferOverflow());
|
|
R_UNLESS(m_data_size + size <= m_buffer_size, htclow::ResultChannelBufferOverflow());
|
|
|
|
/* Determine position and copy sizes. */
|
|
const size_t pos = (m_data_size + m_offset) % m_buffer_size;
|
|
const size_t left = std::min(m_buffer_size - pos, size);
|
|
const size_t over = size - left;
|
|
|
|
/* Copy. */
|
|
if (left != 0) {
|
|
std::memcpy(static_cast<u8 *>(m_buffer) + pos, data, left);
|
|
}
|
|
if (over != 0) {
|
|
std::memcpy(m_buffer, static_cast<const u8 *>(data) + left, over);
|
|
}
|
|
|
|
/* Update our data size. */
|
|
m_data_size += size;
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result RingBuffer::Copy(void *dst, size_t size) {
|
|
/* Select buffer to discard from. */
|
|
void *buffer = m_is_read_only ? m_read_only_buffer : m_buffer;
|
|
R_UNLESS(buffer != nullptr, htclow::ResultChannelBufferHasNotEnoughData());
|
|
|
|
/* Verify that we have enough data. */
|
|
R_UNLESS(m_data_size >= size, htclow::ResultChannelBufferHasNotEnoughData());
|
|
|
|
/* Determine position and copy sizes. */
|
|
const size_t pos = m_offset;
|
|
const size_t left = std::min(m_buffer_size - pos, size);
|
|
const size_t over = size - left;
|
|
|
|
/* Copy. */
|
|
if (left != 0) {
|
|
std::memcpy(dst, static_cast<const u8 *>(buffer) + pos, left);
|
|
}
|
|
if (over != 0) {
|
|
std::memcpy(static_cast<u8 *>(dst) + left, buffer, over);
|
|
}
|
|
|
|
/* Mark that we can discard. */
|
|
m_can_discard = true;
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result RingBuffer::Discard(size_t size) {
|
|
/* Select buffer to discard from. */
|
|
void *buffer = m_is_read_only ? m_read_only_buffer : m_buffer;
|
|
R_UNLESS(buffer != nullptr, htclow::ResultChannelBufferHasNotEnoughData());
|
|
|
|
/* Verify that the data we're discarding has been read. */
|
|
R_UNLESS(m_can_discard, htclow::ResultChannelCannotDiscard());
|
|
|
|
/* Verify that we have enough data. */
|
|
R_UNLESS(m_data_size >= size, htclow::ResultChannelBufferHasNotEnoughData());
|
|
|
|
/* Discard. */
|
|
m_offset = (m_offset + size) % m_buffer_size;
|
|
m_data_size -= size;
|
|
m_can_discard = false;
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
}
|