diff --git a/stratosphere/tma/client/Main.py b/stratosphere/tma/client/Main.py index 580e0ae52..b1469a1e1 100644 --- a/stratosphere/tma/client/Main.py +++ b/stratosphere/tma/client/Main.py @@ -18,8 +18,6 @@ def main(argc, argv): print 'Waiting for connection...' c.wait_connected() print 'Connected!' - while True: - c.send_packet('AAAAAAAA') return 0 if __name__ == '__main__': diff --git a/stratosphere/tma/client/Packet.py b/stratosphere/tma/client/Packet.py new file mode 100644 index 000000000..465be93e8 --- /dev/null +++ b/stratosphere/tma/client/Packet.py @@ -0,0 +1,116 @@ +# Copyright (c) 2018 Atmosphere-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 . +import zlib +import ServiceId +from struct import unpack as up, pack as pk + +HEADER_SIZE = 0x28 + +def crc32(s): + return zlib.crc32(s) & 0xFFFFFFFF + +class Packet(): + def __init__(self): + self.service = 0 + self.task = 0 + self.cmd = 0 + self.continuation = 0 + self.version = 0 + self.body_len = 0 + self.body = '' + self.offset = 0 + def load_header(self, header): + assert len(header) == HEADER_SIZE + self.service, self.task, self.cmd, self.continuation, self.version, self.body_len, \ + _, self.body_chk, self.hdr_chk = up('. + +def hash(s): + h = ord(s[0]) & 0xFFFFFFFF + for c in s: + h = ((1000003 * h) ^ ord(c)) & 0xFFFFFFFF + h ^= len(s) + return h + +USB_QUERY_TARGET = hash("USBQueryTarget") +USB_SEND_HOST_INFO = hash("USBSendHostInfo") +USB_CONNECT = hash("USBConnect") +USB_DISCONNECT = hash("USBDisconnect") + + \ No newline at end of file diff --git a/stratosphere/tma/client/UsbConnection.py b/stratosphere/tma/client/UsbConnection.py index 0f7d2cf8f..a7b009ad6 100644 --- a/stratosphere/tma/client/UsbConnection.py +++ b/stratosphere/tma/client/UsbConnection.py @@ -15,6 +15,8 @@ from UsbInterface import UsbInterface from threading import Thread, Condition from collections import deque import time +import ServiceId +from Packet import Packet class UsbConnection(UsbInterface): # Auto connect thread func. @@ -25,12 +27,6 @@ class UsbConnection(UsbInterface): except ValueError as e: continue def recv_thread(connection): - if connection.is_connected(): - try: - # If we've previously been connected, PyUSB will read garbage... - connection.recv_packet() - except ValueError: - pass while connection.is_connected(): try: connection.recv_packet() @@ -65,6 +61,7 @@ class UsbConnection(UsbInterface): self.conn_thrd.start() return self def __exit__(self, type, value, traceback): + self.disconnect() time.sleep(1) print 'Closing!' time.sleep(1) @@ -80,24 +77,43 @@ class UsbConnection(UsbInterface): self.conn_lock.acquire() assert not self.connected self.intf = intf - self.connected = True - self.conn_lock.notify() - self.conn_lock.release() - self.recv_thrd = Thread(target=UsbConnection.recv_thread, args=(self,)) - self.send_thrd = Thread(target=UsbConnection.send_thread, args=(self,)) - self.recv_thrd.daemon = True - self.send_thrd.daemon = True - self.recv_thrd.start() - self.send_thrd.start() + + try: + # Perform Query + Connection handshake + self.intf.send_packet(Packet().set_service(ServiceId.USB_QUERY_TARGET)) + query_resp = self.intf.read_packet() + print 'Found Switch, Protocol version 0x%x' % query_resp.read_u32() + + self.intf.send_packet(Packet().set_service(ServiceId.USB_SEND_HOST_INFO).write_u32(0).write_u32(0)) + + self.intf.send_packet(Packet().set_service(ServiceId.USB_CONNECT)) + resp = self.intf.read_packet() + + # Spawn threads + self.recv_thrd = Thread(target=UsbConnection.recv_thread, args=(self,)) + self.send_thrd = Thread(target=UsbConnection.send_thread, args=(self,)) + self.recv_thrd.daemon = True + self.send_thrd.daemon = True + self.recv_thrd.start() + self.send_thrd.start() + self.connected = True + finally: + # Finish connection. + self.conn_lock.notify() + self.conn_lock.release() def disconnect(self): self.conn_lock.acquire() if self.connected: self.connected = False + self.intf.send_packet(Packet().set_service(ServiceId.USB_DISCONNECT)) self.conn_lock.release() def recv_packet(self): - hdr, body = self.intf.read_packet() - print('Got Packet: %s' % body.encode('hex')) + packet = self.intf.read_packet() + assert type(packet) is Packet + dat = packet.read_u64() + print('Got Packet: %08x' % dat) def send_packet(self, packet): + assert type(packet) is Packet self.send_lock.acquire() if len(self.send_queue) == 0x40: self.send_lock.wait() diff --git a/stratosphere/tma/client/UsbInterface.py b/stratosphere/tma/client/UsbInterface.py index 90d91cab9..69169cf6e 100644 --- a/stratosphere/tma/client/UsbInterface.py +++ b/stratosphere/tma/client/UsbInterface.py @@ -11,11 +11,8 @@ # # You should have received a copy of the GNU General Public License # along with this program. If not, see . -import usb, zlib -from struct import unpack as up, pack as pk - -def crc32(s): - return zlib.crc32(s) & 0xFFFFFFFF +import usb +import Packet class UsbInterface(): def __init__(self): @@ -50,20 +47,16 @@ class UsbInterface(): def blocking_write(self, data): self.ep_out.write(data, 0xFFFFFFFFFFFFFFFF) def read_packet(self): - hdr = self.blocking_read(0x28) - _, _, _, body_size, _, _, _, _, body_chk, hdr_chk = up(' Packet.HEADER_SIZE): + self.blocking_write(data[Packet.HEADER_SIZE:]) diff --git a/stratosphere/tma/source/tma_conn_packet.hpp b/stratosphere/tma/source/tma_conn_packet.hpp index 03aaffc5a..18537c119 100644 --- a/stratosphere/tma/source/tma_conn_packet.hpp +++ b/stratosphere/tma/source/tma_conn_packet.hpp @@ -190,7 +190,7 @@ class TmaPacket { } template - TmaConnResult Read(const T &t) { + TmaConnResult Read(T &t) { return Read(&t, sizeof(T)); } diff --git a/stratosphere/tma/source/tma_conn_service_ids.hpp b/stratosphere/tma/source/tma_conn_service_ids.hpp index 7dc8ee331..1120d81ec 100644 --- a/stratosphere/tma/source/tma_conn_service_ids.hpp +++ b/stratosphere/tma/source/tma_conn_service_ids.hpp @@ -34,5 +34,13 @@ static constexpr u32 HashServiceName(const char *name) { enum class TmaService : u32 { Invalid = 0, + + /* Special nodes, for facilitating connection over USB. */ + UsbQueryTarget = HashServiceName("USBQueryTarget"), + UsbSendHostInfo = HashServiceName("USBSendHostInfo"), + UsbConnect = HashServiceName("USBConnect"), + UsbDisconnect = HashServiceName("USBDisconnect"), + + TestService = HashServiceName("AtmosphereTestService"), /* Temporary service, will be used to debug communications. */ }; diff --git a/stratosphere/tma/source/tma_conn_usb_connection.cpp b/stratosphere/tma/source/tma_conn_usb_connection.cpp index d9d40a214..75bf1a2a4 100644 --- a/stratosphere/tma/source/tma_conn_usb_connection.cpp +++ b/stratosphere/tma/source/tma_conn_usb_connection.cpp @@ -76,18 +76,54 @@ void TmaUsbConnection::RecvThreadFunc(void *arg) { this_ptr->SetConnected(true); while (res == TmaConnResult::Success) { - if (!this_ptr->IsConnected()) { - break; - } TmaPacket *packet = this_ptr->AllocateRecvPacket(); if (packet == nullptr) { std::abort(); } res = TmaUsbComms::ReceivePacket(packet); if (res == TmaConnResult::Success) { - TmaPacket *send_packet = this_ptr->AllocateSendPacket(); - send_packet->Write(i++); - this_ptr->send_queue.Send(reinterpret_cast(send_packet)); + switch (packet->GetServiceId()) { + case TmaService::UsbQueryTarget: { + this_ptr->SetConnected(false); + + res = this_ptr->SendQueryReply(packet); + + if (!this_ptr->has_woken_up) { + /* TODO: Cancel background work. */ + } + } + break; + case TmaService::UsbSendHostInfo: { + struct { + u32 version; + u32 sleeping; + } host_info; + packet->Read(host_info); + + if (!this_ptr->has_woken_up || !host_info.sleeping) { + /* TODO: Cancel background work. */ + } + } + break; + case TmaService::UsbConnect: { + res = this_ptr->SendQueryReply(packet); + + if (res == TmaConnResult::Success) { + this_ptr->SetConnected(true); + this_ptr->OnConnectionEvent(ConnectionEvent::Connected); + } + } + break; + case TmaService::UsbDisconnect: { + this_ptr->SetConnected(false); + this_ptr->OnDisconnected(); + + /* TODO: Cancel background work. */ + } + break; + default: + break; + } this_ptr->FreePacket(packet); } else { this_ptr->FreePacket(packet); @@ -153,3 +189,13 @@ TmaConnResult TmaUsbConnection::SendPacket(TmaPacket *packet) { return TmaConnResult::Disconnected; } } + +TmaConnResult TmaUsbConnection::SendQueryReply(TmaPacket *packet) { + packet->ClearOffset(); + struct { + u32 version; + } target_info; + target_info.version = 0; + packet->Write(target_info); + return TmaUsbComms::SendPacket(packet); +} diff --git a/stratosphere/tma/source/tma_conn_usb_connection.hpp b/stratosphere/tma/source/tma_conn_usb_connection.hpp index 15876ef3b..2927e6e3f 100644 --- a/stratosphere/tma/source/tma_conn_usb_connection.hpp +++ b/stratosphere/tma/source/tma_conn_usb_connection.hpp @@ -29,6 +29,7 @@ class TmaUsbConnection : public TmaConnection { static void SendThreadFunc(void *arg); static void RecvThreadFunc(void *arg); static void OnUsbStateChange(void *this_ptr, u32 state); + TmaConnResult SendQueryReply(TmaPacket *packet); void ClearSendQueue(); void StartThreads(); void StopThreads(); diff --git a/stratosphere/tma/source/tma_usb_comms.cpp b/stratosphere/tma/source/tma_usb_comms.cpp index f2cd014c8..32390c327 100644 --- a/stratosphere/tma/source/tma_usb_comms.cpp +++ b/stratosphere/tma/source/tma_usb_comms.cpp @@ -436,7 +436,7 @@ TmaConnResult TmaUsbComms::SendPacket(TmaPacket *packet) { res = TmaConnResult::GeneralFailure; } - if (res == TmaConnResult::Success) { + if (res == TmaConnResult::Success && 0 < body_len) { /* Copy body to send buffer. */ packet->CopyBodyTo(g_send_data_buf);