MissionControl/mc_mitm/source/controllers/switch_controller.cpp

135 lines
4.8 KiB
C++

/*
* Copyright (c) 2020-2021 ndeadly
*
* 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 "switch_controller.hpp"
#include "../utils.hpp"
#include "../mcmitm_config.hpp"
#include <string>
namespace ams::controller {
namespace {
const uint8_t led_player_mappings[] = {
SwitchPlayerNumber_Unknown, //0000
SwitchPlayerNumber_One, //0001
SwitchPlayerNumber_Unknown, //0010
SwitchPlayerNumber_Two, //0011
SwitchPlayerNumber_Unknown, //0100
SwitchPlayerNumber_Six, //0101
SwitchPlayerNumber_Eight, //0110
SwitchPlayerNumber_Three, //0111
SwitchPlayerNumber_One, //1000
SwitchPlayerNumber_Five, //1001
SwitchPlayerNumber_Six, //1010
SwitchPlayerNumber_Seven, //1011
SwitchPlayerNumber_Two, //1100
SwitchPlayerNumber_Seven, //1101
SwitchPlayerNumber_Three, //1110
SwitchPlayerNumber_Four, //1111
};
}
Result LedsMaskToPlayerNumber(uint8_t led_mask, uint8_t *player_number) {
*player_number = led_player_mappings[led_mask & 0xf];
if (*player_number == SwitchPlayerNumber_Unknown)
return -1;
return ams::ResultSuccess();
}
std::string GetControllerDirectory(const bluetooth::Address *address) {
char path[0x100];
util::SNPrintf(path, sizeof(path), "sdmc:/config/MissionControl/controllers/%02x%02x%02x%02x%02x%02x",
address->address[0],
address->address[1],
address->address[2],
address->address[3],
address->address[4],
address->address[5]
);
return path;
}
Result SwitchController::Initialize(void) {
if (this->HasSetTsiDisableFlag())
m_settsi_supported = false;
return ams::ResultSuccess();
}
bool SwitchController::HasSetTsiDisableFlag(void) {
std::string flag_file = GetControllerDirectory(&m_address) + "/settsi_disable.flag";
bool file_exists;
if (R_SUCCEEDED(fs::HasFile(&file_exists, flag_file.c_str()))) {
return file_exists;
}
return false;
}
Result SwitchController::HandleIncomingReport(const bluetooth::HidReport *report) {
m_input_report.size = report->size;
std::memcpy(m_input_report.data, report->data, report->size);
auto switch_report = reinterpret_cast<SwitchReportData *>(m_input_report.data);
if (switch_report->id == 0x30) {
this->ApplyButtonCombos(&switch_report->input0x30.buttons);
} else if (switch_report->id == 0x21) {
auto response = reinterpret_cast<SwitchSubcommandResponse *>(&switch_report->input0x21.response);
if (response->id == SubCmd_SpiFlashRead) {
if (response->data.spi_flash_read.address == 0x6050) {
if (ams::mitm::GetSystemLanguage() == 10) {
uint8_t data[] = {0xff, 0xd7, 0x00, 0x00, 0x57, 0xb7, 0x00, 0x57, 0xb7, 0x00, 0x57, 0xb7};
std::memcpy(response->data.spi_flash_read.data, data, sizeof(data));
}
}
}
}
return bluetooth::hid::report::WriteHidDataReport(&m_address, &m_input_report);
}
Result SwitchController::HandleOutgoingReport(const bluetooth::HidReport *report) {
return bluetooth::hid::report::SendHidReport(&m_address, report);
}
Result SwitchController::HandleSetReport(uint32_t status) {
return bluetooth::hid::report::WriteHidSetReport(&m_address, status);
}
Result SwitchController::HandleGetReport(const bluetooth::HidReport *report) {
return bluetooth::hid::report::WriteHidGetReport(&m_address, report);
}
void SwitchController::ApplyButtonCombos(SwitchButtonData *buttons) {
// Home combo = MINUS + DPAD_DOWN
if (buttons->minus && buttons->dpad_down) {
buttons->home = 1;
buttons->minus = 0;
buttons->dpad_down = 0;
}
// Capture combo = MINUS + DPAD_UP
if (buttons->minus && buttons->dpad_up) {
buttons->capture = 1;
buttons->minus = 0;
buttons->dpad_up = 0;
}
}
}