/* * This file is part of Luma3DS * Copyright (C) 2016-2021 Aurora Wright, TuxSH * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that 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 . * * Additional Terms 7.b and 7.c of GPLv3 apply to this file: * * Requiring preservation of specified reasonable legal notices or * author attributions in that material or in the Appropriate Legal * Notices displayed by works containing it. * * Prohibiting misrepresentation of the origin of that material, * or requiring that modified versions of such material be marked in * reasonable ways as different from the original version. */ #include <3ds.h> #include "luma_config.h" #include "menus/sysconfig.h" #include "memory.h" #include "draw.h" #include "fmt.h" #include "utils.h" #include "ifile.h" Menu sysconfigMenu = { "System configuration menu", { { "Control volume", METHOD, .method=&SysConfigMenu_AdjustVolume}, { "Control Wireless connection", METHOD, .method = &SysConfigMenu_ControlWifi }, { "Toggle LEDs", METHOD, .method = &SysConfigMenu_ToggleLEDs }, { "Toggle Wireless", METHOD, .method = &SysConfigMenu_ToggleWireless }, { "Toggle Power Button", METHOD, .method=&SysConfigMenu_TogglePowerButton }, { "Toggle power to card slot", METHOD, .method=&SysConfigMenu_ToggleCardIfPower}, {}, } }; bool isConnectionForced = false; s8 currVolumeSliderOverride = -1; void SysConfigMenu_ToggleLEDs(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); Draw_DrawString(10, 30, COLOR_WHITE, "Press A to toggle, press B to go back."); Draw_DrawString(10, 50, COLOR_RED, "WARNING:"); Draw_DrawString(10, 60, COLOR_WHITE, " * Entering sleep mode will reset the LED state!"); Draw_DrawString(10, 70, COLOR_WHITE, " * LEDs cannot be toggled when the battery is low!"); Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A) { mcuHwcInit(); u8 result; MCUHWC_ReadRegister(0x28, &result, 1); result = ~result; MCUHWC_WriteRegister(0x28, &result, 1); mcuHwcExit(); } else if(pressed & KEY_B) return; } while(!menuShouldExit); } void SysConfigMenu_ToggleWireless(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); bool nwmRunning = isServiceUsable("nwm::EXT"); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); Draw_DrawString(10, 30, COLOR_WHITE, "Press A to toggle, press B to go back."); u8 wireless = (*(vu8 *)((0x10140000 | (1u << 31)) + 0x180)); if(nwmRunning) { Draw_DrawString(10, 50, COLOR_WHITE, "Current status:"); Draw_DrawString(100, 50, (wireless ? COLOR_GREEN : COLOR_RED), (wireless ? " ON " : " OFF")); } else { Draw_DrawString(10, 50, COLOR_RED, "NWM isn't running."); Draw_DrawString(10, 60, COLOR_RED, "If you're currently on Test Menu,"); Draw_DrawString(10, 70, COLOR_RED, "exit then press R+RIGHT to toggle the WiFi."); Draw_DrawString(10, 80, COLOR_RED, "Otherwise, simply exit and wait a few seconds."); } Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A && nwmRunning) { nwmExtInit(); NWMEXT_ControlWirelessEnabled(!wireless); nwmExtExit(); } else if(pressed & KEY_B) return; } while(!menuShouldExit); } void SysConfigMenu_UpdateStatus(bool control) { MenuItem *item = &sysconfigMenu.items[0]; if(control) { item->title = "Control Wireless connection"; item->method = &SysConfigMenu_ControlWifi; } else { item->title = "Disable forced wireless connection"; item->method = &SysConfigMenu_DisableForcedWifiConnection; } } static bool SysConfigMenu_ForceWifiConnection(u32 slot) { char ssid[0x20 + 1] = {0}; isConnectionForced = false; if(R_FAILED(acInit())) return false; acuConfig config = {0}; ACU_CreateDefaultConfig(&config); ACU_SetNetworkArea(&config, 2); ACU_SetAllowApType(&config, 1 << slot); ACU_SetRequestEulaVersion(&config); Handle connectEvent = 0; svcCreateEvent(&connectEvent, RESET_ONESHOT); bool forcedConnection = false; if(R_SUCCEEDED(ACU_ConnectAsync(&config, connectEvent))) { if(R_SUCCEEDED(svcWaitSynchronization(connectEvent, -1)) && R_SUCCEEDED(ACU_GetSSID(ssid))) forcedConnection = true; } svcCloseHandle(connectEvent); if(forcedConnection) { isConnectionForced = true; SysConfigMenu_UpdateStatus(false); } else acExit(); char infoString[80] = {0}; u32 infoStringColor = forcedConnection ? COLOR_GREEN : COLOR_RED; if(forcedConnection) sprintf(infoString, "Succesfully forced a connection to: %s", ssid); else sprintf(infoString, "Failed to connect to slot %d", (int)slot + 1); Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); Draw_DrawString(10, 30, infoStringColor, infoString); Draw_DrawString(10, 40, COLOR_WHITE, "Press B to go back."); Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_B) break; } while(!menuShouldExit); return forcedConnection; } void SysConfigMenu_TogglePowerButton(void) { u32 mcuIRQMask; Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); mcuHwcInit(); MCUHWC_ReadRegister(0x18, (u8*)&mcuIRQMask, 4); mcuHwcExit(); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); Draw_DrawString(10, 30, COLOR_WHITE, "Press A to toggle, press B to go back."); Draw_DrawString(10, 50, COLOR_WHITE, "Current status:"); Draw_DrawString(100, 50, (((mcuIRQMask & 0x00000001) == 0x00000001) ? COLOR_RED : COLOR_GREEN), (((mcuIRQMask & 0x00000001) == 0x00000001) ? " DISABLED" : " ENABLED ")); Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A) { mcuHwcInit(); MCUHWC_ReadRegister(0x18, (u8*)&mcuIRQMask, 4); mcuIRQMask ^= 0x00000001; MCUHWC_WriteRegister(0x18, (u8*)&mcuIRQMask, 4); mcuHwcExit(); } else if(pressed & KEY_B) return; } while(!menuShouldExit); } void SysConfigMenu_ControlWifi(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); u32 slot = 0; char ssids[3][32] = {{0}}; Result resInit = acInit(); for (u32 i = 0; i < 3; i++) { // ssid[0] = '\0' if result is an error here ACI_LoadNetworkSetting(i); ACI_GetNetworkWirelessEssidSecuritySsid(ssids[i]); } if (R_SUCCEEDED(resInit)) acExit(); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); u32 posY = Draw_DrawString(10, 30, COLOR_WHITE, "Press A to force a connection to slot, B to go back\n\n"); for (u32 i = 0; i < 3; i++) { Draw_DrawString(10, posY + SPACING_Y * i, COLOR_TITLE, slot == i ? ">" : " "); Draw_DrawFormattedString(30, posY + SPACING_Y * i, COLOR_WHITE, "[%d] %s", (int)i + 1, ssids[i]); } Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A) { if(SysConfigMenu_ForceWifiConnection(slot)) { // Connection successfully forced, return from this menu to prevent ac handle refcount leakage. break; } Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); } else if(pressed & KEY_DOWN) { slot = (slot + 1) % 3; } else if(pressed & KEY_UP) { slot = (3 + slot - 1) % 3; } else if(pressed & KEY_B) return; } while(!menuShouldExit); } void SysConfigMenu_DisableForcedWifiConnection(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); acExit(); SysConfigMenu_UpdateStatus(true); do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); Draw_DrawString(10, 30, COLOR_WHITE, "Forced connection successfully disabled.\nNote: auto-connection may remain broken."); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_B) return; } while(!menuShouldExit); } void SysConfigMenu_ToggleCardIfPower(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); bool cardIfStatus = false; bool updatedCardIfStatus = false; do { Result res = FSUSER_CardSlotGetCardIFPowerStatus(&cardIfStatus); if (R_FAILED(res)) cardIfStatus = false; Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); u32 posY = Draw_DrawString(10, 30, COLOR_WHITE, "Press A to toggle, press B to go back.\n\n"); posY = Draw_DrawString(10, posY, COLOR_WHITE, "Inserting or removing a card will reset the status,\nand you'll need to reinsert the cart if you want to\nplay it.\n\n"); Draw_DrawString(10, posY, COLOR_WHITE, "Current status:"); Draw_DrawString(100, posY, !cardIfStatus ? COLOR_RED : COLOR_GREEN, !cardIfStatus ? " DISABLED" : " ENABLED "); Draw_FlushFramebuffer(); Draw_Unlock(); u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A) { if (!cardIfStatus) res = FSUSER_CardSlotPowerOn(&updatedCardIfStatus); else res = FSUSER_CardSlotPowerOff(&updatedCardIfStatus); if (R_SUCCEEDED(res)) cardIfStatus = !updatedCardIfStatus; } else if(pressed & KEY_B) return; } while(!menuShouldExit); } static Result SysConfigMenu_ApplyVolumeOverride(void) { // Thanks profi200! u8 tmp; Result res = cdcChkInit(); if (R_SUCCEEDED(res)) res = CDCCHK_ReadRegisters2(0, 116, &tmp, 1); // CDC_REG_VOL_MICDET_PIN_SAR_ADC if (currVolumeSliderOverride >= 0) tmp &= ~0x80; else tmp |= 0x80; if (R_SUCCEEDED(res)) res = CDCCHK_WriteRegisters2(0, 116, &tmp, 1); if (currVolumeSliderOverride >= 0) { s8 calculated = -128 + (((float)currVolumeSliderOverride/100.f) * 108); if (calculated > -20) return -1; // Just in case s8 volumes[2] = {calculated, calculated}; // Volume in 0.5 dB steps. -128 (muted) to 48. Do not go above -20 (100%). if (R_SUCCEEDED(res)) res = CDCCHK_WriteRegisters2(0, 65, volumes, 2); // CDC_REG_DAC_L_VOLUME_CTRL, CDC_REG_DAC_R_VOLUME_CTRL } cdcChkExit(); return res; } void SysConfigMenu_LoadConfig(void) { s64 out = -1; svcGetSystemInfo(&out, 0x10000, 7); currVolumeSliderOverride = (s8)out; if (currVolumeSliderOverride >= 0) SysConfigMenu_ApplyVolumeOverride(); } void SysConfigMenu_AdjustVolume(void) { Draw_Lock(); Draw_ClearFramebuffer(); Draw_FlushFramebuffer(); Draw_Unlock(); s8 tempVolumeOverride = currVolumeSliderOverride; static s8 backupVolumeOverride = -1; if (backupVolumeOverride == -1) backupVolumeOverride = tempVolumeOverride >= 0 ? tempVolumeOverride : 85; do { Draw_Lock(); Draw_DrawString(10, 10, COLOR_TITLE, "System configuration menu"); u32 posY = Draw_DrawString(10, 30, COLOR_WHITE, "Y: Toggle volume slider override.\nDPAD: Adjust the volume level.\nA: Apply\nB: Go back\n\n"); Draw_DrawString(10, posY, COLOR_WHITE, "Current status:"); posY = Draw_DrawString(100, posY, (tempVolumeOverride == -1) ? COLOR_RED : COLOR_GREEN, (tempVolumeOverride == -1) ? " DISABLED" : " ENABLED "); if (tempVolumeOverride != -1) { posY = Draw_DrawFormattedString(30, posY, COLOR_WHITE, "\nValue: [%d%%]", tempVolumeOverride); } else { posY = Draw_DrawString(30, posY, COLOR_WHITE, "\n "); } u32 pressed = waitInputWithTimeout(1000); if(pressed & KEY_A) { currVolumeSliderOverride = tempVolumeOverride; Result res = SysConfigMenu_ApplyVolumeOverride(); LumaConfig_SaveSettings(); if (R_SUCCEEDED(res)) Draw_DrawString(10, posY, COLOR_GREEN, "\nSuccess!"); else Draw_DrawFormattedString(10, posY, COLOR_RED, "\nFailed: 0x%08lX", res); } else if(pressed & KEY_B) return; else if(pressed & KEY_Y) { Draw_DrawString(10, posY, COLOR_WHITE, "\n "); if (tempVolumeOverride == -1) { tempVolumeOverride = backupVolumeOverride; } else { backupVolumeOverride = tempVolumeOverride; tempVolumeOverride = -1; } } else if ((pressed & (KEY_DUP | KEY_DDOWN | KEY_DLEFT | KEY_DRIGHT)) && tempVolumeOverride != -1) { Draw_DrawString(10, posY, COLOR_WHITE, "\n "); if (pressed & KEY_DUP) tempVolumeOverride++; else if (pressed & KEY_DDOWN) tempVolumeOverride--; else if (pressed & KEY_DRIGHT) tempVolumeOverride+=10; else if (pressed & KEY_DLEFT) tempVolumeOverride-=10; if (tempVolumeOverride < 0) tempVolumeOverride = 0; if (tempVolumeOverride > 100) tempVolumeOverride = 100; } } while(!menuShouldExit); }