/* * 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/>. */ #pragma once #include <stratosphere.hpp> namespace ams::lm::srv { class LogBuffer { NON_COPYABLE(LogBuffer); NON_MOVEABLE(LogBuffer); private: struct BufferInfo { u8 *m_head; size_t m_stored_size; size_t m_reference_count; }; public: using FlushFunction = bool (*)(const u8 *data, size_t size); private: BufferInfo m_buffers[2]; BufferInfo *m_push_buffer; BufferInfo *m_flush_buffer; size_t m_buffer_size; FlushFunction m_flush_function; os::SdkMutex m_push_buffer_mutex; os::SdkMutex m_flush_buffer_mutex; os::SdkConditionVariable m_cv_push_ready; os::SdkConditionVariable m_cv_flush_ready; bool m_push_canceled; size_t m_push_ready_wait_count; public: constexpr explicit LogBuffer(u8 *buffer, size_t buffer_size, FlushFunction f) : m_buffers{}, m_push_buffer(m_buffers + 0), m_flush_buffer(m_buffers + 1), m_buffer_size(buffer_size / 2), m_flush_function(f), m_push_buffer_mutex{}, m_flush_buffer_mutex{}, m_cv_push_ready{}, m_cv_flush_ready{}, m_push_canceled(false), m_push_ready_wait_count(0) { AMS_ASSERT(buffer != nullptr); AMS_ASSERT(buffer_size > 0); AMS_ASSERT(f != nullptr); m_buffers[0].m_head = buffer; m_buffers[1].m_head = buffer + (buffer_size / 2); } static LogBuffer &GetDefaultInstance(); bool Push(const void *data, size_t size) { return this->PushImpl(data, size, true); } bool TryPush(const void *data, size_t size) { return this->PushImpl(data, size, false); } void CancelPush(); bool Flush() { return this->FlushImpl(true); } bool TryFlush() { return this->FlushImpl(false); } private: bool PushImpl(const void *data, size_t size, bool blocking); bool FlushImpl(bool blocking); }; }