2018-11-10 12:16:13 +01:00
/*
2021-10-04 21:59:10 +02:00
* Copyright ( c ) Atmosphère - NX
2018-11-10 12:16:13 +01:00
*
* 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/>.
*/
2020-05-12 00:02:10 +02:00
# include <stratosphere.hpp>
2018-11-10 12:16:13 +01:00
# include "fatal_task_screen.hpp"
2018-11-10 21:38:24 +01:00
# include "fatal_config.hpp"
2018-11-13 15:03:30 +01:00
# include "fatal_font.hpp"
2019-10-24 11:30:10 +02:00
namespace ams : : fatal : : srv {
2018-11-13 15:03:30 +01:00
2019-07-19 04:09:35 +02:00
/* Include Atmosphere logo into its own anonymous namespace. */
namespace {
2018-11-13 15:03:30 +01:00
2019-07-19 04:09:35 +02:00
# include "fatal_ams_logo.inc"
2018-11-13 15:03:30 +01:00
2019-07-19 04:09:35 +02:00
}
2018-11-13 15:03:30 +01:00
2019-07-19 04:09:35 +02:00
namespace {
2018-11-10 12:16:13 +01:00
2019-07-19 04:09:35 +02:00
/* Screen definitions. */
constexpr u32 FatalScreenWidth = 1280 ;
constexpr u32 FatalScreenHeight = 720 ;
constexpr u32 FatalScreenBpp = 2 ;
constexpr u32 FatalLayerZ = 100 ;
constexpr u32 FatalScreenWidthAlignedBytes = ( FatalScreenWidth * FatalScreenBpp + 63 ) & ~ 63 ;
constexpr u32 FatalScreenWidthAligned = FatalScreenWidthAlignedBytes / FatalScreenBpp ;
2021-01-18 17:48:47 +01:00
/* There should only be a single transfer memory (for nv). */
alignas ( os : : MemoryPageSize ) constinit u8 g_nv_transfer_memory [ 0x40000 ] ;
/* There should only be a single (1280*768) framebuffer. */
alignas ( os : : MemoryPageSize ) constinit u8 g_framebuffer_memory [ FatalScreenWidthAlignedBytes * util : : AlignUp ( FatalScreenHeight , 128 ) ] ;
}
}
2021-01-19 01:54:55 +01:00
extern " C " : : Result __nx_nv_create_tmem ( TransferMemory * t , u32 * out_size , Permission perm ) {
* out_size = sizeof ( ams : : fatal : : srv : : g_nv_transfer_memory ) ;
return tmemCreateFromMemory ( t , ams : : fatal : : srv : : g_nv_transfer_memory , sizeof ( ams : : fatal : : srv : : g_nv_transfer_memory ) , perm ) ;
2021-01-18 17:48:47 +01:00
}
namespace ams : : fatal : : srv {
namespace {
2019-07-19 04:09:35 +02:00
/* Pixel calculation helper. */
constexpr u32 GetPixelOffset ( u32 x , u32 y ) {
u32 tmp_pos = ( ( y & 127 ) / 16 ) + ( x / 32 * 8 ) + ( ( y / 16 / 8 ) * ( ( ( FatalScreenWidthAligned / 2 ) / 16 * 8 ) ) ) ;
tmp_pos * = 16 * 16 * 4 ;
tmp_pos + = ( ( y % 16 ) / 8 ) * 512 + ( ( x % 32 ) / 16 ) * 256 + ( ( y % 8 ) / 2 ) * 64 + ( ( x % 16 ) / 8 ) * 32 + ( y % 2 ) * 16 + ( x % 8 ) * 2 ; //This line is a modified version of code from the Tegra X1 datasheet.
return tmp_pos / 2 ;
2018-11-13 07:26:13 +01:00
}
2019-06-18 01:41:03 +02:00
2019-07-19 04:09:35 +02:00
/* Task definitions. */
2019-10-19 05:31:15 +02:00
class ShowFatalTask : public ITaskWithStack < 0x8000 > {
2019-07-19 04:09:35 +02:00
private :
ViDisplay display ;
ViLayer layer ;
NWindow win ;
2021-01-19 01:54:55 +01:00
NvMap map ;
2019-07-19 04:09:35 +02:00
private :
Result SetupDisplayInternal ( ) ;
Result SetupDisplayExternal ( ) ;
Result PrepareScreenForDrawing ( ) ;
2021-01-18 17:48:47 +01:00
void PreRenderFrameBuffer ( ) ;
2021-01-19 01:54:55 +01:00
Result InitializeNativeWindow ( ) ;
void DisplayPreRenderedFrame ( ) ;
2019-07-19 04:09:35 +02:00
Result ShowFatal ( ) ;
public :
virtual Result Run ( ) override ;
virtual const char * GetName ( ) const override {
return " ShowFatal " ;
}
} ;
2019-10-19 05:31:15 +02:00
class BacklightControlTask : public ITaskWithDefaultStack {
2019-07-19 04:09:35 +02:00
private :
void TurnOnBacklight ( ) ;
public :
virtual Result Run ( ) override ;
virtual const char * GetName ( ) const override {
return " BacklightControlTask " ;
}
} ;
/* Task globals. */
ShowFatalTask g_show_fatal_task ;
BacklightControlTask g_backlight_control_task ;
/* Task implementations. */
Result ShowFatalTask : : SetupDisplayInternal ( ) {
ViDisplay temp_display ;
/* Try to open the display. */
R_TRY_CATCH ( viOpenDisplay ( " Internal " , & temp_display ) ) {
2019-10-24 10:40:44 +02:00
R_CONVERT ( vi : : ResultNotFound , ResultSuccess ( ) ) ;
2019-07-19 04:09:35 +02:00
} R_END_TRY_CATCH ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Guarantee we close the display. */
ON_SCOPE_EXIT { viCloseDisplay ( & temp_display ) ; } ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Turn on the screen. */
2020-04-14 07:19:44 +02:00
if ( hos : : GetVersion ( ) > = hos : : Version_3_0_0 ) {
2019-08-06 03:55:04 +02:00
R_TRY ( viSetDisplayPowerState ( & temp_display , ViPowerState_On ) ) ;
} else {
/* Prior to 3.0.0, the ViPowerState enum was different (0 = Off, 1 = On). */
2019-08-07 22:02:02 +02:00
R_TRY ( viSetDisplayPowerState ( & temp_display , ViPowerState_On_Deprecated ) ) ;
2019-08-06 03:55:04 +02:00
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Set alpha to 1.0f. */
R_TRY ( viSetDisplayAlpha ( & temp_display , 1.0f ) ) ;
2018-11-13 07:26:13 +01:00
2019-10-24 10:40:44 +02:00
return ResultSuccess ( ) ;
2018-11-13 07:26:13 +01:00
}
2019-06-18 01:41:03 +02:00
2019-07-19 04:09:35 +02:00
Result ShowFatalTask : : SetupDisplayExternal ( ) {
ViDisplay temp_display ;
/* Try to open the display. */
R_TRY_CATCH ( viOpenDisplay ( " External " , & temp_display ) ) {
2019-10-24 10:40:44 +02:00
R_CONVERT ( vi : : ResultNotFound , ResultSuccess ( ) ) ;
2019-07-19 04:09:35 +02:00
} R_END_TRY_CATCH ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Guarantee we close the display. */
ON_SCOPE_EXIT { viCloseDisplay ( & temp_display ) ; } ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Set alpha to 1.0f. */
R_TRY ( viSetDisplayAlpha ( & temp_display , 1.0f ) ) ;
2018-11-13 07:26:13 +01:00
2019-10-24 10:40:44 +02:00
return ResultSuccess ( ) ;
2019-07-19 04:09:35 +02:00
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
Result ShowFatalTask : : PrepareScreenForDrawing ( ) {
/* Connect to vi. */
R_TRY ( viInitialize ( ViServiceType_Manager ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Close other content. */
viSetContentVisibility ( false ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Setup the two displays. */
R_TRY ( SetupDisplayInternal ( ) ) ;
R_TRY ( SetupDisplayExternal ( ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Open the default display. */
R_TRY ( viOpenDefaultDisplay ( & this - > display ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Reset the display magnification to its default value. */
2019-10-27 23:57:30 +01:00
s32 display_width , display_height ;
2019-07-19 04:09:35 +02:00
R_TRY ( viGetDisplayLogicalResolution ( & this - > display , & display_width , & display_height ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* viSetDisplayMagnification was added in 3.0.0. */
2020-04-14 07:19:44 +02:00
if ( hos : : GetVersion ( ) > = hos : : Version_3_0_0 ) {
2019-07-19 04:09:35 +02:00
R_TRY ( viSetDisplayMagnification ( & this - > display , 0 , 0 , display_width , display_height ) ) ;
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Create layer to draw to. */
R_TRY ( viCreateLayer ( & this - > display , & this - > layer ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Setup the layer. */
{
/* Display a layer of 1280 x 720 at 1.5x magnification */
/* NOTE: N uses 2 (770x400) RGBA4444 buffers (tiled buffer + linear). */
/* We use a single 1280x720 tiled RGB565 buffer. */
2021-01-19 01:54:55 +01:00
constexpr s32 RawWidth = FatalScreenWidth ;
constexpr s32 RawHeight = FatalScreenHeight ;
constexpr s32 LayerWidth = ( ( RawWidth ) * 3 ) / 2 ;
constexpr s32 LayerHeight = ( ( RawHeight ) * 3 ) / 2 ;
2019-04-22 21:40:53 +02:00
2021-01-19 01:54:55 +01:00
const float layer_x = static_cast < float > ( ( display_width - LayerWidth ) / 2 ) ;
const float layer_y = static_cast < float > ( ( display_height - LayerHeight ) / 2 ) ;
2018-11-10 20:59:55 +01:00
2021-01-19 01:54:55 +01:00
R_TRY ( viSetLayerSize ( & this - > layer , LayerWidth , LayerHeight ) ) ;
2018-11-10 20:59:55 +01:00
2019-07-19 04:09:35 +02:00
/* Set the layer's Z at display maximum, to be above everything else .*/
R_TRY ( viSetLayerZ ( & this - > layer , FatalLayerZ ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Center the layer in the screen. */
R_TRY ( viSetLayerPosition ( & this - > layer , layer_x , layer_y ) ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Create framebuffer. */
R_TRY ( nwindowCreateFromLayer ( & this - > win , & this - > layer ) ) ;
2021-01-19 01:54:55 +01:00
R_TRY ( this - > InitializeNativeWindow ( ) ) ;
2019-07-19 04:09:35 +02:00
}
2019-04-22 21:40:53 +02:00
2019-10-24 10:40:44 +02:00
return ResultSuccess ( ) ;
2018-11-13 15:03:30 +01:00
}
2019-04-22 21:40:53 +02:00
2021-01-18 17:48:47 +01:00
void ShowFatalTask : : PreRenderFrameBuffer ( ) {
2019-07-19 04:09:35 +02:00
const FatalConfig & config = GetFatalConfig ( ) ;
2019-04-22 21:40:53 +02:00
2021-01-18 17:48:47 +01:00
/* Pre-render the image into the static framebuffer. */
u16 * tiled_buf = reinterpret_cast < u16 * > ( g_framebuffer_memory ) ;
2019-04-22 21:40:53 +02:00
2021-01-18 17:48:47 +01:00
/* Temporarily use the NV transfer memory as font backing heap. */
font : : SetHeapMemory ( g_nv_transfer_memory , sizeof ( g_nv_transfer_memory ) ) ;
ON_SCOPE_EXIT { std : : memset ( g_nv_transfer_memory , 0 , sizeof ( g_nv_transfer_memory ) ) ; } ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Let the font manager know about our framebuffer. */
font : : ConfigureFontFramebuffer ( tiled_buf , GetPixelOffset ) ;
font : : SetFontColor ( 0xFFFF ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Draw a background. */
2021-01-18 17:48:47 +01:00
for ( size_t i = 0 ; i < sizeof ( g_framebuffer_memory ) / sizeof ( * tiled_buf ) ; i + + ) {
2019-07-19 04:09:35 +02:00
tiled_buf [ i ] = 0x39C9 ;
2018-11-14 12:23:28 +01:00
}
2018-11-14 02:53:26 +01:00
2020-01-04 07:35:11 +01:00
/* Draw the atmosphere logo in the upper right corner. */
const u32 start_x = 32 , start_y = 64 ;
2019-07-19 04:09:35 +02:00
for ( size_t y = 0 ; y < AtmosphereLogoHeight ; y + + ) {
for ( size_t x = 0 ; x < AtmosphereLogoWidth ; x + + ) {
2020-01-04 07:35:11 +01:00
tiled_buf [ GetPixelOffset ( FatalScreenWidth - AtmosphereLogoWidth - start_x + x , start_x + y ) ] = AtmosphereLogoData [ y * AtmosphereLogoWidth + x ] ;
2019-07-19 04:09:35 +02:00
}
2018-11-14 04:30:40 +01:00
}
2019-04-22 21:40:53 +02:00
2020-01-04 07:35:11 +01:00
/* Draw error message and firmware. */
font : : SetPosition ( start_x , start_y ) ;
2019-07-19 04:09:35 +02:00
font : : SetFontSize ( 16.0f ) ;
2019-10-24 10:40:44 +02:00
font : : PrintFormat ( config . GetErrorMessage ( ) , this - > context - > result . GetModule ( ) , this - > context - > result . GetDescription ( ) , this - > context - > result . GetValue ( ) ) ;
2019-07-19 04:09:35 +02:00
font : : AddSpacingLines ( 0.5f ) ;
2019-10-28 05:43:01 +01:00
font : : PrintFormatLine ( " Program: %016lX " , static_cast < u64 > ( this - > context - > program_id ) ) ;
2019-07-19 04:09:35 +02:00
font : : AddSpacingLines ( 0.5f ) ;
2020-09-18 06:18:44 +02:00
2021-09-05 19:08:30 +02:00
font : : PrintFormatLine ( " Firmware: %s (Atmosphère %u.%u.%u-%s) " , config . GetFirmwareVersion ( ) . display_version , ATMOSPHERE_RELEASE_VERSION , ams : : GetGitRevision ( ) ) ;
2019-07-19 04:09:35 +02:00
font : : AddSpacingLines ( 1.5f ) ;
2019-10-24 11:30:10 +02:00
if ( ! exosphere : : ResultVersionMismatch : : Includes ( this - > context - > result ) ) {
2019-07-19 04:09:35 +02:00
font : : Print ( config . GetErrorDescription ( ) ) ;
2018-11-14 12:23:28 +01:00
} else {
2019-07-19 04:09:35 +02:00
/* Print a special message for atmosphere version mismatch. */
2020-05-12 00:02:10 +02:00
font : : Print ( " Atmosphère version mismatch detected. \n \n "
" Please press the POWER Button to restart the console normally, or a VOL button \n "
" to reboot to a payload (or RCM, if none is present). If you are unable to \n "
" restart the console, hold the POWER Button for 12 seconds to turn the console off. \n \n "
" Please ensure that all Atmosphère components are updated. \n "
" github.com/Atmosphere-NX/Atmosphere/releases \n " ) ;
2018-11-14 12:23:28 +01:00
}
2018-11-14 02:53:26 +01:00
2019-07-19 04:09:35 +02:00
/* Add a line. */
2020-01-04 07:35:11 +01:00
for ( size_t x = start_x ; x < FatalScreenWidth - start_x ; x + + ) {
2019-07-19 04:09:35 +02:00
tiled_buf [ GetPixelOffset ( x , font : : GetY ( ) ) ] = 0xFFFF ;
2018-11-14 02:53:26 +01:00
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
font : : AddSpacingLines ( 1.5f ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
u32 backtrace_y = font : : GetY ( ) ;
u32 backtrace_x = 0 ;
2020-01-04 07:35:11 +01:00
u32 pc_x = 0 ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Note architecutre. */
const bool is_aarch32 = this - > context - > cpu_ctx . architecture = = CpuContext : : Architecture_Aarch32 ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
/* Print GPRs. */
font : : SetFontSize ( 14.0f ) ;
font : : Print ( " General Purpose Registers " ) ;
font : : PrintLine ( " " ) ;
2020-01-04 07:35:11 +01:00
font : : SetPosition ( start_x , font : : GetY ( ) ) ;
2019-07-19 04:09:35 +02:00
font : : AddSpacingLines ( 0.5f ) ;
if ( is_aarch32 ) {
for ( size_t i = 0 ; i < ( aarch32 : : RegisterName_GeneralPurposeCount / 2 ) ; i + + ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " %s: " , aarch32 : : CpuContext : : RegisterNameStrings [ i ] ) ;
font : : SetPosition ( x + 47 , font : : GetY ( ) ) ;
if ( this - > context - > cpu_ctx . aarch32_ctx . HasRegisterValue ( static_cast < aarch32 : : RegisterName > ( i ) ) ) {
font : : PrintMonospaceU32 ( this - > context - > cpu_ctx . aarch32_ctx . r [ i ] ) ;
font : : PrintMonospaceBlank ( 8 ) ;
2020-01-04 05:21:12 +01:00
} else {
font : : PrintMonospaceBlank ( 16 ) ;
2019-07-19 04:09:35 +02:00
}
font : : Print ( " " ) ;
2020-01-04 07:35:11 +01:00
pc_x = font : : GetX ( ) ;
2019-07-19 04:09:35 +02:00
font : : PrintFormat ( " %s: " , aarch32 : : CpuContext : : RegisterNameStrings [ i + ( aarch32 : : RegisterName_GeneralPurposeCount / 2 ) ] ) ;
2020-01-04 07:35:11 +01:00
font : : SetPosition ( pc_x + 47 , font : : GetY ( ) ) ;
2019-07-19 04:09:35 +02:00
if ( this - > context - > cpu_ctx . aarch32_ctx . HasRegisterValue ( static_cast < aarch32 : : RegisterName > ( i + ( aarch32 : : RegisterName_GeneralPurposeCount / 2 ) ) ) ) {
font : : PrintMonospaceU32 ( this - > context - > cpu_ctx . aarch32_ctx . r [ i + ( aarch32 : : RegisterName_GeneralPurposeCount / 2 ) ] ) ;
font : : PrintMonospaceBlank ( 8 ) ;
2020-01-04 05:21:12 +01:00
} else {
font : : PrintMonospaceBlank ( 16 ) ;
2019-07-19 04:09:35 +02:00
}
if ( i = = ( aarch32 : : RegisterName_GeneralPurposeCount / 2 ) - 1 ) {
font : : Print ( " " ) ;
backtrace_x = font : : GetX ( ) ;
}
font : : PrintLine ( " " ) ;
2020-01-04 07:35:11 +01:00
font : : SetPosition ( start_x , font : : GetY ( ) ) ;
2018-11-14 03:11:08 +01:00
}
2019-07-19 04:09:35 +02:00
} else {
for ( size_t i = 0 ; i < aarch64 : : RegisterName_GeneralPurposeCount / 2 ; i + + ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " %s: " , aarch64 : : CpuContext : : RegisterNameStrings [ i ] ) ;
font : : SetPosition ( x + 47 , font : : GetY ( ) ) ;
if ( this - > context - > cpu_ctx . aarch64_ctx . HasRegisterValue ( static_cast < aarch64 : : RegisterName > ( i ) ) ) {
font : : PrintMonospaceU64 ( this - > context - > cpu_ctx . aarch64_ctx . x [ i ] ) ;
} else {
font : : PrintMonospaceBlank ( 16 ) ;
}
font : : Print ( " " ) ;
2020-01-04 07:35:11 +01:00
pc_x = font : : GetX ( ) ;
2019-07-19 04:09:35 +02:00
font : : PrintFormat ( " %s: " , aarch64 : : CpuContext : : RegisterNameStrings [ i + ( aarch64 : : RegisterName_GeneralPurposeCount / 2 ) ] ) ;
2020-01-04 07:35:11 +01:00
font : : SetPosition ( pc_x + 47 , font : : GetY ( ) ) ;
2019-07-19 04:09:35 +02:00
if ( this - > context - > cpu_ctx . aarch64_ctx . HasRegisterValue ( static_cast < aarch64 : : RegisterName > ( i + ( aarch64 : : RegisterName_GeneralPurposeCount / 2 ) ) ) ) {
font : : PrintMonospaceU64 ( this - > context - > cpu_ctx . aarch64_ctx . x [ i + ( aarch64 : : RegisterName_GeneralPurposeCount / 2 ) ] ) ;
} else {
font : : PrintMonospaceBlank ( 16 ) ;
}
if ( i = = ( aarch64 : : RegisterName_GeneralPurposeCount / 2 ) - 1 ) {
font : : Print ( " " ) ;
backtrace_x = font : : GetX ( ) ;
}
font : : PrintLine ( " " ) ;
2020-01-04 07:35:11 +01:00
font : : SetPosition ( start_x , font : : GetY ( ) ) ;
2018-11-14 03:11:08 +01:00
}
2019-07-19 04:09:35 +02:00
}
2019-04-22 21:40:53 +02:00
2020-01-04 07:35:11 +01:00
/* Print PC. */
{
font : : SetPosition ( pc_x , backtrace_y ) ;
const u32 x = font : : GetX ( ) ;
font : : Print ( " PC: " ) ;
font : : SetPosition ( x + 47 , font : : GetY ( ) ) ;
}
if ( is_aarch32 ) {
font : : PrintMonospaceU32 ( this - > context - > cpu_ctx . aarch32_ctx . pc ) ;
} else {
font : : PrintMonospaceU64 ( this - > context - > cpu_ctx . aarch64_ctx . pc ) ;
}
2019-07-19 04:09:35 +02:00
/* Print Backtrace. */
u32 bt_size ;
if ( is_aarch32 ) {
bt_size = this - > context - > cpu_ctx . aarch32_ctx . stack_trace_size ;
} else {
bt_size = this - > context - > cpu_ctx . aarch64_ctx . stack_trace_size ;
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
font : : SetPosition ( backtrace_x , backtrace_y ) ;
if ( bt_size = = 0 ) {
if ( is_aarch32 ) {
font : : Print ( " Start Address: " ) ;
font : : PrintMonospaceU32 ( this - > context - > cpu_ctx . aarch32_ctx . base_address ) ;
font : : PrintLine ( " " ) ;
} else {
font : : Print ( " Start Address: " ) ;
font : : PrintMonospaceU64 ( this - > context - > cpu_ctx . aarch64_ctx . base_address ) ;
font : : PrintLine ( " " ) ;
2018-11-14 03:11:08 +01:00
}
2019-07-19 04:09:35 +02:00
} else {
if ( is_aarch32 ) {
font : : Print ( " Backtrace - Start Address: " ) ;
font : : PrintMonospaceU32 ( this - > context - > cpu_ctx . aarch32_ctx . base_address ) ;
font : : PrintLine ( " " ) ;
font : : AddSpacingLines ( 0.5f ) ;
for ( u32 i = 0 ; i < aarch32 : : CpuContext : : MaxStackTraceDepth / 2 ; i + + ) {
u32 bt_cur = 0 , bt_next = 0 ;
if ( i < this - > context - > cpu_ctx . aarch32_ctx . stack_trace_size ) {
bt_cur = this - > context - > cpu_ctx . aarch32_ctx . stack_trace [ i ] ;
}
if ( i + aarch32 : : CpuContext : : MaxStackTraceDepth / 2 < this - > context - > cpu_ctx . aarch32_ctx . stack_trace_size ) {
bt_next = this - > context - > cpu_ctx . aarch32_ctx . stack_trace [ i + aarch32 : : CpuContext : : MaxStackTraceDepth / 2 ] ;
}
if ( i < this - > context - > cpu_ctx . aarch32_ctx . stack_trace_size ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " BT[%02d]: " , i ) ;
font : : SetPosition ( x + 72 , font : : GetY ( ) ) ;
font : : PrintMonospaceU32 ( bt_cur ) ;
2020-01-04 05:21:12 +01:00
font : : PrintMonospaceBlank ( 8 ) ;
2019-07-19 04:09:35 +02:00
font : : Print ( " " ) ;
}
if ( i + aarch32 : : CpuContext : : MaxStackTraceDepth / 2 < this - > context - > cpu_ctx . aarch32_ctx . stack_trace_size ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " BT[%02d]: " , i + aarch32 : : CpuContext : : MaxStackTraceDepth / 2 ) ;
font : : SetPosition ( x + 72 , font : : GetY ( ) ) ;
font : : PrintMonospaceU32 ( bt_next ) ;
2020-01-04 05:21:12 +01:00
font : : PrintMonospaceBlank ( 8 ) ;
2019-07-19 04:09:35 +02:00
}
font : : PrintLine ( " " ) ;
font : : SetPosition ( backtrace_x , font : : GetY ( ) ) ;
}
} else {
font : : Print ( " Backtrace - Start Address: " ) ;
font : : PrintMonospaceU64 ( this - > context - > cpu_ctx . aarch64_ctx . base_address ) ;
font : : PrintLine ( " " ) ;
font : : AddSpacingLines ( 0.5f ) ;
for ( u32 i = 0 ; i < aarch64 : : CpuContext : : MaxStackTraceDepth / 2 ; i + + ) {
u64 bt_cur = 0 , bt_next = 0 ;
if ( i < this - > context - > cpu_ctx . aarch64_ctx . stack_trace_size ) {
bt_cur = this - > context - > cpu_ctx . aarch64_ctx . stack_trace [ i ] ;
}
if ( i + aarch64 : : CpuContext : : MaxStackTraceDepth / 2 < this - > context - > cpu_ctx . aarch64_ctx . stack_trace_size ) {
bt_next = this - > context - > cpu_ctx . aarch64_ctx . stack_trace [ i + aarch64 : : CpuContext : : MaxStackTraceDepth / 2 ] ;
}
if ( i < this - > context - > cpu_ctx . aarch64_ctx . stack_trace_size ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " BT[%02d]: " , i ) ;
font : : SetPosition ( x + 72 , font : : GetY ( ) ) ;
font : : PrintMonospaceU64 ( bt_cur ) ;
font : : Print ( " " ) ;
}
if ( i + aarch64 : : CpuContext : : MaxStackTraceDepth / 2 < this - > context - > cpu_ctx . aarch64_ctx . stack_trace_size ) {
u32 x = font : : GetX ( ) ;
font : : PrintFormat ( " BT[%02d]: " , i + aarch64 : : CpuContext : : MaxStackTraceDepth / 2 ) ;
font : : SetPosition ( x + 72 , font : : GetY ( ) ) ;
font : : PrintMonospaceU64 ( bt_next ) ;
}
font : : PrintLine ( " " ) ;
font : : SetPosition ( backtrace_x , font : : GetY ( ) ) ;
}
2018-11-14 03:11:08 +01:00
}
2019-07-19 04:09:35 +02:00
}
2021-01-18 17:48:47 +01:00
}
2021-01-19 01:54:55 +01:00
Result ShowFatalTask : : InitializeNativeWindow ( ) {
/* Setup nv driver. */
R_TRY ( nvInitialize ( ) ) ;
R_TRY ( nvMapInit ( ) ) ;
R_TRY ( nvFenceInit ( ) ) ;
/* Create nvmap. */
R_TRY ( nvMapCreate ( & this - > map , g_framebuffer_memory , sizeof ( g_framebuffer_memory ) , 0x20000 , NvKind_Pitch , true ) ) ;
/* Setup graphics buffer. */
{
NvGraphicBuffer grbuf = { } ;
grbuf . header . num_ints = ( sizeof ( NvGraphicBuffer ) - sizeof ( NativeHandle ) ) / 4 ;
grbuf . unk0 = - 1 ;
grbuf . magic = 0xDAFFCAFF ;
grbuf . pid = 42 ;
grbuf . usage = GRALLOC_USAGE_HW_COMPOSER | GRALLOC_USAGE_HW_RENDER | GRALLOC_USAGE_HW_TEXTURE ;
grbuf . format = PIXEL_FORMAT_RGB_565 ;
grbuf . ext_format = PIXEL_FORMAT_RGB_565 ;
grbuf . num_planes = 1 ;
grbuf . planes [ 0 ] . width = FatalScreenWidth ;
grbuf . planes [ 0 ] . height = FatalScreenHeight ;
grbuf . planes [ 0 ] . color_format = NvColorFormat_R5G6B5 ;
grbuf . planes [ 0 ] . layout = NvLayout_BlockLinear ;
grbuf . planes [ 0 ] . kind = NvKind_Generic_16BX2 ;
grbuf . planes [ 0 ] . block_height_log2 = 4 ;
grbuf . nvmap_id = nvMapGetId ( & this - > map ) ;
grbuf . stride = FatalScreenWidthAligned ;
grbuf . total_size = sizeof ( g_framebuffer_memory ) ;
grbuf . planes [ 0 ] . pitch = FatalScreenWidthAlignedBytes ;
grbuf . planes [ 0 ] . size = sizeof ( g_framebuffer_memory ) ;
grbuf . planes [ 0 ] . offset = 0 ;
R_TRY ( nwindowConfigureBuffer ( & this - > win , 0 , & grbuf ) ) ;
}
return ResultSuccess ( ) ;
}
void ShowFatalTask : : DisplayPreRenderedFrame ( ) {
s32 slot ;
R_ABORT_UNLESS ( nwindowDequeueBuffer ( & this - > win , & slot , nullptr ) ) ;
dd : : FlushDataCache ( g_framebuffer_memory , sizeof ( g_framebuffer_memory ) ) ;
R_ABORT_UNLESS ( nwindowQueueBuffer ( & this - > win , this - > win . cur_slot , NULL ) ) ;
}
2021-01-18 17:48:47 +01:00
Result ShowFatalTask : : ShowFatal ( ) {
/* Pre-render the framebuffer. */
PreRenderFrameBuffer ( ) ;
/* Prepare screen for drawing. */
2021-04-14 08:58:10 +02:00
R_ABORT_UNLESS ( PrepareScreenForDrawing ( ) ) ;
2021-01-18 17:48:47 +01:00
2021-01-19 01:54:55 +01:00
/* Display the pre-rendered frame. */
this - > DisplayPreRenderedFrame ( ) ;
2019-04-22 21:40:53 +02:00
2019-10-24 10:40:44 +02:00
return ResultSuccess ( ) ;
2018-11-14 02:53:26 +01:00
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
Result ShowFatalTask : : Run ( ) {
/* Don't show the fatal error screen until we've verified the battery is okay. */
2020-04-08 11:21:35 +02:00
this - > context - > battery_event - > Wait ( ) ;
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
return ShowFatal ( ) ;
}
2019-04-22 21:40:53 +02:00
2019-07-19 04:09:35 +02:00
void BacklightControlTask : : TurnOnBacklight ( ) {
lblSwitchBacklightOn ( 0 ) ;
}
2018-11-10 20:59:55 +01:00
2019-07-19 04:09:35 +02:00
Result BacklightControlTask : : Run ( ) {
TurnOnBacklight ( ) ;
2019-10-24 10:40:44 +02:00
return ResultSuccess ( ) ;
2019-07-19 04:09:35 +02:00
}
2018-11-10 20:59:55 +01:00
2019-07-19 04:09:35 +02:00
}
2018-11-10 12:16:13 +01:00
2019-07-19 04:09:35 +02:00
ITask * GetShowFatalTask ( const ThrowContext * ctx ) {
g_show_fatal_task . Initialize ( ctx ) ;
return & g_show_fatal_task ;
}
ITask * GetBacklightControlTask ( const ThrowContext * ctx ) {
g_backlight_control_task . Initialize ( ctx ) ;
return & g_backlight_control_task ;
}
2018-11-10 12:16:13 +01:00
}