mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-08 21:47:57 +00:00
fusee_cpp: Add display init/fatal error display logic
This commit is contained in:
parent
e7d7d8adfb
commit
ee1d1ea527
9 changed files with 6128 additions and 7 deletions
595
fusee_cpp/program/source/fusee_display.cpp
Normal file
595
fusee_cpp/program/source/fusee_display.cpp
Normal file
|
@ -0,0 +1,595 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 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 <exosphere.hpp>
|
||||||
|
#include "fusee_registers_di.hpp"
|
||||||
|
#include "fusee_display.hpp"
|
||||||
|
#include "fusee_print.hpp"
|
||||||
|
|
||||||
|
namespace ams::nxboot {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
#include "fusee_display_config.inc"
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
/* Helpful defines. */
|
||||||
|
constexpr int DsiWaitForCommandMilliSecondsMax = 250;
|
||||||
|
constexpr int DsiWaitForCommandCompletionMilliSeconds = 5;
|
||||||
|
constexpr int DsiWaitForHostControlMilliSecondsMax = 150;
|
||||||
|
|
||||||
|
constexpr inline int I2cAddressMax77620Pmic = 0x3C;
|
||||||
|
|
||||||
|
constexpr size_t GPIO_PORT3_CNF_0 = 0x200;
|
||||||
|
constexpr size_t GPIO_PORT3_OE_0 = 0x210;
|
||||||
|
constexpr size_t GPIO_PORT3_OUT_0 = 0x220;
|
||||||
|
|
||||||
|
constexpr size_t GPIO_PORT6_CNF_1 = 0x504;
|
||||||
|
constexpr size_t GPIO_PORT6_OE_1 = 0x514;
|
||||||
|
constexpr size_t GPIO_PORT6_OUT_1 = 0x524;
|
||||||
|
|
||||||
|
/* Globals. */
|
||||||
|
constexpr inline const uintptr_t PMC = secmon::MemoryRegionPhysicalDevicePmc .GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_disp1_regs = secmon::MemoryRegionPhysicalDeviceDisp1 .GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_dsi_regs = secmon::MemoryRegionPhysicalDeviceDsi .GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_clk_rst_regs = secmon::MemoryRegionPhysicalDeviceClkRst .GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_gpio_regs = secmon::MemoryRegionPhysicalDeviceGpio .GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_apb_misc_regs = secmon::MemoryRegionPhysicalDeviceApbMisc.GetAddress();
|
||||||
|
constexpr inline const uintptr_t g_mipi_cal_regs = secmon::MemoryRegionPhysicalDeviceMipiCal.GetAddress();
|
||||||
|
|
||||||
|
constinit u32 *g_frame_buffer = nullptr;
|
||||||
|
|
||||||
|
constinit u32 g_lcd_vendor = 0;
|
||||||
|
|
||||||
|
inline void DoRegisterWrites(uintptr_t base_address, const RegisterWrite *reg_writes, size_t num_writes) {
|
||||||
|
for (size_t i = 0; i < num_writes; i++) {
|
||||||
|
reg::Write(base_address + reg_writes[i].offset, reg_writes[i].value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void DoSocDependentRegisterWrites(uintptr_t base_address, const RegisterWrite *reg_writes_erista, size_t num_writes_erista, const RegisterWrite *reg_writes_mariko, size_t num_writes_mariko) {
|
||||||
|
switch (fuse::GetSocType()) {
|
||||||
|
case fuse::SocType_Erista: DoRegisterWrites(base_address, reg_writes_erista, num_writes_erista); break;
|
||||||
|
case fuse::SocType_Mariko: DoRegisterWrites(base_address, reg_writes_mariko, num_writes_mariko); break;
|
||||||
|
AMS_UNREACHABLE_DEFAULT_CASE();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void DoSleepOrRegisterWrites(uintptr_t base_address, const SleepOrRegisterWrite *reg_writes, size_t num_writes) {
|
||||||
|
for (size_t i = 0; i < num_writes; i++) {
|
||||||
|
switch (reg_writes[i].kind) {
|
||||||
|
case SleepOrRegisterWriteKind_Write:
|
||||||
|
reg::Write(base_address + sizeof(u32) * reg_writes[i].offset, reg_writes[i].value);
|
||||||
|
break;
|
||||||
|
case SleepOrRegisterWriteKind_Sleep:
|
||||||
|
util::WaitMicroSeconds(reg_writes[i].offset * UINT64_C(1000));
|
||||||
|
break;
|
||||||
|
AMS_UNREACHABLE_DEFAULT_CASE();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void WaitDsiTrigger() {
|
||||||
|
const u32 timeout = util::GetMicroSeconds() + (DsiWaitForCommandMilliSecondsMax * 1000u);
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (util::GetMicroSeconds() >= timeout) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (reg::Read(g_dsi_regs + sizeof(u32) * DSI_TRIGGER) == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
util::WaitMicroSeconds(DsiWaitForCommandCompletionMilliSeconds * 1000u);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WaitDsiHostControl() {
|
||||||
|
const u32 timeout = util::GetMicroSeconds() + (DsiWaitForHostControlMilliSecondsMax * 1000u);
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (util::GetMicroSeconds() >= timeout) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if ((reg::Read(g_dsi_regs + sizeof(u32) * DSI_HOST_CONTROL) & DSI_HOST_CONTROL_IMM_BTA) == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EnableBacklightForVendor2050ForAula(int brightness) {
|
||||||
|
/* Enable FRAME_END_INT */
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_CMD_INT_ENABLE, 2);
|
||||||
|
|
||||||
|
/* Configure DSI_LINE_TYPE as FOUR */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_VIDEO_MODE_CONTROL, 1);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_VIDEO_MODE_CONTROL, 9);
|
||||||
|
|
||||||
|
/* Set and wait for FRAME_END_INT */
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_CMD_INT_STATUS, 2);
|
||||||
|
while ((reg::Read(g_disp1_regs + sizeof(u32) * DC_CMD_INT_STATUS) & 2) != 0) { /* ... */ }
|
||||||
|
|
||||||
|
/* Configure display brightness. */
|
||||||
|
const u32 brightness_val = ((0x7FF * brightness) / 100);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x339);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, (brightness_val & 0x700) | ((brightness_val & 0xFF) << 16) | 0x51);
|
||||||
|
|
||||||
|
/* Set and wait for FRAME_END_INT */
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_CMD_INT_STATUS, 2);
|
||||||
|
while ((reg::Read(g_disp1_regs + sizeof(u32) * DC_CMD_INT_STATUS) & 2) != 0) { /* ... */ }
|
||||||
|
|
||||||
|
/* Set client sync point block reset. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_INCR_SYNCPT_CNTRL, 1);
|
||||||
|
util::WaitMicroSeconds(300'000ul);
|
||||||
|
|
||||||
|
/* Clear client sync point block resest. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_INCR_SYNCPT_CNTRL, 0);
|
||||||
|
util::WaitMicroSeconds(300'000ul);
|
||||||
|
|
||||||
|
/* Clear DSI_LINE_TYPE config. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_VIDEO_MODE_CONTROL, 0);
|
||||||
|
|
||||||
|
/* Disable FRAME_END_INT */
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_CMD_INT_ENABLE, 0);
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_CMD_INT_STATUS, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EnableBacklightForGeneric(int brightness) {
|
||||||
|
AMS_UNUSED(brightness);
|
||||||
|
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define DO_REGISTER_WRITES(base_address, writes) DoRegisterWrites(base_address, writes, util::size(writes))
|
||||||
|
#define DO_SOC_DEPENDENT_REGISTER_WRITES(base_address, writes) DoSocDependentRegisterWrites(base_address, writes##Erista, util::size(writes##Erista), writes##Mariko, util::size(writes##Mariko))
|
||||||
|
#define DO_SLEEP_OR_REGISTER_WRITES(base_address, writes) DoSleepOrRegisterWrites(base_address, writes, util::size(writes))
|
||||||
|
|
||||||
|
void InitializeFrameBuffer() {
|
||||||
|
if (g_frame_buffer != nullptr) {
|
||||||
|
std::memset(g_frame_buffer, 0, FrameBufferSize);
|
||||||
|
hw::FlushDataCache(g_frame_buffer, FrameBufferSize);
|
||||||
|
} else {
|
||||||
|
/* Clear the frame buffer. */
|
||||||
|
g_frame_buffer = /* TODO */reinterpret_cast<u32 *>(0xD0000000);
|
||||||
|
std::memset(g_frame_buffer, 0, FrameBufferSize);
|
||||||
|
hw::FlushDataCache(g_frame_buffer, FrameBufferSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[[maybe_unused]] void FinalizeFrameBuffer() {
|
||||||
|
/* We don't actually support finalizing the framebuffer, so do nothing here. */
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr const char *GetErrorDescription(u32 error_desc) {
|
||||||
|
switch (error_desc) {
|
||||||
|
case 0x100:
|
||||||
|
return "Instruction Abort";
|
||||||
|
case 0x101:
|
||||||
|
return "Data Abort";
|
||||||
|
case 0x102:
|
||||||
|
return "PC Misalignment";
|
||||||
|
case 0x103:
|
||||||
|
return "SP Misalignment";
|
||||||
|
case 0x104:
|
||||||
|
return "Trap";
|
||||||
|
case 0x106:
|
||||||
|
return "SError";
|
||||||
|
case 0x301:
|
||||||
|
return "Bad SVC";
|
||||||
|
case 0xF00:
|
||||||
|
return "Kernel Panic";
|
||||||
|
case 0xFFD:
|
||||||
|
return "Stack overflow";
|
||||||
|
case 0xFFE:
|
||||||
|
return "std::abort() called";
|
||||||
|
default:
|
||||||
|
return "Unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PrintSuggestedErrorFix(const ams::impl::FatalErrorContext *f_ctx) {
|
||||||
|
/* Try to recognize certain errors automatically, and suggest fixes for them. */
|
||||||
|
const char *suggestion = nullptr;
|
||||||
|
|
||||||
|
constexpr u64 ProgramIdAmsMitm = UINT64_C(0x010041544D530000);
|
||||||
|
constexpr u64 ProgramIdBoot = UINT64_C(0x0100000000000005);
|
||||||
|
if (f_ctx->error_desc == 0xFFE) {
|
||||||
|
if (f_ctx->program_id == ProgramIdAmsMitm) {
|
||||||
|
/* When a user has archive bits set improperly, attempting to create an automatic backup will fail */
|
||||||
|
/* to create the file path with error 0x202 */
|
||||||
|
if (f_ctx->gprs[0] == fs::ResultPathNotFound().GetValue()) {
|
||||||
|
/* When the archive bit error is occurring, it manifests as failure to create automatic backup. */
|
||||||
|
/* Thus, we can search the stack for the automatic backups path. */
|
||||||
|
const char * const automatic_backups_prefix = "automatic_backups/X" /* ..... */;
|
||||||
|
const int prefix_len = std::strlen(automatic_backups_prefix);
|
||||||
|
|
||||||
|
for (size_t i = 0; i + prefix_len < f_ctx->stack_dump_size; ++i) {
|
||||||
|
if (std::memcmp(&f_ctx->stack_dump[i], automatic_backups_prefix, prefix_len) == 0) {
|
||||||
|
suggestion = "The atmosphere directory may improperly have archive bits set.\n"
|
||||||
|
"Please try running an archive bit fixer tool (for example, the one in Hekate).\n";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (f_ctx->gprs[0] == fs::ResultExFatUnavailable().GetValue()) {
|
||||||
|
/* When a user installs non-exFAT firm but has an exFAT formatted SD card, this error will */
|
||||||
|
/* be returned on attempt to access the SD card. */
|
||||||
|
suggestion = "Your console has non-exFAT firmware installed, but your SD card\n"
|
||||||
|
"is formatted as exFAT. Format your SD card as FAT32, or manually\n"
|
||||||
|
"flash exFAT firmware to package2.\n";
|
||||||
|
}
|
||||||
|
} else if (f_ctx->program_id == ProgramIdBoot) {
|
||||||
|
/* 9.x -> 10.x updated the API for SvcQueryIoMapping. */
|
||||||
|
/* This can cause the kernel to reject incorrect-ABI calls by boot when a partial update is applied */
|
||||||
|
/* (older kernel in package2, for some reason). */
|
||||||
|
for (size_t i = 0; i < 8; ++i) {
|
||||||
|
if (f_ctx->gprs[i] == svc::ResultNotFound().GetValue()) {
|
||||||
|
suggestion = "A partial update may have been improperly performed.\n"
|
||||||
|
"To fix, try manually flashing latest package2 to MMC.\n"
|
||||||
|
"\n"
|
||||||
|
"For help doing this, seek support in the ReSwitched or\n"
|
||||||
|
"Nintendo Homebrew discord servers.\n";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (f_ctx->error_desc == 0xF00) { /* Kernel Panic */
|
||||||
|
suggestion = "Please contact SciresM#0524 on Discord, or create an issue on the Atmosphere\n"
|
||||||
|
"GitHub issue tracker. Thank you very much for helping to test mesosphere.\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If we found a suggestion, print it. */
|
||||||
|
if (suggestion != nullptr) {
|
||||||
|
Print("%s", suggestion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void InitializeDisplay() {
|
||||||
|
/* Setup the framebuffer. */
|
||||||
|
InitializeFrameBuffer();
|
||||||
|
|
||||||
|
/* Get the hardware type. */
|
||||||
|
const auto hw_type = fuse::GetHardwareType();
|
||||||
|
|
||||||
|
/* Turn on DSI/voltage rail. */
|
||||||
|
{
|
||||||
|
if (fuse::GetSocType() == fuse::SocType_Mariko) {
|
||||||
|
i2c::SendByte(i2c::Port_5, I2cAddressMax77620Pmic, 0x18, 0x3A);
|
||||||
|
i2c::SendByte(i2c::Port_5, I2cAddressMax77620Pmic, 0x18, 0x3A);
|
||||||
|
}
|
||||||
|
i2c::SendByte(i2c::Port_5, I2cAddressMax77620Pmic, 0x23, 0xD0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Enable MIPI CAL, DSI, DISP1, HOST1X, UART_FST_MIPI_CAL, DSIA LP clocks. */
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_RST_DEV_H_CLR, CLK_RST_REG_BITS_ENUM(RST_DEV_H_CLR_CLR_MIPI_CAL_RST, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(RST_DEV_H_CLR_CLR_DSI_RST, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_H_SET, CLK_RST_REG_BITS_ENUM(CLK_ENB_H_SET_SET_CLK_ENB_MIPI_CAL, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(CLK_ENB_H_SET_SET_CLK_ENB_DSI, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_RST_DEV_L_CLR, CLK_RST_REG_BITS_ENUM(RST_DEV_L_CLR_CLR_HOST1X_RST, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(RST_DEV_L_CLR_CLR_DISP1_RST, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_L_SET, CLK_RST_REG_BITS_ENUM(CLK_ENB_L_SET_SET_CLK_ENB_HOST1X, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(CLK_ENB_L_SET_SET_CLK_ENB_DISP1, ENABLE));
|
||||||
|
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_X_SET, CLK_RST_REG_BITS_ENUM(CLK_ENB_X_SET_SET_CLK_ENB_UART_FST_MIPI_CAL, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_SOURCE_UART_FST_MIPI_CAL, CLK_RST_REG_BITS_VALUE(CLK_SOURCE_UART_FST_MIPI_CAL_UART_FST_MIPI_CAL_CLK_DIVISOR, 10),
|
||||||
|
CLK_RST_REG_BITS_ENUM (CLK_SOURCE_UART_FST_MIPI_CAL_UART_FST_MIPI_CAL_CLK_SRC, PLLP_OUT3));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_W_SET, CLK_RST_REG_BITS_ENUM(CLK_ENB_W_SET_SET_CLK_ENB_DSIA_LP, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_SOURCE_DSIA_LP, CLK_RST_REG_BITS_VALUE(CLK_SOURCE_DSIA_LP_DSIA_LP_CLK_DIVISOR, 10),
|
||||||
|
CLK_RST_REG_BITS_ENUM (CLK_SOURCE_DSIA_LP_DSIA_LP_CLK_SRC, PLLP_OUT0));
|
||||||
|
|
||||||
|
/* Set IO_DPD_REQ to DPD_OFF. */
|
||||||
|
reg::ReadWrite(PMC + APBDEV_PMC_IO_DPD_REQ, PMC_REG_BITS_ENUM(IO_DPD_REQ_CODE, DPD_OFF));
|
||||||
|
reg::ReadWrite(PMC + APBDEV_PMC_IO_DPD2_REQ, PMC_REG_BITS_ENUM(IO_DPD2_REQ_CODE, DPD_OFF));
|
||||||
|
|
||||||
|
/* Configure LCD pinmux tristate + passthrough. */
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_NFC_EN, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_NFC_INT, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_BL_PWM, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_BL_EN, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_RST, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
|
||||||
|
if (hw_type == fuse::HardwareType_Aula) {
|
||||||
|
/* Configure LCD backlight. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_CNF_1, 0x4);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OE_1, 0x4);
|
||||||
|
} else {
|
||||||
|
/* Configure LCD power, VDD. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_CNF_0, 0x3);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OE_0, 0x3);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x1);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x2);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
/* Configure LCD backlight. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_CNF_1, 0x7);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OE_1, 0x7);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Configure display interface and display. */
|
||||||
|
reg::Write(g_mipi_cal_regs + MIPI_CAL_MIPI_BIAS_PAD_CFG2, 0);
|
||||||
|
if (fuse::GetSocType() == fuse::SocType_Mariko) {
|
||||||
|
reg::Write(g_mipi_cal_regs + MIPI_CAL_MIPI_BIAS_PAD_CFG0, 0);
|
||||||
|
reg::Write(g_apb_misc_regs + APB_MISC_GP_DSI_PAD_CONTROL, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Execute configs. */
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_clk_rst_regs, DisplayConfigPlld01);
|
||||||
|
DO_SLEEP_OR_REGISTER_WRITES(g_disp1_regs, DisplayConfigDc01);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init01);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init02);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init03);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init04);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init05);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsiPhyTiming);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init06);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsiPhyTiming);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init07);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
/* Enable backlight reset. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x4);
|
||||||
|
util::WaitMicroSeconds(60'000ul);
|
||||||
|
|
||||||
|
if (hw_type == fuse::HardwareType_Aula) {
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_BTA_TIMING, 0x40103);
|
||||||
|
} else {
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_BTA_TIMING, 0x50204);
|
||||||
|
}
|
||||||
|
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x337);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
WaitDsiTrigger();
|
||||||
|
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x406);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
WaitDsiTrigger();
|
||||||
|
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_TX_TRIG_HOST | DSI_HOST_CONTROL_IMM_BTA | DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC);
|
||||||
|
WaitDsiHostControl();
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
|
||||||
|
/* Parse LCD vendor. */
|
||||||
|
{
|
||||||
|
u32 host_response[3];
|
||||||
|
for (size_t i = 0; i < util::size(host_response); i++) {
|
||||||
|
host_response[i] = reg::Read(g_dsi_regs + sizeof(u32) * DSI_RD_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The last word from host response is:
|
||||||
|
Bits 0-7: FAB
|
||||||
|
Bits 8-15: REV
|
||||||
|
Bits 16-23: Minor REV
|
||||||
|
*/
|
||||||
|
u32 lcd_vendor;
|
||||||
|
if ((host_response[2] & 0xFF) == 0x10) {
|
||||||
|
lcd_vendor = 0;
|
||||||
|
} else {
|
||||||
|
lcd_vendor = (host_response[2] >> 8) & 0xFF00;
|
||||||
|
}
|
||||||
|
g_lcd_vendor = (lcd_vendor & 0xFFFFFF00) | (host_response[2] & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* LCD vendor specific configuration. */
|
||||||
|
switch (g_lcd_vendor) {
|
||||||
|
case 0x10: /* Japan Display Inc screens. */
|
||||||
|
DO_SLEEP_OR_REGISTER_WRITES(g_dsi_regs, DisplayConfigJdiSpecificInit01);
|
||||||
|
break;
|
||||||
|
case 0xF20: /* Innolux first revision screens. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x1105);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(180'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x439);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x9483FFB9);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x739);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x751548B1);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x143209);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x2905);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
break;
|
||||||
|
case 0xF30: /* AUO first revision screens. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x1105);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(180'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x439);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x9483FFB9);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x739);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x711148B1);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x143209);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x2905);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
break;
|
||||||
|
case 0x2050: /* Unknown (hardware type 5) screen. */
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x1105);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(180'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0xA015);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x205315);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x339);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x51);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(5'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x2905);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
break;
|
||||||
|
case 0x1020: /* Innolux second revision screen. */
|
||||||
|
case 0x1030: /* AUO second revision screen. */
|
||||||
|
case 0x1040: /* Unknown second revision screen. */
|
||||||
|
default:
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x1105);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
util::WaitMicroSeconds(120'000ul);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_WR_DATA, 0x2905);
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_TRIGGER, DSI_TRIGGER_HOST);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
util::WaitMicroSeconds(20'000ul);
|
||||||
|
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_clk_rst_regs, DisplayConfigPlld02);
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init08);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsiPhyTiming);
|
||||||
|
DO_SLEEP_OR_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init09);
|
||||||
|
|
||||||
|
reg::Write(g_disp1_regs + sizeof(u32) * DC_DISP_DISP_CLOCK_CONTROL, SHIFT_CLK_DIVIDER(4));
|
||||||
|
DO_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init10);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
/* Configure MIPI CAL. */
|
||||||
|
DO_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal01);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal02);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init11);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal03);
|
||||||
|
DO_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal04);
|
||||||
|
if (fuse::GetSocType() == fuse::SocType_Mariko) {
|
||||||
|
/* On Mariko the above configurations are executed twice, for some reason. */
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal02);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_dsi_regs, DisplayConfigDsi01Init11);
|
||||||
|
DO_SOC_DEPENDENT_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal03);
|
||||||
|
DO_REGISTER_WRITES(g_mipi_cal_regs, DisplayConfigMipiCal04);
|
||||||
|
}
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
/* Write DISP1, FrameBuffer config. */
|
||||||
|
DO_SLEEP_OR_REGISTER_WRITES(g_disp1_regs, DisplayConfigDc02);
|
||||||
|
DO_SLEEP_OR_REGISTER_WRITES(g_disp1_regs, DisplayConfigFrameBuffer);
|
||||||
|
if (g_lcd_vendor != 0x2050) {
|
||||||
|
util::WaitMicroSeconds(35'000ul);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void FinalizeDisplay() {
|
||||||
|
/* TODO: What other configuration is needed, if any? */
|
||||||
|
|
||||||
|
/* Configure LCD pinmux tristate + passthrough. */
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_NFC_EN, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_NFC_INT, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_BL_PWM, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_BL_EN, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
reg::ClearBits(g_apb_misc_regs + PINMUX_AUX_LCD_RST, reg::EncodeMask(PINMUX_REG_BITS_MASK(AUX_TRISTATE)));
|
||||||
|
|
||||||
|
if (fuse::GetHardwareType() == fuse::HardwareType_Aula) {
|
||||||
|
/* Configure LCD backlight. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_CNF_1, 0x4);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OE_1, 0x4);
|
||||||
|
} else {
|
||||||
|
/* Configure LCD power, VDD. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_CNF_0, 0x3);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OE_0, 0x3);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x1);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x2);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
|
||||||
|
/* Configure LCD backlight. */
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_CNF_1, 0x7);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OE_1, 0x7);
|
||||||
|
reg::SetBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Disable the LCD backlight. */
|
||||||
|
if (g_lcd_vendor == 0x2050) {
|
||||||
|
/* TODO: We're not sure display is alive. How to manage this? */
|
||||||
|
/* This is probably incorrect backlight disable for hw-type 5. */
|
||||||
|
reg::ClearBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x1);
|
||||||
|
} else {
|
||||||
|
reg::ClearBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Disable backlight RST/Voltage. */
|
||||||
|
reg::ClearBits(g_gpio_regs + GPIO_PORT6_OUT_1, 0x4);
|
||||||
|
if (g_lcd_vendor == 0x2050) {
|
||||||
|
util::WaitMicroSeconds(30'000ul);
|
||||||
|
} else {
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
reg::ClearBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x2);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
reg::ClearBits(g_gpio_regs + GPIO_PORT3_OUT_0, 0x1);
|
||||||
|
util::WaitMicroSeconds(10'000ul);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Cut clock to DSI. */
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_RST_DEV_H_SET, CLK_RST_REG_BITS_ENUM(RST_DEV_H_SET_SET_MIPI_CAL_RST, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(RST_DEV_H_SET_SET_DSI_RST, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_H_CLR, CLK_RST_REG_BITS_ENUM(CLK_ENB_H_CLR_CLR_CLK_ENB_MIPI_CAL, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(CLK_ENB_H_CLR_CLR_CLK_ENB_DSI, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_RST_DEV_L_SET, CLK_RST_REG_BITS_ENUM(RST_DEV_L_SET_SET_HOST1X_RST, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(RST_DEV_L_SET_SET_DISP1_RST, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_clk_rst_regs + CLK_RST_CONTROLLER_CLK_ENB_L_CLR, CLK_RST_REG_BITS_ENUM(CLK_ENB_L_CLR_CLR_CLK_ENB_HOST1X, ENABLE),
|
||||||
|
CLK_RST_REG_BITS_ENUM(CLK_ENB_L_CLR_CLR_CLK_ENB_DISP1, ENABLE));
|
||||||
|
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_PAD_CONTROL_0, (DSI_PAD_CONTROL_VS1_PULLDN_CLK | DSI_PAD_CONTROL_VS1_PULLDN(0xF) | DSI_PAD_CONTROL_VS1_PDIO_CLK | DSI_PAD_CONTROL_VS1_PDIO(0xF)));
|
||||||
|
reg::Write(g_dsi_regs + sizeof(u32) * DSI_POWER_CONTROL, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ShowDisplay(const ams::impl::FatalErrorContext *f_ctx, const Result save_result) {
|
||||||
|
/* Initialize the console. */
|
||||||
|
InitializeConsole(g_frame_buffer);
|
||||||
|
|
||||||
|
{
|
||||||
|
Print("%s\n", "A fatal error occurred when running Atmosph\xe8re.");
|
||||||
|
Print("Program ID: %016" PRIx64 "\n", f_ctx->program_id);
|
||||||
|
Print("Error Desc: %s (0x%" PRIx32 ")\n", GetErrorDescription(f_ctx->error_desc), f_ctx->error_desc);
|
||||||
|
Print("\n");
|
||||||
|
|
||||||
|
if (R_SUCCEEDED(save_result)) {
|
||||||
|
Print("Report saved to /atmosphere/fatal_errors/report_%016" PRIx64 ".bin", f_ctx->report_identifier);
|
||||||
|
} else {
|
||||||
|
Print("Failed to save report to the SD card! (%08" PRIx32 ")\n", save_result.GetValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintSuggestedErrorFix(f_ctx);
|
||||||
|
|
||||||
|
Print("\nPress POWER to reboot.\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure the device will see consistent data. */
|
||||||
|
hw::FlushDataCache(g_frame_buffer, FrameBufferSize);
|
||||||
|
|
||||||
|
/* Enable backlight. */
|
||||||
|
constexpr auto DisplayBrightness = 100;
|
||||||
|
if (g_lcd_vendor == 0x2050) {
|
||||||
|
EnableBacklightForVendor2050ForAula(DisplayBrightness);
|
||||||
|
} else {
|
||||||
|
EnableBacklightForGeneric(DisplayBrightness);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
29
fusee_cpp/program/source/fusee_display.hpp
Normal file
29
fusee_cpp/program/source/fusee_display.hpp
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 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/>.
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include <exosphere.hpp>
|
||||||
|
|
||||||
|
namespace ams::nxboot {
|
||||||
|
|
||||||
|
constexpr inline size_t FrameBufferHeight = 768;
|
||||||
|
constexpr inline size_t FrameBufferWidth = 1280;
|
||||||
|
constexpr inline size_t FrameBufferSize = FrameBufferHeight * FrameBufferWidth * sizeof(u32);
|
||||||
|
|
||||||
|
void InitializeDisplay();
|
||||||
|
void FinalizeDisplay();
|
||||||
|
void ShowDisplay(const ams::impl::FatalErrorContext *f_ctx, const Result save_result);
|
||||||
|
|
||||||
|
}
|
682
fusee_cpp/program/source/fusee_display_config.inc
Normal file
682
fusee_cpp/program/source/fusee_display_config.inc
Normal file
|
@ -0,0 +1,682 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct RegisterWrite {
|
||||||
|
u32 offset;
|
||||||
|
u32 value;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SleepOrRegisterWriteKind : u16 {
|
||||||
|
SleepOrRegisterWriteKind_Write = 0,
|
||||||
|
SleepOrRegisterWriteKind_Sleep = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SleepOrRegisterWrite {
|
||||||
|
SleepOrRegisterWriteKind kind;
|
||||||
|
u16 offset;
|
||||||
|
u32 value;
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigPlld01Erista[] = {
|
||||||
|
{CLK_RST_CONTROLLER_CLK_SOURCE_DISP1, 0x40000000},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_BASE, 0x4830A001},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC1, 0x00000020},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC, 0x002D0AAA},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigPlld01Mariko[] = {
|
||||||
|
{CLK_RST_CONTROLLER_CLK_SOURCE_DISP1, 0x40000000},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_BASE, 0x4830A001},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC1, 0x00000000},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC, 0x002DFC00},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigDc01[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_REG_ACT_CONTROL, 0x54},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DC_MCCIF_FIFOCTRL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_MEM_HIGH_PRIORITY, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_MEM_HIGH_PRIORITY_TIMER, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_POWER_CONTROL, PW0_ENABLE | PW1_ENABLE | PW2_ENABLE | PW3_ENABLE | PW4_ENABLE | PM0_ENABLE | PM1_ENABLE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_GENERAL_INCR_SYNCPT_CNTRL, SYNCPT_CNTRL_NO_STALL},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_CONT_SYNCPT_VSYNC, SYNCPT_VSYNC_ENABLE | 0x9}, // 9: SYNCPT
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE | WIN_B_UPDATE | WIN_C_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ | WIN_B_ACT_REQ | WIN_C_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_COLOR_CONTROL, BASE_COLOR_SIZE_888},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_INTERFACE_CONTROL, DISP_DATA_FORMAT_DF1P1C},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_PIN_OUTPUT_POLARITY(1), 0x1000000},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_PIN_OUTPUT_POLARITY(3), 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x4E4, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_CRC_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE | WIN_B_UPDATE | WIN_C_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ | WIN_B_ACT_REQ | WIN_C_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND_OPTION0, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE | WIN_B_UPDATE | WIN_C_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ | WIN_B_ACT_REQ | WIN_C_ACT_REQ}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init01[] = {
|
||||||
|
{sizeof(u32) * DSI_WR_DATA, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INT_ENABLE, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INT_STATUS, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INT_MASK, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_0, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_1, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_2, 0x0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_3, 0x0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init02Erista[] = {
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_15, 0x0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init02Mariko[] = {
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_DATA_15_MARIKO, 0x0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init03[] = {
|
||||||
|
{sizeof(u32) * DSI_DCS_CMDS, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_0_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_1_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_2_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_3_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_4_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_5_LO, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_0_HI, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_1_HI, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_2_HI, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_3_HI, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_4_HI, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_SEQ_5_HI, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init04Erista[] = {
|
||||||
|
/* No register writes. */
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init04Mariko[] = {
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_2, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_3, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_4, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_5_MARIKO, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_6_MARIKO, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_7_MARIKO, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init05[] = {
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_CD, 0},
|
||||||
|
{sizeof(u32) * DSI_SOL_DELAY, 0x18},
|
||||||
|
{sizeof(u32) * DSI_MAX_THRESHOLD, 0x1E0},
|
||||||
|
{sizeof(u32) * DSI_TRIGGER, 0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_CONTROL, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_LEN_0_1, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_LEN_2_3, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_LEN_4_5, 0},
|
||||||
|
{sizeof(u32) * DSI_PKT_LEN_6_7, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init06[] = {
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_1, 0x40A0E05},
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_2, 0x30109},
|
||||||
|
{sizeof(u32) * DSI_BTA_TIMING, 0x190A14},
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_0, DSI_TIMEOUT_LRX(0x2000) | DSI_TIMEOUT_HTX(0xFFFF)},
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_1, DSI_TIMEOUT_PR(0x765) | DSI_TIMEOUT_TA(0x2000)},
|
||||||
|
{sizeof(u32) * DSI_TO_TALLY, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_0, DSI_PAD_CONTROL_VS1_PULLDN(0) | DSI_PAD_CONTROL_VS1_PDIO(0)}, // Enable
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, 0},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init07[] = {
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_1, 0x40A0E05},
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_2, 0x30118},
|
||||||
|
{sizeof(u32) * DSI_BTA_TIMING, 0x190A14},
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_0, DSI_TIMEOUT_LRX(0x2000) | DSI_TIMEOUT_HTX(0xFFFF)},
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_1, DSI_TIMEOUT_PR(0x1343) | DSI_TIMEOUT_TA(0x2000)},
|
||||||
|
{sizeof(u32) * DSI_TO_TALLY, 0},
|
||||||
|
{sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_CRC_RESET | DSI_HOST_CONTROL_TX_TRIG_HOST | DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC},
|
||||||
|
{sizeof(u32) * DSI_CONTROL, DSI_CONTROL_LANES(3) | DSI_CONTROL_HOST_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_MAX_THRESHOLD, 0x40},
|
||||||
|
{sizeof(u32) * DSI_TRIGGER, 0},
|
||||||
|
{sizeof(u32) * DSI_TX_CRC, 0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_CONTROL, 0}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsiPhyTimingErista[] = {
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_0, 0x6070601},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsiPhyTimingMariko[] = {
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_0, 0x6070603},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigJdiSpecificInit01[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x9483FFB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xBD15},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x1939},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAAAAAD8},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAAAAAEB},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAEBAAAA},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAAAAAAA},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAAAAAEB},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAAEBAAAA},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xAA},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x1BD15},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2739},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFD8},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2BD15},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xF39},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFD8},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xFFFFFF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xBD15},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x6D915},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x1105},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Sleep, 180, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2905},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigPlld02Erista[] = {
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_BASE, 0x4810c001},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC1, 0x00000020},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC, 0x002D0AAA},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigPlld02Mariko[] = {
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_BASE, 0x4810c001},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC1, 0x00000000},
|
||||||
|
{CLK_RST_CONTROLLER_PLLD_MISC, 0x002DFC00},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init08[] = {
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigDsi01Init09[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PHY_TIMING_1, 0x40A0E05},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PHY_TIMING_2, 0x30172},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_BTA_TIMING, 0x190A14},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TIMEOUT_0, DSI_TIMEOUT_LRX(0x2000) | DSI_TIMEOUT_HTX(0xA40)},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TIMEOUT_1, DSI_TIMEOUT_PR(0x5A2F) | DSI_TIMEOUT_TA(0x2000)},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TO_TALLY, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_0_LO, 0x40000208},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_2_LO, 0x40000308},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_4_LO, 0x40000308},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_1_LO, 0x40000308},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_3_LO, 0x3F3B2B08},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_3_HI, 0x2CC},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_5_LO, 0x3F3B2B08},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_SEQ_5_HI, 0x2CC},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_LEN_0_1, 0xCE0000},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_LEN_2_3, 0x87001A2},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_LEN_4_5, 0x190},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_PKT_LEN_6_7, 0x190},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_HOST_CONTROL, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init10[] = {
|
||||||
|
{sizeof(u32) * DSI_TRIGGER, 0},
|
||||||
|
{sizeof(u32) * DSI_CONTROL, 0},
|
||||||
|
{sizeof(u32) * DSI_SOL_DELAY, 6},
|
||||||
|
{sizeof(u32) * DSI_MAX_THRESHOLD, 0x1E0},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_CONTROL, DSI_CONTROL_HS_CLK_CTRL | DSI_CONTROL_FORMAT(3) | DSI_CONTROL_LANES(3) | DSI_CONTROL_VIDEO_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_HS | DSI_HOST_CONTROL_FIFO_SEL | DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC},
|
||||||
|
{sizeof(u32) * DSI_CONTROL, DSI_CONTROL_HS_CLK_CTRL | DSI_CONTROL_FORMAT(3) | DSI_CONTROL_LANES(3) | DSI_CONTROL_VIDEO_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC},
|
||||||
|
{sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_HS | DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init11Erista[] = {
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_2, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_3, DSI_PAD_PREEMP_PD_CLK(0x3) | DSI_PAD_PREEMP_PU_CLK(0x3) | DSI_PAD_PREEMP_PD(0x03) | DSI_PAD_PREEMP_PU(0x3)},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_4, 0}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Init11Mariko[] = {
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_2, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_3, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_4, 0x77777},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_5_MARIKO, 0x77777},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_6_MARIKO, DSI_PAD_PREEMP_PD_CLK(0x1) | DSI_PAD_PREEMP_PU_CLK(0x1) | DSI_PAD_PREEMP_PD(0x01) | DSI_PAD_PREEMP_PU(0x1)},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_7_MARIKO, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal01[] = {
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG2, 0},
|
||||||
|
{MIPI_CAL_CIL_MIPI_CAL_STATUS, 0xF3F10000},
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG0, 1},
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG2, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal02Erista[] = {
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG2, 0x10010},
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG1, 0x300},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal02Mariko[] = {
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG2, 0x10010},
|
||||||
|
{MIPI_CAL_MIPI_BIAS_PAD_CFG1, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal03Erista[] = {
|
||||||
|
{MIPI_CAL_DSIA_MIPI_CAL_CONFIG, 0x200200},
|
||||||
|
{MIPI_CAL_DSIB_MIPI_CAL_CONFIG, 0x200200},
|
||||||
|
{MIPI_CAL_DSIA_MIPI_CAL_CONFIG_2, 0x200002},
|
||||||
|
{MIPI_CAL_DSIB_MIPI_CAL_CONFIG_2, 0x200002},
|
||||||
|
{MIPI_CAL_CILA_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_CILB_MIPI_CAL_CONFIG, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal03Mariko[] = {
|
||||||
|
{MIPI_CAL_DSIA_MIPI_CAL_CONFIG, 0x200006},
|
||||||
|
{MIPI_CAL_DSIB_MIPI_CAL_CONFIG, 0x200006},
|
||||||
|
{MIPI_CAL_DSIA_MIPI_CAL_CONFIG_2, 0x260000},
|
||||||
|
{MIPI_CAL_DSIB_MIPI_CAL_CONFIG_2, 0x260000},
|
||||||
|
{MIPI_CAL_CILA_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_CILB_MIPI_CAL_CONFIG, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigMipiCal04[] = {
|
||||||
|
{MIPI_CAL_CILC_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_CILD_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_CILE_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_CILF_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_DSIC_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_DSID_MIPI_CAL_CONFIG, 0},
|
||||||
|
{MIPI_CAL_DSIB_MIPI_CAL_CONFIG_2, 0},
|
||||||
|
{MIPI_CAL_DSIC_MIPI_CAL_CONFIG_2, 0},
|
||||||
|
{MIPI_CAL_DSID_MIPI_CAL_CONFIG_2, 0},
|
||||||
|
{MIPI_CAL_MIPI_CAL_CTRL, 0x2A000001},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigDc02[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DV_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
/* Setup default YUV colorspace conversion coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_YOF, 0xF0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KYRGB, 0x12A},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUR, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVR, 0x198},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUG, 0x39B},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVG, 0x32F},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KUB, 0x204},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_CSC_KVB, 0},
|
||||||
|
/* End of color coefficients */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_COLOR_CONTROL, BASE_COLOR_SIZE_888},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_INTERFACE_CONTROL, DISP_DATA_FORMAT_DF1P1C},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_PIN_OUTPUT_POLARITY(1), 0x1000000},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_PIN_OUTPUT_POLARITY(3), 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x4E4, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_CRC_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE | WIN_B_UPDATE | WIN_C_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ | WIN_B_ACT_REQ | WIN_C_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, 0x716, 0x10000FF},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND_OPTION0, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE | WIN_B_UPDATE | WIN_C_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ | WIN_B_ACT_REQ | WIN_C_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
/* Set Display timings */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_TIMING_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_REF_TO_SYNC, (1 << 16)}, // h_ref_to_sync = 0, v_ref_to_sync = 1.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_SYNC_WIDTH, 0x10048},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_BACK_PORCH, 0x90048},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_ACTIVE, 0x50002D0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_FRONT_PORCH, 0xA0088}, // Sources say that this should be above the DC_DISP_ACTIVE cmd.
|
||||||
|
/* End of Display timings */
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_SHIFT_CLOCK_OPTIONS, SC1_H_QUALIFIER_NONE | SC0_H_QUALIFIER_NONE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_COM_PIN_OUTPUT_ENABLE(1), 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DATA_ENABLE_OPTIONS, DE_SELECT_ACTIVE | DE_CONTROL_NORMAL},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_INTERFACE_CONTROL, DISP_DATA_FORMAT_DF1P1C},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_CLOCK_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND_OPTION0, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND, DISP_CTRL_MODE_C_DISPLAY},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, READ_MUX | WRITE_MUX},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_FRONT_PORCH, 0xA0088},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_GENERAL_INCR_SYNCPT, 0x301},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_GENERAL_INCR_SYNCPT, 0x301},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_CLOCK_CONTROL, PIXEL_CLK_DIVIDER_PCD1 | SHIFT_CLK_DIVIDER(4)},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_COLOR_CONTROL, BASE_COLOR_SIZE_888},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND_OPTION0, 0}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr u32 DisplayConfigFrameBufferAddress = 0xD0000000;
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigFrameBuffer[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_C_SELECT}, //Enable window C.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_B_SELECT}, //Enable window B.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_WINDOW_HEADER, WINDOW_A_SELECT}, //Enable window A.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, DSI_ENABLE}, //DSI_ENABLE
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_COLOR_DEPTH, WIN_COLOR_DEPTH_B8G8R8A8}, //T_A8R8G8B8 //NX Default: T_A8B8G8R8, WIN_COLOR_DEPTH_R8G8B8A8
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_POSITION, 0}, //(0,0)
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_H_INITIAL_DDA, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_V_INITIAL_DDA, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_PRESCALED_SIZE, V_PRESCALED_SIZE(1280) | H_PRESCALED_SIZE(2880)}, //Pre-scaled size: 1280x2880 bytes.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_DDA_INC, V_DDA_INC(0x1000) | H_DDA_INC(0x1000)},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_SIZE, V_SIZE(1280) | H_SIZE(720)}, //Window size: 1280 vertical lines x 720 horizontal pixels.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_LINE_STRIDE, 0x6000C00}, //768*2x768*4 (= 0x600 x 0xC00) bytes, see TRM for alignment requirements.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_BUFFER_CONTROL, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WINBUF_SURFACE_KIND, 0}, //Regular surface.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WINBUF_START_ADDR, DisplayConfigFrameBufferAddress}, //Framebuffer address.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WINBUF_ADDR_H_OFFSET, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WINBUF_ADDR_V_OFFSET, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, DSI_ENABLE}, //DSI_ENABLE
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, DSI_ENABLE}, //DSI_ENABLE
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, 0},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_DISP_DISP_WIN_OPTIONS, DSI_ENABLE}, //DSI_ENABLE
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_WIN_WIN_OPTIONS, WIN_ENABLE}, //Enable window AD.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_DISPLAY_COMMAND, DISP_CTRL_MODE_C_DISPLAY}, //DISPLAY_CTRL_MODE: continuous display.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_UPDATE | WIN_A_UPDATE}, //General update; window A update.
|
||||||
|
{SleepOrRegisterWriteKind_Write, DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ | WIN_A_ACT_REQ} //General activation request; window A activation request.
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDc01Fini01[] = {
|
||||||
|
{sizeof(u32) * DC_DISP_FRONT_PORCH, 0xA0088},
|
||||||
|
{sizeof(u32) * DC_CMD_INT_MASK, 0},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_ACCESS, 0},
|
||||||
|
{sizeof(u32) * DC_CMD_INT_ENABLE, 0},
|
||||||
|
{sizeof(u32) * DC_CMD_CONT_SYNCPT_VSYNC, 0},
|
||||||
|
{sizeof(u32) * DC_CMD_DISPLAY_COMMAND, DISP_CTRL_MODE_STOP},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{sizeof(u32) * DC_CMD_GENERAL_INCR_SYNCPT, 0x301},
|
||||||
|
{sizeof(u32) * DC_CMD_GENERAL_INCR_SYNCPT, 0x301},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_CONTROL, GENERAL_UPDATE},
|
||||||
|
{sizeof(u32) * DC_CMD_STATE_CONTROL, GENERAL_ACT_REQ},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Fini01[] = {
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, 0},
|
||||||
|
{sizeof(u32) * DSI_PAD_CONTROL_1, 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const RegisterWrite DisplayConfigDsi01Fini02[] = {
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_1, 0x40A0E05},
|
||||||
|
{sizeof(u32) * DSI_PHY_TIMING_2, 0x30118},
|
||||||
|
{sizeof(u32) * DSI_BTA_TIMING, 0x190A14},
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_0, DSI_TIMEOUT_LRX(0x2000) | DSI_TIMEOUT_HTX(0xFFFF) },
|
||||||
|
{sizeof(u32) * DSI_TIMEOUT_1, DSI_TIMEOUT_PR(0x1343) | DSI_TIMEOUT_TA(0x2000)},
|
||||||
|
{sizeof(u32) * DSI_TO_TALLY, 0},
|
||||||
|
{sizeof(u32) * DSI_HOST_CONTROL, DSI_HOST_CONTROL_CRC_RESET | DSI_HOST_CONTROL_TX_TRIG_HOST | DSI_HOST_CONTROL_CS | DSI_HOST_CONTROL_ECC},
|
||||||
|
{sizeof(u32) * DSI_CONTROL, DSI_CONTROL_LANES(3) | DSI_CONTROL_HOST_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_POWER_CONTROL, DSI_POWER_CONTROL_ENABLE},
|
||||||
|
{sizeof(u32) * DSI_MAX_THRESHOLD, 0x40},
|
||||||
|
{sizeof(u32) * DSI_TRIGGER, 0},
|
||||||
|
{sizeof(u32) * DSI_TX_CRC, 0},
|
||||||
|
{sizeof(u32) * DSI_INIT_SEQ_CONTROL, 0}
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigJdiSpecificFini01[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x9483FFB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2139},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x191919D5},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xB39},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x4F0F41B1},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xF179A433},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2D81},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr const SleepOrRegisterWrite DisplayConfigAuoRev1SpecificFini01[] = {
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x9483FFB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2C39},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x191919D5},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x2C39},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x191919D6},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x19191919},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xB39},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x711148B1},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x71143209},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x114D31},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0x439},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_WR_DATA, 0xB9},
|
||||||
|
{SleepOrRegisterWriteKind_Write, DSI_TRIGGER, DSI_TRIGGER_HOST},
|
||||||
|
{SleepOrRegisterWriteKind_Sleep, 5, 0},
|
||||||
|
};
|
4655
fusee_cpp/program/source/fusee_font.inc
Normal file
4655
fusee_cpp/program/source/fusee_font.inc
Normal file
File diff suppressed because it is too large
Load diff
|
@ -14,6 +14,7 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
#include <exosphere.hpp>
|
#include <exosphere.hpp>
|
||||||
|
#include "fusee_display.hpp"
|
||||||
#include "fusee_secure_initialize.hpp"
|
#include "fusee_secure_initialize.hpp"
|
||||||
#include "fusee_sdram.hpp"
|
#include "fusee_sdram.hpp"
|
||||||
#include "fusee_sd_card.hpp"
|
#include "fusee_sd_card.hpp"
|
||||||
|
@ -33,6 +34,13 @@ namespace ams::nxboot {
|
||||||
/* Initialize SD card. */
|
/* Initialize SD card. */
|
||||||
Result result = InitializeSdCard();
|
Result result = InitializeSdCard();
|
||||||
|
|
||||||
|
/* DEBUG: Fatal error context */
|
||||||
|
{
|
||||||
|
const ams::impl::FatalErrorContext *f_ctx = reinterpret_cast<const ams::impl::FatalErrorContext *>(0x4003E000);
|
||||||
|
InitializeDisplay();
|
||||||
|
ShowDisplay(f_ctx, ResultSuccess());
|
||||||
|
}
|
||||||
|
|
||||||
/* DEBUG: Check SD card connection. */
|
/* DEBUG: Check SD card connection. */
|
||||||
{
|
{
|
||||||
*reinterpret_cast<volatile u32 *>(0x40038000) = 0xAAAAAAAA;
|
*reinterpret_cast<volatile u32 *>(0x40038000) = 0xAAAAAAAA;
|
||||||
|
@ -45,7 +53,7 @@ namespace ams::nxboot {
|
||||||
*reinterpret_cast<volatile u32 *>(0x40038010) = static_cast<u32>(bw);
|
*reinterpret_cast<volatile u32 *>(0x40038010) = static_cast<u32>(bw);
|
||||||
}
|
}
|
||||||
|
|
||||||
*reinterpret_cast<volatile u32 *>(0x7000E400) = 0x10;
|
//*reinterpret_cast<volatile u32 *>(0x7000E400) = 0x10;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO */
|
/* TODO */
|
||||||
|
|
128
fusee_cpp/program/source/fusee_print.cpp
Normal file
128
fusee_cpp/program/source/fusee_print.cpp
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 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 <exosphere.hpp>
|
||||||
|
#include "fusee_display.hpp"
|
||||||
|
#include "fusee_print.hpp"
|
||||||
|
|
||||||
|
namespace ams::nxboot {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
#include "fusee_font.inc"
|
||||||
|
|
||||||
|
constexpr inline const u32 TextColor = 0xFFA0A0A0;
|
||||||
|
|
||||||
|
constexpr inline const size_t ConsoleWidth = FrameBufferWidth / FontWidth;
|
||||||
|
constexpr inline const size_t ConsoleHeight = FrameBufferHeight / FontHeight;
|
||||||
|
|
||||||
|
constinit u32 *g_frame_buffer = nullptr;
|
||||||
|
constinit size_t g_col = 1;
|
||||||
|
constinit size_t g_row = 0;
|
||||||
|
|
||||||
|
void SetPixel(size_t x, size_t y, u32 color) {
|
||||||
|
g_frame_buffer[(FrameBufferWidth - x) * FrameBufferHeight + y] = color;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PutCarriageReturn() {
|
||||||
|
g_col = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PutNewLine() {
|
||||||
|
g_col = 1;
|
||||||
|
++g_row;
|
||||||
|
|
||||||
|
/* TODO: Support scrolling? */
|
||||||
|
}
|
||||||
|
|
||||||
|
void PutCharImpl(const char c) {
|
||||||
|
/* Get the character data for the font. */
|
||||||
|
const u8 * cdata = FontData + c * (FontHeight * util::DivideUp(FontWidth, BITSIZEOF(u8)));
|
||||||
|
|
||||||
|
/* Determine where to start drawing. */
|
||||||
|
const size_t x = g_col * FontWidth;
|
||||||
|
const size_t y = g_row * FontHeight;
|
||||||
|
|
||||||
|
for (size_t cur_y = 0; cur_y < FontHeight; ++cur_y) {
|
||||||
|
size_t cur_x = 0;
|
||||||
|
int wbits = FontWidth;
|
||||||
|
while (wbits > 0) {
|
||||||
|
const auto bits = *(cdata++);
|
||||||
|
|
||||||
|
SetPixel(x + cur_x + 0, y + cur_y, FontDrawTable[(bits >> 4) & 0xF][0] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 1, y + cur_y, FontDrawTable[(bits >> 4) & 0xF][1] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 2, y + cur_y, FontDrawTable[(bits >> 4) & 0xF][2] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 3, y + cur_y, FontDrawTable[(bits >> 4) & 0xF][3] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 4, y + cur_y, FontDrawTable[(bits >> 0) & 0xF][0] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 5, y + cur_y, FontDrawTable[(bits >> 0) & 0xF][1] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 6, y + cur_y, FontDrawTable[(bits >> 0) & 0xF][2] & TextColor);
|
||||||
|
SetPixel(x + cur_x + 7, y + cur_y, FontDrawTable[(bits >> 0) & 0xF][3] & TextColor);
|
||||||
|
|
||||||
|
cur_x += BITSIZEOF(u8);
|
||||||
|
wbits -= BITSIZEOF(u8);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PutChar(const char c) {
|
||||||
|
switch (c) {
|
||||||
|
case '\r':
|
||||||
|
PutCarriageReturn();
|
||||||
|
break;
|
||||||
|
case '\n':
|
||||||
|
PutNewLine();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
PutCharImpl(c);
|
||||||
|
if ((++g_col) >= ConsoleWidth) {
|
||||||
|
PutNewLine();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void InitializeConsole(u32 *frame_buffer) {
|
||||||
|
/* Setup the console variables. */
|
||||||
|
g_frame_buffer = frame_buffer;
|
||||||
|
g_col = 1;
|
||||||
|
g_row = 0;
|
||||||
|
|
||||||
|
/* Clear the console. */
|
||||||
|
/* TODO: Assume it's already cleared? */
|
||||||
|
std::memset(g_frame_buffer, 0, FrameBufferSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Print(const char *fmt, ...) {
|
||||||
|
/* Generate the string. */
|
||||||
|
char log_str[1_KB];
|
||||||
|
{
|
||||||
|
std::va_list vl;
|
||||||
|
va_start(vl, fmt);
|
||||||
|
util::TVSNPrintf(log_str, sizeof(log_str), fmt, vl);
|
||||||
|
va_end(vl);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Print each character. */
|
||||||
|
const size_t len = std::strlen(log_str);
|
||||||
|
for (size_t i = 0; i < len; ++i) {
|
||||||
|
PutChar(log_str[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Flush the console. */
|
||||||
|
hw::FlushDataCache(g_frame_buffer, FrameBufferSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
24
fusee_cpp/program/source/fusee_print.hpp
Normal file
24
fusee_cpp/program/source/fusee_print.hpp
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018-2020 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/>.
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include <exosphere.hpp>
|
||||||
|
|
||||||
|
namespace ams::nxboot {
|
||||||
|
|
||||||
|
void InitializeConsole(u32 *frame_buffer);
|
||||||
|
void Print(const char *fmt, ...) __attribute__((format(printf, 1, 2)));
|
||||||
|
|
||||||
|
}
|
|
@ -86,13 +86,11 @@ namespace ams::nxboot {
|
||||||
}
|
}
|
||||||
|
|
||||||
Result ReadSdCard(void *dst, size_t size, size_t sector_index, size_t sector_count) {
|
Result ReadSdCard(void *dst, size_t size, size_t sector_index, size_t sector_count) {
|
||||||
/* TODO */
|
return sdmmc::Read(dst, size, SdCardPort, sector_index, sector_count);
|
||||||
return ResultSuccess();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Result WriteSdCard(size_t sector_index, size_t sector_count, const void *src, size_t size) {
|
Result WriteSdCard(size_t sector_index, size_t sector_count, const void *src, size_t size) {
|
||||||
/* TODO */
|
return sdmmc::Write(SdCardPort, sector_index, sector_count, src, size);
|
||||||
return ResultSuccess();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -42,7 +42,8 @@ __aeabi_ldivmod:
|
||||||
mov r0, sp
|
mov r0, sp
|
||||||
bl __l_divmod
|
bl __l_divmod
|
||||||
pop {r0-r3}
|
pop {r0-r3}
|
||||||
pop {ip, pc}
|
pop {ip, lr}
|
||||||
|
bx lr
|
||||||
|
|
||||||
.type __aeabi_ldivmod, %function
|
.type __aeabi_ldivmod, %function
|
||||||
.size __aeabi_ldivmod, .-__aeabi_ldivmod
|
.size __aeabi_ldivmod, .-__aeabi_ldivmod
|
||||||
|
@ -61,7 +62,8 @@ __aeabi_uldivmod :
|
||||||
mov r0, sp
|
mov r0, sp
|
||||||
bl __ul_divmod
|
bl __ul_divmod
|
||||||
pop {r0-r3}
|
pop {r0-r3}
|
||||||
pop {ip, pc}
|
pop {ip, lr}
|
||||||
|
bx lr
|
||||||
|
|
||||||
.type __aeabi_uldivmod, %function
|
.type __aeabi_uldivmod, %function
|
||||||
.size __aeabi_uldivmod, .-__aeabi_uldivmod
|
.size __aeabi_uldivmod, .-__aeabi_uldivmod
|
||||||
|
|
Loading…
Reference in a new issue