/* * Copyright (c) 2018-2020 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 . */ #include #include "htclow_ctrl_send_buffer.hpp" #include "htclow_ctrl_packet_factory.hpp" namespace ams::htclow::ctrl { bool HtcctrlSendBuffer::IsPriorPacket(HtcctrlPacketType packet_type) const { return packet_type == HtcctrlPacketType_DisconnectFromTarget; } bool HtcctrlSendBuffer::IsPosteriorPacket(HtcctrlPacketType packet_type) const { switch (packet_type) { case HtcctrlPacketType_ConnectFromTarget: case HtcctrlPacketType_ReadyFromTarget: case HtcctrlPacketType_SuspendFromTarget: case HtcctrlPacketType_ResumeFromTarget: case HtcctrlPacketType_BeaconResponse: case HtcctrlPacketType_InformationFromTarget: return true; default: return false; } } void HtcctrlSendBuffer::AddPacket(std::unique_ptr ptr) { /* Get the packet. */ HtcctrlPacket *packet = ptr.release(); /* Get the packet type. */ const auto packet_type = packet->GetHeader()->packet_type; /* Add the packet to the appropriate list. */ if (this->IsPriorPacket(packet_type)) { m_prior_packet_list.push_back(*packet); } else { AMS_ABORT_UNLESS(this->IsPosteriorPacket(packet_type)); m_posterior_packet_list.push_back(*packet); } } void HtcctrlSendBuffer::RemovePacket(const HtcctrlPacketHeader &header) { /* Get the packet type. */ const auto packet_type = header.packet_type; /* Remove the front from the appropriate list. */ HtcctrlPacket *packet; if (this->IsPriorPacket(packet_type)) { packet = std::addressof(m_prior_packet_list.front()); m_prior_packet_list.pop_front(); } else { AMS_ABORT_UNLESS(this->IsPosteriorPacket(packet_type)); packet = std::addressof(m_posterior_packet_list.front()); m_posterior_packet_list.pop_front(); } /* Delete the packet. */ m_packet_factory->Delete(packet); } bool HtcctrlSendBuffer::QueryNextPacket(HtcctrlPacketHeader *header, HtcctrlPacketBody *body, int *out_body_size) { if (!m_prior_packet_list.empty()) { this->CopyPacket(header, body, out_body_size, m_prior_packet_list.front()); return true; } else if (!m_posterior_packet_list.empty()) { this->CopyPacket(header, body, out_body_size, m_posterior_packet_list.front()); return true; } else { return false; } } void HtcctrlSendBuffer::CopyPacket(HtcctrlPacketHeader *header, HtcctrlPacketBody *body, int *out_body_size, const HtcctrlPacket &packet) { /* Get the body size. */ const int body_size = packet.GetBodySize(); AMS_ASSERT(0 <= body_size && body_size <= static_cast(sizeof(*body))); /* Copy the header. */ std::memcpy(header, packet.GetHeader(), sizeof(*header)); /* Copy the body. */ std::memcpy(body, packet.GetBody(), body_size); /* Set the output body size. */ *out_body_size = body_size; } }