htcs: hook service objects up to (unimplemented) manager apis

This commit is contained in:
Michael Scire 2021-02-17 04:08:58 -08:00 committed by SciresM
parent f0ef9fb918
commit 61929d6e21
11 changed files with 396 additions and 41 deletions

View file

@ -206,11 +206,11 @@ namespace ams::sf {
} }
constexpr explicit operator Span<const T>() const { constexpr explicit operator Span<const T>() const {
return {this->GetPointer(), static_cast<ptrdiff_t>(this->GetSize())}; return {this->GetPointer(), this->GetSize()};
} }
constexpr Span<const T> ToSpan() const { constexpr Span<const T> ToSpan() const {
return {this->GetPointer(), static_cast<ptrdiff_t>(this->GetSize())}; return {this->GetPointer(), this->GetSize()};
} }
}; };
@ -238,11 +238,11 @@ namespace ams::sf {
} }
constexpr explicit operator Span<T>() const { constexpr explicit operator Span<T>() const {
return {this->GetPointer(), static_cast<ptrdiff_t>(this->GetSize())}; return {this->GetPointer(), this->GetSize()};
} }
constexpr Span<T> ToSpan() const { constexpr Span<T> ToSpan() const {
return {this->GetPointer(), static_cast<ptrdiff_t>(this->GetSize())}; return {this->GetPointer(), this->GetSize()};
} }
}; };

View file

@ -0,0 +1,31 @@
/*
* 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/>.
*/
#include <stratosphere.hpp>
#include "htcs_impl.hpp"
namespace ams::htcs::impl {
namespace {
constexpr const htcs::HtcsPeerName PeerNameAny = {""};
constexpr const htcs::HtcsPeerName DefaultHostName = {""};
}
const htcs::HtcsPeerName GetPeerNameAny() { return PeerNameAny; }
const htcs::HtcsPeerName GetDefaultHostName() { return DefaultHostName; }
}

View file

