mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2025-11-27 13:38:21 +01:00
* troposphere: add haze MTP server * haze: adjust banner, new lines between class sections, single-statement if * haze: remove additional newlines between sections * haze: avoid use of reference out parameter * haze: console_main_loop: style * haze: event_reactor, file_system_proxy, ptp: style * haze: ptp_data_builder, ptp_object_database, ptp_object_heap, results, usb_session: style * haze: event_reactor, ptp_data_parser, async_usb_server, ptp_object_database, ptp_object_heap: style * haze: ptp_responder: style * haze: usb_session: style * haze: use svc defs from vapours * haze: misc comments * haze: fix pointer overflow check * haze: fix copy paste error * haze: use alignment, invalidate cached use values in console * haze: fix stray hex constant * haze: misc style * haze: fixes for windows * haze: add GetObjectPropsSupported, GetObjectPropDesc, GetObjectPropValue * haze: add SetObjectPropValue * haze: improve object database API * haze: improve WriteVariableLengthData readability * haze: fix directory renames on windows * haze: ptp_object_database: fix collation * haze: event_reactor: fix size validation * haze: ptp_responder: improve documentation * haze: ptp_responder: avoid unnecessary fs interaction in GetObjectPropValue * haze: ptp_responder: fix object deletion on windows * haze: tear down sessions on suspension * haze: prefer more specific data types * haze: misc * haze: fix usb 3.0 support * haze: improve host-to-device file transfer performance * haze: report device serial * haze: move static_assert to requires, fix alignment
175 lines
6.1 KiB
C++
175 lines
6.1 KiB
C++
/*
|
|
* 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 <haze/async_usb_server.hpp>
|
|
#include <haze/common.hpp>
|
|
#include <haze/ptp.hpp>
|
|
|
|
namespace haze {
|
|
|
|
class PtpDataBuilder final {
|
|
private:
|
|
AsyncUsbServer *m_server;
|
|
u32 m_transmitted_size;
|
|
u32 m_offset;
|
|
u8 *m_data;
|
|
bool m_disabled;
|
|
private:
|
|
Result Flush() {
|
|
ON_SCOPE_EXIT {
|
|
m_transmitted_size += m_offset;
|
|
m_offset = 0;
|
|
};
|
|
|
|
/* If we're disabled, we have nothing to do. */
|
|
R_SUCCEED_IF(m_disabled);
|
|
|
|
/* Otherwise, we should write our buffered data. */
|
|
R_RETURN(m_server->WritePacket(m_data, m_offset));
|
|
}
|
|
public:
|
|
constexpr explicit PtpDataBuilder(void *data, AsyncUsbServer *server) : m_server(server), m_transmitted_size(), m_offset(), m_data(static_cast<u8 *>(data)), m_disabled() { /* ... */ }
|
|
|
|
Result Commit() {
|
|
if (m_offset > 0) {
|
|
/* If there is remaining data left to write, write it now. */
|
|
R_TRY(this->Flush());
|
|
}
|
|
|
|
if (util::IsAligned(m_transmitted_size, PtpUsbBulkHighSpeedMaxPacketLength)) {
|
|
/* If the transmission size was a multiple of wMaxPacketSize, send a zero length packet. */
|
|
R_TRY(this->Flush());
|
|
}
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result AddBuffer(const u8 *buffer, u32 count) {
|
|
while (count > 0) {
|
|
/* Calculate how many bytes we can write now. */
|
|
const u32 write_size = std::min<u32>(count, haze::UsbBulkPacketBufferSize - m_offset);
|
|
|
|
/* Write this number of bytes. */
|
|
std::memcpy(m_data + m_offset, buffer, write_size);
|
|
m_offset += write_size;
|
|
buffer += write_size;
|
|
count -= write_size;
|
|
|
|
/* If our buffer is full, flush it. */
|
|
if (m_offset == haze::UsbBulkPacketBufferSize) {
|
|
R_TRY(this->Flush());
|
|
}
|
|
}
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
template <typename T>
|
|
Result Add(T value) {
|
|
u8 bytes[sizeof(T)];
|
|
|
|
std::memcpy(bytes, std::addressof(value), sizeof(T));
|
|
|
|
R_RETURN(this->AddBuffer(bytes, sizeof(T)));
|
|
}
|
|
|
|
Result AddDataHeader(PtpUsbBulkContainer &request, u32 data_size) {
|
|
R_TRY(this->Add<u32>(PtpUsbBulkHeaderLength + data_size));
|
|
R_TRY(this->Add<u16>(PtpUsbBulkContainerType_Data));
|
|
R_TRY(this->Add<u16>(request.code));
|
|
R_TRY(this->Add<u32>(request.trans_id));
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result AddResponseHeader(PtpUsbBulkContainer &request, PtpResponseCode code, u32 params_size) {
|
|
R_TRY(this->Add<u32>(PtpUsbBulkHeaderLength + params_size));
|
|
R_TRY(this->Add<u16>(PtpUsbBulkContainerType_Response));
|
|
R_TRY(this->Add<u16>(code));
|
|
R_TRY(this->Add<u32>(request.trans_id));
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
template <typename F>
|
|
Result WriteVariableLengthData(PtpUsbBulkContainer &request, F &&func) {
|
|
HAZE_ASSERT(m_offset == 0 && m_transmitted_size == 0);
|
|
|
|
/* Declare how many bytes the data will require to write. */
|
|
u32 data_size = 0;
|
|
{
|
|
/* Temporarily disable writing to calculate the size.*/
|
|
m_disabled = true;
|
|
|
|
ON_SCOPE_EXIT {
|
|
/* Report how many bytes were required. */
|
|
data_size = m_transmitted_size;
|
|
|
|
/* On exit, enable writing and reset sizes. */
|
|
m_transmitted_size = 0;
|
|
m_disabled = false;
|
|
m_offset = 0;
|
|
};
|
|
|
|
R_TRY(func());
|
|
R_TRY(this->Commit());
|
|
}
|
|
|
|
/* Actually copy and write the data. */
|
|
R_TRY(this->AddDataHeader(request, data_size));
|
|
R_TRY(func());
|
|
R_TRY(this->Commit());
|
|
|
|
/* We succeeded. */
|
|
R_SUCCEED();
|
|
}
|
|
|
|
template <typename T>
|
|
Result AddString(const T *str) {
|
|
/* Use one less than the maximum string length for maximum length with null terminator. */
|
|
const u8 len = static_cast<u8>(std::min<s32>(util::Strlen(str), PtpStringMaxLength - 1));
|
|
|
|
if (len > 0) {
|
|
/* Length is padded by null terminator for non-empty strings. */
|
|
R_TRY(this->Add<u8>(len + 1));
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
R_TRY(this->Add<u16>(str[i]));
|
|
}
|
|
|
|
R_TRY(this->Add<u16>(0));
|
|
} else {
|
|
R_TRY(this->Add<u8>(len));
|
|
}
|
|
|
|
R_SUCCEED();
|
|
}
|
|
|
|
template <typename T>
|
|
Result AddArray(const T *arr, u32 count) {
|
|
R_TRY(this->Add<u32>(count));
|
|
|
|
for (size_t i = 0; i < count; i++) {
|
|
R_TRY(this->Add<T>(arr[i]));
|
|
}
|
|
|
|
R_SUCCEED();
|
|
}
|
|
};
|
|
|
|
}
|