/* * 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 "cs_command_impl.hpp" /* Include command implementations. */ #include "cs_command_impl.inc" namespace ams::cs { namespace { struct ResponseFirmwareVersion { ResponseHeader header; settings::system::FirmwareVersion firmware_version; }; struct ResponseTakeScreenShot { ResponseHeader header; s32 data_size; s32 width; s32 height; }; constinit u8 g_data[0x1000]; } bool CommandProcessor::ProcessCommand(const CommandHeader &header, const u8 *body, s32 socket) { switch (header.command) { case Command_GetFirmwareVersion: SendFirmwareVersion(socket, header); break; case Command_TakeScreenShot: this->TakeScreenShot(header, socket, vi::LayerStack_ApplicationForDebug); break; case Command_TakeForegroundScreenShot: this->TakeScreenShot(header, socket, vi::LayerStack_LastFrame); break; /* TODO: Command support. */ default: scs::CommandProcessor::ProcessCommand(header, body, socket); break; } return true; } void CommandProcessor::SendFirmwareVersion(s32 socket, const CommandHeader &header) { /* Build the response. */ ResponseFirmwareVersion response = { .header = { .id = header.id, .response = Response_FirmwareVersion, .body_size = sizeof(response) - sizeof(response.header), }, .firmware_version = {}, }; /* Get the firmware version. */ const Result result = DoGetFirmwareVersionCommand(std::addressof(response.firmware_version)); if (R_SUCCEEDED(result)) { /* Send the response. */ auto lk = MakeSendGuardBlock(); Send(socket, std::addressof(response), sizeof(response)); } else { SendErrorResult(socket, header, result); } } void CommandProcessor::TakeScreenShot(const CommandHeader &header, s32 socket, vi::LayerStack layer_stack) { /* Create the command data. */ const CommandDataTakeScreenShot params = { .layer_stack = layer_stack, .buffer = g_data, .buffer_size = sizeof(g_data), }; /* Take the screenshot. */ Result result; { /* Acquire the send lock. */ auto lk = MakeSendGuardBlock(); /* Perform the command. */ result = DoTakeScreenShotCommand(params, [&](s32 data_size, s32 width, s32 height) { /* Use global buffer for response. */ ResponseTakeScreenShot *response = reinterpret_cast<ResponseTakeScreenShot *>(g_data); /* Set response header. */ *response = { .header = { .id = header.id, .response = Response_ScreenShot, .body_size = static_cast<u32>(sizeof(data_size) + sizeof(width) + sizeof(height) + data_size), }, .data_size = data_size, .width = width, .height = height, }; /* Send data. */ Send(socket, response, sizeof(*response)); }, [&](u8 *data, size_t data_size) { /* Send data. */ Send(socket, data, data_size); }); } /* Handle the error case. */ if (R_FAILED(result)) { SendErrorResult(socket, header, result); } } }