/* * 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 . */ #pragma once #include #include "uart_mitm_logger.hpp" #include "uart_shim.h" #define AMS_UART_IPORTSESSION_MITM_INTERFACE_INFO(C, H) \ AMS_SF_METHOD_INFO(C, H, 0, Result, OpenPort, (sf::Out out, u32 port, u32 baud_rate, UartFlowControlMode flow_control_mode, u32 device_variation, bool is_invert_tx, bool is_invert_rx, bool is_invert_rts, bool is_invert_cts, sf::CopyHandle send_handle, sf::CopyHandle receive_handle, u64 send_buffer_length, u64 receive_buffer_length), (out, port, baud_rate, flow_control_mode, device_variation, is_invert_tx, is_invert_rx, is_invert_rts, is_invert_cts, send_handle, receive_handle, send_buffer_length, receive_buffer_length)) \ AMS_SF_METHOD_INFO(C, H, 1, Result, OpenPortForDev, (sf::Out out, u32 port, u32 baud_rate, UartFlowControlMode flow_control_mode, u32 device_variation, bool is_invert_tx, bool is_invert_rx, bool is_invert_rts, bool is_invert_cts, sf::CopyHandle send_handle, sf::CopyHandle receive_handle, u64 send_buffer_length, u64 receive_buffer_length), (out, port, baud_rate, flow_control_mode, device_variation, is_invert_tx, is_invert_rx, is_invert_rts, is_invert_cts, send_handle, receive_handle, send_buffer_length, receive_buffer_length)) \ AMS_SF_METHOD_INFO(C, H, 2, Result, GetWritableLength, (sf::Out out), (out)) \ AMS_SF_METHOD_INFO(C, H, 3, Result, Send, (sf::Out out_size, const sf::InAutoSelectBuffer &data), (out_size, data)) \ AMS_SF_METHOD_INFO(C, H, 4, Result, GetReadableLength, (sf::Out out), (out)) \ AMS_SF_METHOD_INFO(C, H, 5, Result, Receive, (sf::Out out_size, const sf::OutAutoSelectBuffer &data), (out_size, data)) \ AMS_SF_METHOD_INFO(C, H, 6, Result, BindPortEvent, (sf::Out out, sf::OutCopyHandle out_event_handle, UartPortEventType port_event_type, s64 threshold), (out, out_event_handle, port_event_type, threshold)) \ AMS_SF_METHOD_INFO(C, H, 7, Result, UnbindPortEvent, (sf::Out out, UartPortEventType port_event_type), (out, port_event_type)) AMS_SF_DEFINE_INTERFACE(ams::mitm::uart::impl, IPortSession, AMS_UART_IPORTSESSION_MITM_INTERFACE_INFO) #define AMS_UART_MITM_INTERFACE_INFO(C, H) \ AMS_SF_METHOD_INFO(C, H, 6, Result, CreatePortSession, (sf::Out> out), (out)) AMS_SF_DEFINE_MITM_INTERFACE(ams::mitm::uart::impl, IUartMitmInterface, AMS_UART_MITM_INTERFACE_INFO) namespace ams::mitm::uart { class UartPortService { private: sm::MitmProcessInfo m_client_info; std::unique_ptr<::UartPortSession> m_srv; static constexpr inline size_t CacheBufferSize = 0x1000; s64 m_timestamp_base; s64 m_tick_base; char m_base_path[256]; size_t m_cmdlog_pos; size_t m_datalog_pos; bool m_datalog_ready; bool m_data_logging_enabled; FsFile m_datalog_file; u8 *m_send_cache_buffer; u8 *m_receive_cache_buffer; size_t m_send_cache_pos; size_t m_receive_cache_pos; bool TryGetCurrentTimestamp(u64 *out); void WriteCmdLog(const char *str); void WriteUartData(bool dir, const void* buffer, size_t size); public: UartPortService(const sm::MitmProcessInfo &cl, std::unique_ptr<::UartPortSession> s); virtual ~UartPortService() { std::shared_ptr logger = mitm::uart::g_logger; logger->WaitFinished(); uartPortSessionClose(this->m_srv.get()); fsFileClose(&this->m_datalog_file); std::free(this->m_send_cache_buffer); std::free(this->m_receive_cache_buffer); } public: /* Actual command API. */ Result OpenPort(sf::Out out, u32 port, u32 baud_rate, UartFlowControlMode flow_control_mode, u32 device_variation, bool is_invert_tx, bool is_invert_rx, bool is_invert_rts, bool is_invert_cts, sf::CopyHandle send_handle, sf::CopyHandle receive_handle, u64 send_buffer_length, u64 receive_buffer_length); Result OpenPortForDev(sf::Out out, u32 port, u32 baud_rate, UartFlowControlMode flow_control_mode, u32 device_variation, bool is_invert_tx, bool is_invert_rx, bool is_invert_rts, bool is_invert_cts, sf::CopyHandle send_handle, sf::CopyHandle receive_handle, u64 send_buffer_length, u64 receive_buffer_length); Result GetWritableLength(sf::Out out); Result Send(sf::Out out_size, const sf::InAutoSelectBuffer &data); Result GetReadableLength(sf::Out out); Result Receive(sf::Out out_size, const sf::OutAutoSelectBuffer &data); Result BindPortEvent(sf::Out out, sf::OutCopyHandle out_event_handle, UartPortEventType port_event_type, s64 threshold); Result UnbindPortEvent(sf::Out out, UartPortEventType port_event_type); }; static_assert(impl::IsIPortSession); class UartMitmService : public sf::MitmServiceImplBase { public: using MitmServiceImplBase::MitmServiceImplBase; public: static bool ShouldMitm(const sm::MitmProcessInfo &client_info) { /* We will mitm: * - bluetooth, for logging HCI. */ return client_info.program_id == ncm::SystemProgramId::Bluetooth; } public: Result CreatePortSession(sf::Out> out); }; static_assert(impl::IsIUartMitmInterface); }