/*
 * 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 .
 */
#include 
#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);
        AMS_ABORT_UNLESS(metadata->target_firmware <= ams::TargetFirmware_Max);
        /* 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(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();
    }
}