@ -0,0 +1,24 @@
/*
* 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 <stratosphere.hpp>
namespace ams::htcs::impl {
const htcs::HtcsPeerName GetPeerNameAny();
const htcs::HtcsPeerName GetDefaultHostName();
}

View file

@ -32,6 +32,36 @@ namespace ams::htcs::impl {
os::EventType *GetServiceAvailabilityEvent(); os::EventType *GetServiceAvailabilityEvent();
bool IsServiceAvailable(); bool IsServiceAvailable();
public:
void Socket(s32 *out_err, s32 *out_desc, bool enable_disconnection_emulation);
void Close(s32 *out_err, s32 *out_res, s32 desc);
void Connect(s32 *out_err, s32 *out_res, const SockAddrHtcs &address, s32 desc);
void Bind(s32 *out_err, s32 *out_res, const SockAddrHtcs &address, s32 desc);
void Listen(s32 *out_err, s32 *out_res, s32 backlog_count, s32 desc);
void Recv(s32 *out_err, s64 *out_size, char *buffer, size_t size, s32 flags, s32 desc);
void Send(s32 *out_err, s64 *out_size, const char *buffer, size_t size, s32 flags, s32 desc);
void Shutdown(s32 *out_err, s32 *out_res, s32 how, s32 desc);
void Fcntl(s32 *out_err, s32 *out_res, s32 command, s32 value, s32 desc);
Result AcceptStart(u32 *out_task_id, Handle *out_handle, s32 desc);
void AcceptResults(s32 *out_err, s32 *out_desc, SockAddrHtcs *out_address, u32 task_id, s32 desc);
Result RecvStart(u32 *out_task_id, Handle *out_handle, s64 size, s32 desc, s32 flags);
void RecvResults(s32 *out_err, s64 *out_size, char *buffer, s64 buffer_size, u32 task_id, s32 desc);
Result SendStart(u32 *out_task_id, Handle *out_handle, const char *buffer, s64 size, s32 desc, s32 flags);
Result SendLargeStart(u32 *out_task_id, Handle *out_handle, const char **buffers, const s64 *sizes, s32 count, s32 desc, s32 flags);
void SendResults(s32 *out_err, s64 *out_size, u32 task_id, s32 desc);
Result StartSend(u32 *out_task_id, Handle *out_handle, s32 desc, s64 size, s32 flags);
Result ContinueSend(s64 *out_size, const char *buffer, s64 buffer_size, u32 task_id, s32 desc);
void EndSend(s32 *out_err, s64 *out_size, u32 task_id, s32 desc);
Result StartRecv(u32 *out_task_id, Handle *out_handle, s64 size, s32 desc, s32 flags);
void EndRecv(s32 *out_err, s64 *out_size, char *buffer, s64 buffer_size, u32 task_id, s32 desc);
Result StartSelect(u32 *out_task_id, Handle *out_handle, Span<const int> read_handles, Span<const int> write_handles, Span<const int> exception_handles, s64 tv_sec, s64 tv_usec);
Result EndSelect(s32 *out_err, s32 *out_res, Span<int> read_handles, Span<int> write_handles, Span<int> exception_handles, u32 task_id);
}; };
} }

View file

@ -16,15 +16,14 @@
#include <stratosphere.hpp> #include <stratosphere.hpp>
#include "htcs_manager_service_object.hpp" #include "htcs_manager_service_object.hpp"
#include "htcs_socket_service_object.hpp" #include "htcs_socket_service_object.hpp"
#include "htcs_service_object_allocator.hpp"
#include "../impl/htcs_manager.hpp"
#include "../impl/htcs_impl.hpp"
namespace ams::htcs::server { namespace ams::htcs::server {
namespace { namespace {
struct ServiceObjectAllocatorTag;
using ServiceObjectAllocator = ams::sf::ExpHeapStaticAllocator<32_KB, ServiceObjectAllocatorTag>;
using ServiceObjectFactory = ams::sf::ObjectFactory<typename ServiceObjectAllocator::Policy>;
class StaticAllocatorInitializer { class StaticAllocatorInitializer {
public: public:
StaticAllocatorInitializer() { StaticAllocatorInitializer() {
@ -35,11 +34,13 @@ namespace ams::htcs::server {
} }
Result ManagerServiceObject::GetPeerNameAny(sf::Out<htcs::HtcsPeerName> out) { Result ManagerServiceObject::GetPeerNameAny(sf::Out<htcs::HtcsPeerName> out) {
AMS_ABORT("ManagerServiceObject::GetPeerNameAny"); *out = impl::GetPeerNameAny();
return ResultSuccess();
} }
Result ManagerServiceObject::GetDefaultHostName(sf::Out<htcs::HtcsPeerName> out) { Result ManagerServiceObject::GetDefaultHostName(sf::Out<htcs::HtcsPeerName> out) {
AMS_ABORT("ManagerServiceObject::GetDefaultHostName"); *out = impl::GetDefaultHostName();
return ResultSuccess();
} }
Result ManagerServiceObject::CreateSocketOld(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out) { Result ManagerServiceObject::CreateSocketOld(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out) {
@ -47,7 +48,20 @@ namespace ams::htcs::server {
} }
Result ManagerServiceObject::CreateSocket(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, bool enable_disconnection_emulation) { Result ManagerServiceObject::CreateSocket(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, bool enable_disconnection_emulation) {
AMS_ABORT("ManagerServiceObject::CreateSocket"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Create a new socket. */
s32 desc;
manager->Socket(out_err.GetPointer(), std::addressof(desc), enable_disconnection_emulation);
/* If an error occurred, we're done. */
R_SUCCEED_IF(*out_err != 0);
/* Create a new socket object. */
*out = ServiceObjectFactory::CreateSharedEmplaced<tma::ISocket, SocketServiceObject>(this, desc);
return ResultSuccess();
} }
Result ManagerServiceObject::RegisterProcessId(const sf::ClientProcessId &client_pid) { Result ManagerServiceObject::RegisterProcessId(const sf::ClientProcessId &client_pid) {
@ -61,11 +75,19 @@ namespace ams::htcs::server {
} }
Result ManagerServiceObject::StartSelect(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InMapAliasArray<s32> &read_handles, const sf::InMapAliasArray<s32> &write_handles, const sf::InMapAliasArray<s32> &exception_handles, s64 tv_sec, s64 tv_usec) { Result ManagerServiceObject::StartSelect(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InMapAliasArray<s32> &read_handles, const sf::InMapAliasArray<s32> &write_handles, const sf::InMapAliasArray<s32> &exception_handles, s64 tv_sec, s64 tv_usec) {
AMS_ABORT("ManagerServiceObject::StartSelect"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the select. */
return manager->StartSelect(out_task_id.GetPointer(), out_event.GetHandlePointer(), read_handles.ToSpan(), write_handles.ToSpan(), exception_handles.ToSpan(), tv_sec, tv_usec);
} }
Result ManagerServiceObject::EndSelect(sf::Out<s32> out_err, sf::Out<s32> out_res, const sf::OutMapAliasArray<s32> &read_handles, const sf::OutMapAliasArray<s32> &write_handles, const sf::OutMapAliasArray<s32> &exception_handles, u32 task_id) { Result ManagerServiceObject::EndSelect(sf::Out<s32> out_err, sf::Out<s32> out_res, const sf::OutMapAliasArray<s32> &read_handles, const sf::OutMapAliasArray<s32> &write_handles, const sf::OutMapAliasArray<s32> &exception_handles, u32 task_id) {
AMS_ABORT("ManagerServiceObject::EndSelect"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* End the select. */
return manager->EndSelect(out_err.GetPointer(), out_res.GetPointer(), read_handles.ToSpan(), write_handles.ToSpan(), exception_handles.ToSpan(), task_id);
} }
} }

View file

@ -0,0 +1,25 @@
/*
* 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 <stratosphere.hpp>
namespace ams::htcs::server {
struct ServiceObjectAllocatorTag;
using ServiceObjectAllocator = ams::sf::ExpHeapStaticAllocator<32_KB, ServiceObjectAllocatorTag>;
using ServiceObjectFactory = ams::sf::ObjectFactory<typename ServiceObjectAllocator::Policy>;
}

View file

@ -15,6 +15,8 @@
*/ */
#include <stratosphere.hpp> #include <stratosphere.hpp>
#include "htcs_socket_service_object.hpp" #include "htcs_socket_service_object.hpp"
#include "htcs_service_object_allocator.hpp"
#include "../impl/htcs_manager.hpp"
namespace ams::htcs::server { namespace ams::htcs::server {
@ -23,107 +25,273 @@ namespace ams::htcs::server {
} }
SocketServiceObject::~SocketServiceObject() { SocketServiceObject::~SocketServiceObject() {
AMS_ABORT("SocketServiceObject::~SocketServiceObject"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Close the underlying socket. */
s32 dummy_err, dummy_res;
manager->Close(std::addressof(dummy_err), std::addressof(dummy_res), m_desc);
} }
Result SocketServiceObject::Close(sf::Out<s32> out_err, sf::Out<s32> out_res) { Result SocketServiceObject::Close(sf::Out<s32> out_err, sf::Out<s32> out_res) {
AMS_ABORT("SocketServiceObject::Close"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Close the underlying socket. */
manager->Close(out_err.GetPointer(), out_res.GetPointer(), m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Connect(sf::Out<s32> out_err, sf::Out<s32> out_res, const htcs::SockAddrHtcs &address) { Result SocketServiceObject::Connect(sf::Out<s32> out_err, sf::Out<s32> out_res, const htcs::SockAddrHtcs &address) {
AMS_ABORT("SocketServiceObject::Connect"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the connect. */
manager->Connect(out_err.GetPointer(), out_res.GetPointer(), address, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Bind(sf::Out<s32> out_err, sf::Out<s32> out_res, const htcs::SockAddrHtcs &address) { Result SocketServiceObject::Bind(sf::Out<s32> out_err, sf::Out<s32> out_res, const htcs::SockAddrHtcs &address) {
AMS_ABORT("SocketServiceObject::Bind"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the bind. */
manager->Bind(out_err.GetPointer(), out_res.GetPointer(), address, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Listen(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 backlog_count) { Result SocketServiceObject::Listen(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 backlog_count) {
AMS_ABORT("SocketServiceObject::Listen"); /* Get the htcs manager. */
} auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
Result SocketServiceObject::Accept(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, sf::Out<htcs::SockAddrHtcs> out_address) { /* Perform the listen. */
AMS_ABORT("SocketServiceObject::Accept"); manager->Listen(out_err.GetPointer(), out_res.GetPointer(), backlog_count, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Recv(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, s32 flags) { Result SocketServiceObject::Recv(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, s32 flags) {
AMS_ABORT("SocketServiceObject::Recv"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the recv. */
manager->Recv(out_err.GetPointer(), out_size.GetPointer(), reinterpret_cast<char *>(buffer.GetPointer()), buffer.GetSize(), flags, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Send(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::InAutoSelectBuffer &buffer, s32 flags) { Result SocketServiceObject::Send(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::InAutoSelectBuffer &buffer, s32 flags) {
AMS_ABORT("SocketServiceObject::Send"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the send. */
manager->Send(out_err.GetPointer(), out_size.GetPointer(), reinterpret_cast<const char *>(buffer.GetPointer()), buffer.GetSize(), flags, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Shutdown(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 how) { Result SocketServiceObject::Shutdown(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 how) {
AMS_ABORT("SocketServiceObject::Shutdown"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the shutdown. */
manager->Shutdown(out_err.GetPointer(), out_res.GetPointer(), how, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::Fcntl(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 command, s32 value) { Result SocketServiceObject::Fcntl(sf::Out<s32> out_err, sf::Out<s32> out_res, s32 command, s32 value) {
AMS_ABORT("SocketServiceObject::Fcntl"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Perform the fcntl. */
manager->Fcntl(out_err.GetPointer(), out_res.GetPointer(), command, value, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::AcceptStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event) { Result SocketServiceObject::AcceptStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event) {
AMS_ABORT("SocketServiceObject::AcceptStart"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the accept. */
return manager->AcceptStart(out_task_id.GetPointer(), out_event.GetHandlePointer(), m_desc);
} }
Result SocketServiceObject::AcceptResults(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, sf::Out<htcs::SockAddrHtcs> out_address, u32 task_id) { Result SocketServiceObject::AcceptResults(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, sf::Out<htcs::SockAddrHtcs> out_address, u32 task_id) {
AMS_ABORT("SocketServiceObject::AcceptResults"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Get the accept results. */
s32 desc;
manager->AcceptResults(out_err.GetPointer(), std::addressof(desc), out_address.GetPointer(), task_id, m_desc);
/* If an error occurred, we're done. */
R_SUCCEED_IF(*out_err != 0);
/* Create a new socket object. */
*out = ServiceObjectFactory::CreateSharedEmplaced<tma::ISocket, SocketServiceObject>(m_manager.Get(), desc);
return ResultSuccess();
} }
Result SocketServiceObject::RecvStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s32 mem_size, s32 flags) { Result SocketServiceObject::RecvStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s32 mem_size, s32 flags) {
AMS_ABORT("SocketServiceObject::RecvStart"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the recv. */
return manager->RecvStart(out_task_id.GetPointer(), out_event.GetHandlePointer(), mem_size, m_desc, flags);
} }
Result SocketServiceObject::RecvResults(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, u32 task_id) { Result SocketServiceObject::RecvResults(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, u32 task_id) {
AMS_ABORT("SocketServiceObject::RecvResults"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Get the recv results. */
manager->RecvResults(out_err.GetPointer(), out_size.GetPointer(), reinterpret_cast<char *>(buffer.GetPointer()), buffer.GetSize(), task_id, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::RecvLargeStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s32 unaligned_size_start, s32 unaligned_size_end, s64 aligned_size, sf::CopyHandle mem_handle, s32 flags) { Result SocketServiceObject::RecvLargeStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s32 unaligned_size_start, s32 unaligned_size_end, s64 aligned_size, sf::CopyHandle mem_handle, s32 flags) {
AMS_ABORT("SocketServiceObject::RecvLargeStart"); /* Check that the transfer memory size is okay. */
R_UNLESS(util::IsIntValueRepresentable<size_t>(aligned_size), htcs::ResultInvalidSize());
/* Attach the transfer memory. */
os::TransferMemoryType tmem;
R_ABORT_UNLESS(os::AttachTransferMemory(std::addressof(tmem), static_cast<size_t>(aligned_size), mem_handle.GetValue(), true));
ON_SCOPE_EXIT { os::DestroyTransferMemory(std::addressof(tmem)); };
/* Map the transfer memory. */
void *address;
R_TRY(os::MapTransferMemory(std::addressof(address), std::addressof(tmem), os::MemoryPermission_None));
ON_SCOPE_EXIT { os::UnmapTransferMemory(std::addressof(tmem)); };
/* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the large receive. */
return manager->RecvStart(out_task_id.GetPointer(), out_event.GetHandlePointer(), unaligned_size_start + aligned_size + unaligned_size_end, m_desc, flags);
} }
Result SocketServiceObject::SendStartOld(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InAutoSelectBuffer &buffer, s32 flags) { Result SocketServiceObject::SendStartOld(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InAutoSelectBuffer &buffer, s32 flags) {
AMS_ABORT("SocketServiceObject::SendStartOld"); return this->SendStart(out_task_id, out_event, sf::InNonSecureAutoSelectBuffer(buffer.GetPointer(), buffer.GetSize()), flags);
} }
Result SocketServiceObject::SendLargeStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InAutoSelectBuffer &start_buffer, const sf::InAutoSelectBuffer &end_buffer, sf::CopyHandle mem_handle, s64 aligned_size, s32 flags) { Result SocketServiceObject::SendLargeStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InAutoSelectBuffer &start_buffer, const sf::InAutoSelectBuffer &end_buffer, sf::CopyHandle mem_handle, s64 aligned_size, s32 flags) {
AMS_ABORT("SocketServiceObject::SendLargeStart"); /* Check that the sizes are okay. */
R_UNLESS(util::IsIntValueRepresentable<s64>(start_buffer.GetSize()), htcs::ResultInvalidSize());
R_UNLESS(util::IsIntValueRepresentable<s64>(end_buffer.GetSize()), htcs::ResultInvalidSize());
R_UNLESS(util::IsIntValueRepresentable<size_t>(aligned_size), htcs::ResultInvalidSize());
/* Attach the transfer memory. */
os::TransferMemoryType tmem;
R_ABORT_UNLESS(os::AttachTransferMemory(std::addressof(tmem), static_cast<size_t>(aligned_size), mem_handle.GetValue(), true));
ON_SCOPE_EXIT { os::DestroyTransferMemory(std::addressof(tmem)); };
/* Map the transfer memory. */
void *address;
R_TRY(os::MapTransferMemory(std::addressof(address), std::addressof(tmem), os::MemoryPermission_None));
ON_SCOPE_EXIT { os::UnmapTransferMemory(std::addressof(tmem)); };
/* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the large send. */
constexpr auto NumBuffers = 3;
const char *pointers[NumBuffers] = { reinterpret_cast<const char *>(start_buffer.GetPointer()), static_cast<const char *>(address), reinterpret_cast<const char *>(end_buffer.GetPointer()) };
s64 sizes[NumBuffers] = { static_cast<s64>(start_buffer.GetSize()), aligned_size, static_cast<s64>(end_buffer.GetSize()) };
return manager->SendLargeStart(out_task_id.GetPointer(), out_event.GetHandlePointer(), pointers, sizes, NumBuffers, m_desc, flags);
} }
Result SocketServiceObject::SendResults(sf::Out<s32> out_err, sf::Out<s64> out_size, u32 task_id) { Result SocketServiceObject::SendResults(sf::Out<s32> out_err, sf::Out<s64> out_size, u32 task_id) {
AMS_ABORT("SocketServiceObject::SendResults"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Get the send results. */
manager->SendResults(out_err.GetPointer(), out_size.GetPointer(), task_id, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::StartSend(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, sf::Out<s64> out_max_size, s64 size, s32 flags) { Result SocketServiceObject::StartSend(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, sf::Out<s64> out_max_size, s64 size, s32 flags) {
AMS_ABORT("SocketServiceObject::StartSend"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the send. */
R_TRY(manager->StartSend(out_task_id.GetPointer(), out_event.GetHandlePointer(), m_desc, size, flags));
/* Set the output max size to the size. */
*out_max_size = size;
return ResultSuccess();
} }
Result SocketServiceObject::ContinueSendOld(sf::Out<s64> out_size, sf::Out<bool> out_wait, const sf::InAutoSelectBuffer &buffer, u32 task_id) { Result SocketServiceObject::ContinueSendOld(sf::Out<s64> out_size, sf::Out<bool> out_wait, const sf::InAutoSelectBuffer &buffer, u32 task_id) {
AMS_ABORT("SocketServiceObject::ContinueSendOld"); return this->ContinueSend(out_size, out_wait, sf::InNonSecureAutoSelectBuffer(buffer.GetPointer(), buffer.GetSize()), task_id);
} }
Result SocketServiceObject::EndSend(sf::Out<s32> out_err, sf::Out<s64> out_size, u32 task_id) { Result SocketServiceObject::EndSend(sf::Out<s32> out_err, sf::Out<s64> out_size, u32 task_id) {
AMS_ABORT("SocketServiceObject::EndSend"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* End the send. */
manager->EndSend(out_err.GetPointer(), out_size.GetPointer(), task_id, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::StartRecv(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s64 size, s32 flags) { Result SocketServiceObject::StartRecv(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, s64 size, s32 flags) {
AMS_ABORT("SocketServiceObject::StartRecv"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the recv. */
return manager->StartRecv(out_task_id.GetPointer(), out_event.GetHandlePointer(), size, m_desc, flags);
} }
Result SocketServiceObject::EndRecv(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, u32 task_id) { Result SocketServiceObject::EndRecv(sf::Out<s32> out_err, sf::Out<s64> out_size, const sf::OutAutoSelectBuffer &buffer, u32 task_id) {
AMS_ABORT("SocketServiceObject::EndRecv"); /* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* End the recv. */
manager->EndRecv(out_err.GetPointer(), out_size.GetPointer(), reinterpret_cast<char *>(buffer.GetPointer()), buffer.GetSize(), task_id, m_desc);
return ResultSuccess();
} }
Result SocketServiceObject::SendStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InNonSecureAutoSelectBuffer &buffer, s32 flags) { Result SocketServiceObject::SendStart(sf::Out<u32> out_task_id, sf::OutCopyHandle out_event, const sf::InNonSecureAutoSelectBuffer &buffer, s32 flags) {
AMS_ABORT("SocketServiceObject::SendStart"); /* Check that the sizes are okay. */
R_UNLESS(util::IsIntValueRepresentable<s64>(buffer.GetSize()), htcs::ResultInvalidSize());
/* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Start the send. */
return manager->SendStart(out_task_id.GetPointer(), out_event.GetHandlePointer(), reinterpret_cast<const char *>(buffer.GetPointer()), buffer.GetSize(), m_desc, flags);
} }
Result SocketServiceObject::ContinueSend(sf::Out<s64> out_size, sf::Out<bool> out_wait, const sf::InNonSecureAutoSelectBuffer &buffer, u32 task_id) { Result SocketServiceObject::ContinueSend(sf::Out<s64> out_size, sf::Out<bool> out_wait, const sf::InNonSecureAutoSelectBuffer &buffer, u32 task_id) {
AMS_ABORT("SocketServiceObject::ContinueSend"); /* Check that the sizes are okay. */
R_UNLESS(util::IsIntValueRepresentable<s64>(buffer.GetSize()), htcs::ResultInvalidSize());
/* Get the htcs manager. */
auto *manager = impl::HtcsManagerHolder::GetHtcsManager();
/* Continue the send. */
R_TRY(manager->ContinueSend(out_size.GetPointer(), reinterpret_cast<const char *>(buffer.GetPointer()), buffer.GetSize(), task_id, m_desc));
/* We aren't doing a waiting send. */
*out_wait = false;
return ResultSuccess();
} }
Result SocketServiceObject::GetPrimitive(sf::Out<s32> out) { Result SocketServiceObject::GetPrimitive(sf::Out<s32> out) {
AMS_ABORT("SocketServiceObject::GetPrimitive"); /* Get our descriptor. */
*out = m_desc;
return ResultSuccess();
} }
} }

View file

@ -0,0 +1,29 @@
/*
* 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/>.
*/
#include <stratosphere.hpp>
#include "htcs_manager_service_object.hpp"
#include "htcs_socket_service_object.hpp"
namespace ams::htcs::server {
#define AMS_HTCS_MANAGER_DEPRECATED_API() AMS_ABORT("Deprecated IHtcsManager API %s was called.\n", AMS_CURRENT_FUNCTION_NAME)
Result SocketServiceObject::Accept(sf::Out<s32> out_err, sf::Out<sf::SharedPointer<tma::ISocket>> out, sf::Out<htcs::SockAddrHtcs> out_address) {
/* NOTE: This is a deprecated API, and Nintendo aborts when it is called. */
AMS_HTCS_MANAGER_DEPRECATED_API();
}
}

View file

@ -39,6 +39,7 @@
#include <vapours/results/htc_results.hpp> #include <vapours/results/htc_results.hpp>
#include <vapours/results/htcfs_results.hpp> #include <vapours/results/htcfs_results.hpp>
#include <vapours/results/htclow_results.hpp> #include <vapours/results/htclow_results.hpp>
#include <vapours/results/htcs_results.hpp>
#include <vapours/results/i2c_results.hpp> #include <vapours/results/i2c_results.hpp>
#include <vapours/results/kvdb_results.hpp> #include <vapours/results/kvdb_results.hpp>
#include <vapours/results/loader_results.hpp> #include <vapours/results/loader_results.hpp>

View file

@ -0,0 +1,25 @@
/*
* 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/results/results_common.hpp>
namespace ams::htcs {
R_DEFINE_NAMESPACE_RESULT_MODULE(4);
R_DEFINE_ERROR_RESULT(InvalidSize, 2014);
}