mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2024-12-24 13:31:12 +00:00
204 lines
6.2 KiB
C++
204 lines
6.2 KiB
C++
/*
|
|
* Copyright (c) 2018 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 <switch.h>
|
|
#include <stratosphere.hpp>
|
|
#include "tma_conn_usb_connection.hpp"
|
|
#include "tma_usb_comms.hpp"
|
|
|
|
static HosThread g_SendThread, g_RecvThread;
|
|
|
|
TmaConnResult TmaUsbConnection::InitializeComms() {
|
|
return TmaUsbComms::Initialize();
|
|
}
|
|
|
|
TmaConnResult TmaUsbConnection::FinalizeComms() {
|
|
return TmaUsbComms::Finalize();
|
|
}
|
|
|
|
void TmaUsbConnection::ClearSendQueue() {
|
|
uintptr_t _packet;
|
|
while (this->send_queue.TryReceive(&_packet)) {
|
|
TmaPacket *packet = reinterpret_cast<TmaPacket *>(_packet);
|
|
if (packet != nullptr) {
|
|
this->FreePacket(packet);
|
|
}
|
|
}
|
|
}
|
|
|
|
void TmaUsbConnection::SendThreadFunc(void *arg) {
|
|
TmaUsbConnection *this_ptr = reinterpret_cast<TmaUsbConnection *>(arg);
|
|
TmaConnResult res = TmaConnResult::Success;
|
|
TmaPacket *packet = nullptr;
|
|
|
|
while (res == TmaConnResult::Success) {
|
|
/* Receive a packet from the send queue. */
|
|
{
|
|
uintptr_t _packet;
|
|
this_ptr->send_queue.Receive(&_packet);
|
|
packet = reinterpret_cast<TmaPacket *>(_packet);
|
|
}
|
|
|
|
if (packet != nullptr) {
|
|
/* Send the packet if we're connected. */
|
|
if (this_ptr->IsConnected()) {
|
|
res = TmaUsbComms::SendPacket(packet);
|
|
}
|
|
|
|
this_ptr->FreePacket(packet);
|
|
this_ptr->Tick();
|
|
} else {
|
|
res = TmaConnResult::Disconnected;
|
|
}
|
|
}
|
|
|
|
this_ptr->SetConnected(false);
|
|
this_ptr->OnDisconnected();
|
|
}
|
|
|
|
void TmaUsbConnection::RecvThreadFunc(void *arg) {
|
|
TmaUsbConnection *this_ptr = reinterpret_cast<TmaUsbConnection *>(arg);
|
|
TmaConnResult res = TmaConnResult::Success;
|
|
this_ptr->SetConnected(true);
|
|
|
|
while (res == TmaConnResult::Success) {
|
|
TmaPacket *packet = this_ptr->AllocateRecvPacket();
|
|
if (packet == nullptr) { std::abort(); }
|
|
|
|
res = TmaUsbComms::ReceivePacket(packet);
|
|
|
|
if (res == TmaConnResult::Success) {
|
|
if (!IsMetaService(packet->GetServiceId())) {
|
|
this_ptr->OnReceivePacket(packet);
|
|
} else {
|
|
switch (packet->GetServiceId()) {
|
|
case TmaServiceId::UsbQueryTarget: {
|
|
this_ptr->SetConnected(false);
|
|
|
|
res = this_ptr->SendQueryReply(packet);
|
|
|
|
if (!this_ptr->has_woken_up) {
|
|
this_ptr->CancelTasks();
|
|
}
|
|
}
|
|
break;
|
|
case TmaServiceId::UsbSendHostInfo: {
|
|
struct {
|
|
u32 version;
|
|
u32 sleeping;
|
|
} host_info;
|
|
packet->Read<decltype(host_info)>(host_info);
|
|
|
|
if (!this_ptr->has_woken_up || !host_info.sleeping) {
|
|
this_ptr->CancelTasks();
|
|
}
|
|
}
|
|
break;
|
|
case TmaServiceId::UsbConnect: {
|
|
res = this_ptr->SendQueryReply(packet);
|
|
|
|
if (res == TmaConnResult::Success) {
|
|
this_ptr->SetConnected(true);
|
|
this_ptr->OnConnectionEvent(ConnectionEvent::Connected);
|
|
}
|
|
}
|
|
break;
|
|
case TmaServiceId::UsbDisconnect: {
|
|
this_ptr->SetConnected(false);
|
|
this_ptr->OnDisconnected();
|
|
|
|
this_ptr->CancelTasks();
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
this_ptr->FreePacket(packet);
|
|
}
|
|
} else {
|
|
this_ptr->FreePacket(packet);
|
|
}
|
|
}
|
|
|
|
this_ptr->SetConnected(false);
|
|
this_ptr->send_queue.Send(reinterpret_cast<uintptr_t>(nullptr));
|
|
}
|
|
|
|
void TmaUsbConnection::OnUsbStateChange(void *arg, u32 state) {
|
|
TmaUsbConnection *this_ptr = reinterpret_cast<TmaUsbConnection *>(arg);
|
|
switch (state) {
|
|
case 0:
|
|
case 6:
|
|
this_ptr->StopThreads();
|
|
break;
|
|
case 5:
|
|
this_ptr->StartThreads();
|
|
break;
|
|
}
|
|
}
|
|
|
|
void TmaUsbConnection::StartThreads() {
|
|
g_SendThread.Join();
|
|
g_RecvThread.Join();
|
|
|
|
g_SendThread.Initialize(&TmaUsbConnection::SendThreadFunc, this, 0x4000, 38);
|
|
g_RecvThread.Initialize(&TmaUsbConnection::RecvThreadFunc, this, 0x4000, 38);
|
|
|
|
this->ClearSendQueue();
|
|
g_SendThread.Start();
|
|
g_RecvThread.Start();
|
|
}
|
|
|
|
void TmaUsbConnection::StopThreads() {
|
|
TmaUsbComms::CancelComms();
|
|
g_SendThread.Join();
|
|
g_RecvThread.Join();
|
|
}
|
|
|
|
bool TmaUsbConnection::IsConnected() {
|
|
return this->is_connected;
|
|
}
|
|
|
|
TmaConnResult TmaUsbConnection::Disconnect() {
|
|
TmaUsbComms::SetStateChangeCallback(nullptr, nullptr);
|
|
|
|
this->StopThreads();
|
|
|
|
return TmaConnResult::Success;
|
|
}
|
|
|
|
TmaConnResult TmaUsbConnection::SendPacket(TmaPacket *packet) {
|
|
std::scoped_lock<HosMutex> lk(this->lock);
|
|
|
|
if (this->IsConnected()) {
|
|
this->send_queue.Send(reinterpret_cast<uintptr_t>(packet));
|
|
return TmaConnResult::Success;
|
|
} else {
|
|
this->FreePacket(packet);
|
|
this->Tick();
|
|
return TmaConnResult::Disconnected;
|
|
}
|
|
}
|
|
|
|
TmaConnResult TmaUsbConnection::SendQueryReply(TmaPacket *packet) {
|
|
packet->ClearOffset();
|
|
struct {
|
|
u32 version;
|
|
} target_info;
|
|
target_info.version = 0;
|
|
packet->Write<decltype(target_info)>(target_info);
|
|
return TmaUsbComms::SendPacket(packet);
|
|
}
|