mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2024-12-16 17:46:04 +01:00
108 lines
3.7 KiB
C++
108 lines
3.7 KiB
C++
/*
|
|
* 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 "warmboot_bootrom_workaround.hpp"
|
|
#include "warmboot_clkrst.hpp"
|
|
#include "warmboot_cpu_cluster.hpp"
|
|
#include "warmboot_dram.hpp"
|
|
#include "warmboot_main.hpp"
|
|
#include "warmboot_misc.hpp"
|
|
#include "warmboot_secure_monitor.hpp"
|
|
|
|
namespace ams::warmboot {
|
|
|
|
namespace {
|
|
|
|
constexpr inline const uintptr_t PMC = secmon::MemoryRegionPhysicalDevicePmc.GetAddress();
|
|
constexpr inline const uintptr_t FLOW_CTLR = secmon::MemoryRegionPhysicalDeviceFlowController.GetAddress();
|
|
|
|
constexpr inline const uintptr_t ExpectedMetadataAddress = 0x40010244;
|
|
|
|
}
|
|
|
|
void Main(const Metadata *metadata) {
|
|
/* Ensure that we're running under vaguely sane conditions. */
|
|
AMS_ABORT_UNLESS(metadata->magic == Metadata::Magic);
|
|
|
|
/* Restrict the bpmp's access to dram. */
|
|
if (metadata->target_firmware >= TargetFirmware_4_0_0) {
|
|
RestrictBpmpAccessToMainMemory();
|
|
}
|
|
|
|
/* Configure rtck-daisychaining/jtag. */
|
|
ConfigureMiscSystemDebug();
|
|
|
|
/* NOTE: Here, Nintendo checks that the number of burnt anti-downgrade fuses is valid. */
|
|
|
|
/* NOTE: Here, Nintendo validates that APBDEV_PMC_SECURE_SCRATCH32 contains the correct magic number for the current warmboot firmware revision. */
|
|
|
|
/* Validate that we're executing at the correct address. */
|
|
AMS_ABORT_UNLESS(reinterpret_cast<uintptr_t>(metadata) == ExpectedMetadataAddress);
|
|
|
|
/* Validate that we're executing on the bpmp. */
|
|
AMS_ABORT_UNLESS(reg::Read(PG_UP(PG_UP_TAG)) == PG_UP_TAG_PID_COP);
|
|
|
|
/* Configure fuse bypass. */
|
|
fuse::ConfigureFuseBypass();
|
|
|
|
/* Configure system oscillators. */
|
|
ConfigureOscillators();
|
|
|
|
/* Restore DRAM configuration. */
|
|
RestoreRamSvop();
|
|
ConfigureEmcPmacroTraining();
|
|
|
|
/* If on Erista, work around the bootrom mbist issue. */
|
|
if (fuse::GetSocType() == fuse::SocType_Erista) {
|
|
ApplyMbistWorkaround();
|
|
}
|
|
|
|
/* Initialize the cpu cluster. */
|
|
InitializeCpuCluster();
|
|
|
|
/* Restore the secure monitor to tzram. */
|
|
RestoreSecureMonitorToTzram(metadata->target_firmware);
|
|
|
|
/* Power on the cpu. */
|
|
PowerOnCpu();
|
|
|
|
/* Halt ourselves. */
|
|
while (true) {
|
|
reg::Write(secmon::MemoryRegionPhysicalDeviceFlowController.GetAddress() + FLOW_CTLR_HALT_COP_EVENTS, FLOW_REG_BITS_ENUM(HALT_COP_EVENTS_MODE, FLOW_MODE_STOP),
|
|
FLOW_REG_BITS_ENUM(HALT_COP_EVENTS_JTAG, ENABLED));
|
|
}
|
|
}
|
|
|
|
NORETURN void ExceptionHandler() {
|
|
/* Write enable to MAIN_RESET. */
|
|
reg::Write(PMC + APBDEV_PMC_CNTRL, PMC_REG_BITS_ENUM(CNTRL_MAIN_RESET, ENABLE));
|
|
|
|
/* Wait forever until we're reset. */
|
|
AMS_INFINITE_LOOP();
|
|
}
|
|
|
|
}
|
|
|
|
namespace ams::diag {
|
|
|
|
void AbortImpl() {
|
|
warmboot::ExceptionHandler();
|
|
}
|
|
|
|
#include <exosphere/diag/diag_detailed_assertion_impl.inc>
|
|
|
|
}
|