mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-31 11:15:51 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			146 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			146 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * 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 <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| #pragma once
 | |
| #include <vapours.hpp>
 | |
| #include <stratosphere/os/os_system_event.hpp>
 | |
| #include <stratosphere/sf/sf_lmem_utility.hpp>
 | |
| #include <stratosphere/usb/usb_limits.hpp>
 | |
| #include <stratosphere/usb/usb_device_types.hpp>
 | |
| #include <stratosphere/usb/ds/usb_i_ds_service.hpp>
 | |
| 
 | |
| namespace ams::usb {
 | |
| 
 | |
|     class DsInterface;
 | |
|     class DsEndpoint;
 | |
| 
 | |
|     class DsClient {
 | |
|         friend class DsInterface;
 | |
|         friend class DsEndpoint;
 | |
|         private:
 | |
|             /* NOTE: Nintendo uses a UnitHeap here on newer firmware versions. */
 | |
|             /* For now, we'll use an ExpHeap and do it the old way. */
 | |
|             sf::ExpHeapAllocator m_allocator{};
 | |
|             u8 m_heap_buffer[32_KB];
 | |
|             lmem::HeapHandle m_heap_handle{};
 | |
|             sf::SharedPointer<ds::IDsRootService> m_root_service{};
 | |
|             sf::SharedPointer<ds::IDsService> m_ds_service{};
 | |
|             bool m_is_initialized{false};
 | |
|             std::atomic<int> m_reference_count{0};
 | |
|             os::SystemEventType m_state_change_event{};
 | |
|             DsInterface *m_interfaces[DsLimitMaxInterfacesPerConfigurationCount]{};
 | |
|             bool m_is_enabled{false};
 | |
|         public:
 | |
|             DsClient() = default;
 | |
|             ~DsClient() { /* ... */ }
 | |
|         public:
 | |
|             Result Initialize(ComplexId complex_id);
 | |
|             Result Finalize();
 | |
| 
 | |
|             bool IsInitialized();
 | |
| 
 | |
|             Result EnableDevice();
 | |
|             Result DisableDevice();
 | |
| 
 | |
|             os::SystemEventType *GetStateChangeEvent();
 | |
|             Result GetState(UsbState *out);
 | |
| 
 | |
|             Result ClearDeviceData();
 | |
| 
 | |
|             Result AddUsbStringDescriptor(u8 *out_index, UsbStringDescriptor *desc);
 | |
|             Result DeleteUsbStringDescriptor(u8 index);
 | |
| 
 | |
|             Result SetUsbDeviceDescriptor(UsbDeviceDescriptor *desc, UsbDeviceSpeed speed);
 | |
| 
 | |
|             Result SetBinaryObjectStore(u8 *data, int size);
 | |
|         private:
 | |
|             Result AddInterface(DsInterface *intf, sf::SharedPointer<ds::IDsInterface> *out_srv, uint8_t bInterfaceNumber);
 | |
|             Result DeleteInterface(uint8_t bInterfaceNumber);
 | |
|     };
 | |
| 
 | |
|     class DsInterface {
 | |
|         friend class DsEndpoint;
 | |
|         private:
 | |
|             DsClient *m_client;
 | |
|             sf::SharedPointer<ds::IDsInterface> m_interface;
 | |
|             bool m_is_initialized;
 | |
|             std::atomic<int> m_reference_count;
 | |
|             os::SystemEventType m_setup_event;
 | |
|             os::SystemEventType m_ctrl_in_completion_event;
 | |
|             os::SystemEventType m_ctrl_out_completion_event;
 | |
|             UrbReport m_report;
 | |
|             u8 m_interface_num;
 | |
|             DsEndpoint *m_endpoints[UsbLimitMaxEndpointsCount];
 | |
|         public:
 | |
|             DsInterface() : m_client(nullptr), m_is_initialized(false), m_reference_count(0) { /* ... */ }
 | |
|             ~DsInterface() { /* ... */ }
 | |
|         public:
 | |
|             Result Initialize(DsClient *client, u8 bInterfaceNumber);
 | |
|             Result Finalize();
 | |
| 
 | |
|             Result AppendConfigurationData(UsbDeviceSpeed speed, void *data, u32 size);
 | |
| 
 | |
|             bool IsInitialized();
 | |
| 
 | |
|             os::SystemEventType *GetSetupEvent();
 | |
|             Result GetSetupPacket(UsbCtrlRequest *out);
 | |
| 
 | |
|             Result Enable();
 | |
|             Result Disable();
 | |
| 
 | |
|             Result CtrlRead(u32 *out_transferred, void *dst, u32 size);
 | |
|             Result CtrlWrite(u32 *out_transferred, void *dst, u32 size);
 | |
|             Result CtrlDone();
 | |
|             Result CtrlStall();
 | |
|         private:
 | |
|             Result AddEndpoint(DsEndpoint *ep, u8 bEndpointAddress, sf::SharedPointer<ds::IDsEndpoint> *out);
 | |
|             Result DeleteEndpoint(u8 bEndpointAddress);
 | |
| 
 | |
|             Result CtrlIn(u32 *out_transferred, void *dst, u32 size);
 | |
|             Result CtrlOut(u32 *out_transferred, void *dst, u32 size);
 | |
|     };
 | |
| 
 | |
|     class DsEndpoint {
 | |
|         private:
 | |
|             bool m_is_initialized;
 | |
|             bool m_is_new_format;
 | |
|             std::atomic<int> m_reference_count;
 | |
|             DsInterface *m_interface;
 | |
|             sf::SharedPointer<ds::IDsEndpoint> m_endpoint;
 | |
|             u8 m_address;
 | |
|             os::SystemEventType m_completion_event;
 | |
|             os::SystemEventType m_unknown_event;
 | |
|         public:
 | |
|             DsEndpoint() : m_is_initialized(false), m_is_new_format(false), m_reference_count(0) { /* ... */ }
 | |
|         public:
 | |
|             Result Initialize(DsInterface *interface, u8 bEndpointAddress);
 | |
|             Result Finalize();
 | |
| 
 | |
|             bool IsInitialized();
 | |
| 
 | |
|             Result PostBuffer(u32 *out_transferred, void *buf, u32 size);
 | |
|             Result PostBufferAsync(u32 *out_urb_id, void *buf, u32 size);
 | |
| 
 | |
|             os::SystemEventType *GetCompletionEvent();
 | |
| 
 | |
|             Result GetUrbReport(UrbReport *out);
 | |
| 
 | |
|             Result Cancel();
 | |
| 
 | |
|             Result SetZeroLengthTransfer(bool zlt);
 | |
|     };
 | |
| 
 | |
| }
 |