mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-12 07:44:52 +00:00
228 lines
7.5 KiB
C++
228 lines
7.5 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/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
#include "htclow_ctrl_state_machine.hpp"
|
|
#include "htclow_ctrl_service_channels.hpp"
|
|
|
|
namespace ams::htclow::ctrl {
|
|
|
|
HtcctrlStateMachine::HtcctrlStateMachine() : m_map(), m_state(HtcctrlState_DriverDisconnected), m_prev_state(HtcctrlState_DriverDisconnected), m_mutex() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
/* Initialize our map. */
|
|
m_map.Initialize(MaxChannelCount, m_map_buffer, sizeof(m_map_buffer));
|
|
|
|
/* Insert each service channel the map. */
|
|
for (const auto &channel : ServiceChannels) {
|
|
m_map.insert(std::make_pair<impl::ChannelInternalType, ServiceChannelState>(impl::ChannelInternalType{channel}, ServiceChannelState{}));
|
|
}
|
|
}
|
|
|
|
HtcctrlState HtcctrlStateMachine::GetHtcctrlState() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return m_state;
|
|
}
|
|
|
|
Result HtcctrlStateMachine::SetHtcctrlState(bool *out_transitioned, HtcctrlState state) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
/* Check that the transition is allowed. */
|
|
R_UNLESS(ctrl::IsStateTransitionAllowed(m_state, state), htclow::ResultHtcctrlStateTransitionNotAllowed());
|
|
|
|
/* Get the state pre-transition. */
|
|
const auto old_state = m_state;
|
|
|
|
/* Set the state. */
|
|
this->SetStateWithoutCheckInternal(state);
|
|
|
|
/* Note whether we transitioned. */
|
|
*out_transitioned = state != old_state;
|
|
R_SUCCEED();
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsInformationNeeded() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return !ctrl::IsDisconnected(m_state) && m_state != HtcctrlState_DriverConnected;
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsDisconnectionNeeded() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return !ctrl::IsDisconnected(m_state) && m_state != HtcctrlState_Sleep && m_state != HtcctrlState_DriverConnected;
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsConnected() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsConnected(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsReadied() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsReadied(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsUnconnectable() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return !ctrl::IsConnected(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsDisconnected() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsDisconnected(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsSleeping() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsSleeping(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsConnectedStatusChanged() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsConnected(m_prev_state) ^ ctrl::IsConnected(m_state);
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsSleepingStatusChanged() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return ctrl::IsSleeping(m_prev_state) ^ ctrl::IsSleeping(m_state);
|
|
}
|
|
|
|
void HtcctrlStateMachine::SetStateWithoutCheckInternal(HtcctrlState state) {
|
|
if (m_state != state) {
|
|
/* Clear service channel states, if we should. */
|
|
if (ctrl::IsDisconnected(state)) {
|
|
this->ClearServiceChannelStates();
|
|
}
|
|
|
|
/* Transition our state. */
|
|
m_prev_state = m_state;
|
|
m_state = state;
|
|
}
|
|
}
|
|
|
|
void HtcctrlStateMachine::ClearServiceChannelStates() {
|
|
/* Clear all values in our map. */
|
|
for (auto &pair : m_map) {
|
|
pair.second = {};
|
|
}
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsPossibleToSendReady() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
return m_state == HtcctrlState_SentReadyFromHost && this->AreServiceChannelsConnecting();
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsUnsupportedServiceChannelToShutdown(const impl::ChannelInternalType &channel) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
auto it = m_map.find(channel);
|
|
return it != m_map.end() && it->second.connect == ServiceChannelConnect_ConnectingChecked && it->second.support == ServiceChannelSupport_Unsupported;
|
|
}
|
|
|
|
bool HtcctrlStateMachine::IsConnectable(const impl::ChannelInternalType &channel) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
auto it = m_map.find(channel);
|
|
return ctrl::IsConnected(m_state) && (it == m_map.end() || it->second.connect != ServiceChannelConnect_ConnectingChecked);
|
|
}
|
|
|
|
void HtcctrlStateMachine::SetConnecting(const impl::ChannelInternalType &channel) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
auto it = m_map.find(channel);
|
|
if (it != m_map.end() && it->second.connect != ServiceChannelConnect_ConnectingChecked) {
|
|
it->second.connect = ServiceChannelConnect_Connecting;
|
|
}
|
|
}
|
|
|
|
void HtcctrlStateMachine::SetNotConnecting(const impl::ChannelInternalType &channel) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
auto it = m_map.find(channel);
|
|
if (it != m_map.end() && it->second.connect != ServiceChannelConnect_ConnectingChecked) {
|
|
it->second.connect = ServiceChannelConnect_NotConnecting;
|
|
}
|
|
}
|
|
|
|
void HtcctrlStateMachine::SetConnectingChecked() {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
for (auto &pair : m_map) {
|
|
pair.second.connect = ServiceChannelConnect_ConnectingChecked;
|
|
}
|
|
}
|
|
|
|
void HtcctrlStateMachine::NotifySupportedServiceChannels(const impl::ChannelInternalType *channels, int num_channels) {
|
|
/* Lock ourselves. */
|
|
std::scoped_lock lk(m_mutex);
|
|
|
|
auto IsSupportedServiceChannel = [](const impl::ChannelInternalType &channel, const impl::ChannelInternalType *supported, int num_supported) ALWAYS_INLINE_LAMBDA -> bool {
|
|
for (auto i = 0; i < num_supported; ++i) {
|
|
if (channel.module_id == supported[i].module_id && channel.channel_id == supported[i].channel_id) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
for (auto &pair : m_map) {
|
|
if (IsSupportedServiceChannel(pair.first, channels, num_channels)) {
|
|
pair.second.support = ServiceChannelSupport_Suppported;
|
|
} else {
|
|
pair.second.support = ServiceChannelSupport_Unsupported;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool HtcctrlStateMachine::AreServiceChannelsConnecting() {
|
|
for (auto &pair : m_map) {
|
|
if (pair.second.connect != ServiceChannelConnect_Connecting) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
}
|