mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2025-05-24 19:54:22 -04:00
exo2: rename exosphere2 -> exosphere
This commit is contained in:
parent
282f8f6612
commit
42f1a3bf60
136 changed files with 15 additions and 15 deletions
182
exosphere/program/sc7fw/source/sc7fw_dram.cpp
Normal file
182
exosphere/program/sc7fw/source/sc7fw_dram.cpp
Normal file
|
@ -0,0 +1,182 @@
|
|||
/*
|
||||
* 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 "sc7fw_util.hpp"
|
||||
#include "sc7fw_dram.hpp"
|
||||
|
||||
namespace ams::sc7fw {
|
||||
|
||||
namespace {
|
||||
|
||||
constexpr inline const uintptr_t PMC = secmon::MemoryRegionPhysicalDevicePmc.GetAddress();
|
||||
|
||||
void UpdateEmcTiming() {
|
||||
/* Enable timing update. */
|
||||
reg::Write(EMC_ADDRESS(EMC_TIMING_CONTROL), EMC_REG_BITS_ENUM(TIMING_CONTROL_TIMING_UPDATE, ENABLED));
|
||||
|
||||
/* Wait for the timing update to complete. */
|
||||
while (!reg::HasValue(EMC_ADDRESS(EMC_EMC_STATUS), EMC_REG_BITS_ENUM(EMC_STATUS_TIMING_UPDATE_STALLED, DONE))) {
|
||||
/* ... */
|
||||
}
|
||||
}
|
||||
|
||||
void RequestAllPadsPowerDown(uintptr_t addr, uintptr_t expected) {
|
||||
constexpr u32 DpdAllRequestValue = reg::Encode(PMC_REG_BITS_ENUM(IO_DPD_REQ_CODE, DPD_ON)) | 0x0FFFFFFF;
|
||||
const auto RequestAddress = addr;
|
||||
const auto StatusAddress = addr + 4;
|
||||
|
||||
/* Request all pads enter power down. */
|
||||
reg::Write(PMC + RequestAddress, DpdAllRequestValue);
|
||||
|
||||
/* Wait until the status reflects our expectation (and all pads are shut down). */
|
||||
while (reg::Read(PMC + StatusAddress) != expected) { /* ... */ }
|
||||
|
||||
/* Wait a little while to allow the power down status to propagate. */
|
||||
SpinLoop(0x20);
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
void SaveEmcFsp() {
|
||||
/* We require that the RAM is LPDDR4. */
|
||||
AMS_ABORT_UNLESS(reg::HasValue(EMC_ADDRESS(EMC_FBIO_CFG5), EMC_REG_BITS_ENUM(FBIO_CFG5_DRAM_TYPE, LPDDR4)));
|
||||
|
||||
/* Read the frequency set points from MRW3. */
|
||||
constexpr u32 FspShift = 6;
|
||||
constexpr u32 FspBits = 2;
|
||||
constexpr u32 FspMask = ((1u << FspBits) - 1) << FspShift;
|
||||
static_assert(FspMask == 0x000000C0);
|
||||
const u32 fsp = (reg::Read(EMC_ADDRESS(EMC_MRW3)) & FspMask) >> FspShift;
|
||||
|
||||
/* Write the fsp to PMC_SCRATCH18, where it will be restored to MRW3 by brom. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_SCRATCH18, REG_BITS_VALUE(FspShift, FspBits, fsp));
|
||||
|
||||
/* Write the fsp twice to PMC_SCRATCH12, where it will be restored to MRW12 by brom. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_SCRATCH12, REG_BITS_VALUE(FspShift, FspBits, fsp), REG_BITS_VALUE(FspShift + 8, FspBits, fsp));
|
||||
|
||||
/* Write the fsp twice to PMC_SCRATCH13, where it will be restored to MRW13 by brom. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_SCRATCH13, REG_BITS_VALUE(FspShift, FspBits, fsp), REG_BITS_VALUE(FspShift + 8, FspBits, fsp));
|
||||
}
|
||||
|
||||
void EnableSdramSelfRefresh() {
|
||||
/* We require that the RAM is dual-channel. */
|
||||
AMS_ABORT_UNLESS(reg::HasValue(EMC_ADDRESS(EMC_FBIO_CFG7), EMC_REG_BITS_ENUM(FBIO_CFG7_CH1_ENABLE, ENABLE)));
|
||||
|
||||
/* Disable RAM's ability to dynamically self-refresh, and to opportunistically perform powerdown. */
|
||||
reg::Write(EMC_ADDRESS(EMC_CFG), EMC_REG_BITS_ENUM(CFG_DYN_SELF_REF, DISABLED),
|
||||
EMC_REG_BITS_ENUM(CFG_DRAM_ACPD, NO_POWERDOWN));
|
||||
|
||||
/* Update the EMC timing. */
|
||||
UpdateEmcTiming();
|
||||
|
||||
/* Wait five microseconds. */
|
||||
util::WaitMicroSeconds(5);
|
||||
|
||||
/* Disable ZQ calibration. */
|
||||
reg::Write(EMC_ADDRESS(EMC_ZCAL_INTERVAL), 0);
|
||||
|
||||
/* Disable automatic calibration. */
|
||||
reg::Write(EMC_ADDRESS(EMC_AUTO_CAL_CONFIG), EMC_REG_BITS_ENUM(AUTO_CAL_CONFIG_AUTO_CAL_MEASURE_STALL, ENABLE),
|
||||
EMC_REG_BITS_ENUM(AUTO_CAL_CONFIG_AUTO_CAL_UPDATE_STALL, ENABLE),
|
||||
EMC_REG_BITS_ENUM(AUTO_CAL_CONFIG_AUTO_CAL_START, DISABLE));
|
||||
|
||||
/* Get whether digital delay locked loops are enabled. */
|
||||
const bool has_dll = reg::HasValue(EMC_ADDRESS(EMC_CFG_DIG_DLL), EMC_REG_BITS_ENUM(CFG_DIG_DLL_CFG_DLL_EN, ENABLED));
|
||||
if (has_dll) {
|
||||
/* If they are, disable them. */
|
||||
reg::ReadWrite(EMC_ADDRESS(EMC_CFG_DIG_DLL), EMC_REG_BITS_ENUM(CFG_DIG_DLL_CFG_DLL_EN, DISABLED));
|
||||
}
|
||||
|
||||
/* Update the EMC timing. */
|
||||
UpdateEmcTiming();
|
||||
|
||||
/* If dll was enabled, wait until both EMC0 and EMC1 have dll disabled. */
|
||||
if (has_dll) {
|
||||
while (!reg::HasValue(EMC0_ADDRESS(EMC_CFG_DIG_DLL), EMC_REG_BITS_ENUM(CFG_DIG_DLL_CFG_DLL_EN, DISABLED))) { /* ... */ }
|
||||
while (!reg::HasValue(EMC1_ADDRESS(EMC_CFG_DIG_DLL), EMC_REG_BITS_ENUM(CFG_DIG_DLL_CFG_DLL_EN, DISABLED))) { /* ... */ }
|
||||
}
|
||||
|
||||
/* Stall all reads and writes. */
|
||||
reg::Write(EMC_ADDRESS(EMC_REQ_CTRL), EMC_REG_BITS_VALUE(REQ_CTRL_STALL_ALL_READS, 1),
|
||||
EMC_REG_BITS_VALUE(REQ_CTRL_STALL_ALL_WRITES, 1));
|
||||
|
||||
/* Wait until both EMC0 and EMC1 have no outstanding transactions. */
|
||||
while (!reg::HasValue(EMC0_ADDRESS(EMC_EMC_STATUS), EMC_REG_BITS_ENUM(EMC_STATUS_NO_OUTSTANDING_TRANSACTIONS, COMPLETED))) { /* ... */ }
|
||||
while (!reg::HasValue(EMC1_ADDRESS(EMC_EMC_STATUS), EMC_REG_BITS_ENUM(EMC_STATUS_NO_OUTSTANDING_TRANSACTIONS, COMPLETED))) { /* ... */ }
|
||||
|
||||
/* Enable self-refresh. */
|
||||
reg::Write(EMC_ADDRESS(EMC_SELF_REF), EMC_REG_BITS_ENUM(SELF_REF_SREF_DEV_SELECTN, BOTH),
|
||||
EMC_REG_BITS_ENUM(SELF_REF_SELF_REF_CMD, ENABLED));
|
||||
|
||||
/* Wait until both EMC and EMC1 are in self-refresh. */
|
||||
const auto desired = reg::HasValue(EMC_ADDRESS(EMC_ADR_CFG), EMC_REG_BITS_ENUM(ADR_CFG_EMEM_NUMDEV, N2)) ? EMC_REG_BITS_ENUM(EMC_STATUS_DRAM_IN_SELF_REFRESH, BOTH_ENABLED)
|
||||
: EMC_REG_BITS_ENUM(EMC_STATUS_DRAM_DEV0_IN_SELF_REFRESH, ENABLED);
|
||||
|
||||
/* NOTE: Nintendo's sc7 entry firmware has a bug here. */
|
||||
/* Instead of waiting for both EMCs to report self-refresh, they just read the EMC_STATUS for each EMC. */
|
||||
/* This is incorrect, per documentation. */
|
||||
while (!reg::HasValue(EMC0_ADDRESS(EMC_EMC_STATUS), desired)) { /* ... */ }
|
||||
while (!reg::HasValue(EMC1_ADDRESS(EMC_EMC_STATUS), desired)) { /* ... */ }
|
||||
}
|
||||
|
||||
void EnableEmcAllSegmentsRefresh() {
|
||||
constexpr int MR17_PASR_Segment = 17;
|
||||
|
||||
/* Write zeros to MR17_PASR_Segment to enable refresh for all segments for dev0. */
|
||||
reg::Write(EMC_ADDRESS(EMC_MRW), EMC_REG_BITS_ENUM (MRW_DEV_SELECTN, DEV0),
|
||||
EMC_REG_BITS_ENUM (MRW_CNT, EXT1),
|
||||
EMC_REG_BITS_VALUE(MRW_MA, MR17_PASR_Segment),
|
||||
EMC_REG_BITS_VALUE(MRW_OP, 0));
|
||||
|
||||
/* If dev1 exists, do the same for dev1. */
|
||||
if (reg::HasValue(EMC_ADDRESS(EMC_ADR_CFG), EMC_REG_BITS_ENUM(ADR_CFG_EMEM_NUMDEV, N2))) {
|
||||
reg::Write(EMC_ADDRESS(EMC_MRW), EMC_REG_BITS_ENUM (MRW_DEV_SELECTN, DEV1),
|
||||
EMC_REG_BITS_ENUM (MRW_CNT, EXT1),
|
||||
EMC_REG_BITS_VALUE(MRW_MA, MR17_PASR_Segment),
|
||||
EMC_REG_BITS_VALUE(MRW_OP, 0));
|
||||
}
|
||||
}
|
||||
|
||||
void EnableDdrDeepPowerDown() {
|
||||
/* Read and decode the parameters Nintendo stores in EMC_PMC_SCRATCH3. */
|
||||
const u32 scratch3 = reg::Read(EMC_ADDRESS(EMC_PMC_SCRATCH3));
|
||||
const bool weak_bias = (scratch3 & reg::EncodeMask(EMC_REG_BITS_MASK(PMC_SCRATCH3_WEAK_BIAS))) == reg::EncodeValue(EMC_REG_BITS_ENUM(PMC_SCRATCH3_WEAK_BIAS, ENABLED));
|
||||
const u32 ddr_cntrl = (scratch3 & reg::EncodeMask(EMC_REG_BITS_MASK(PMC_SCRATCH3_DDR_CNTRL)));
|
||||
|
||||
/* Write the decoded value to PMC_DDR_CNTRL. */
|
||||
reg::Write(PMC + APBDEV_PMC_DDR_CNTRL, ddr_cntrl);
|
||||
|
||||
/* If weak bias is enabled, set all VTT_E_WB bits in APBDEV_PMC_WEAK_BIAS. */
|
||||
if (weak_bias) {
|
||||
constexpr u32 WeakBiasVttEWbAll = 0x7FFF0000;
|
||||
reg::Write(PMC + APBDEV_PMC_WEAK_BIAS, WeakBiasVttEWbAll);
|
||||
}
|
||||
|
||||
/* Request that DPD3 pads power down. */
|
||||
constexpr u32 EristaDpd3Mask = 0x0FFFFFFF;
|
||||
constexpr u32 MarikoDpd3Mask = 0x0FFF9FFF;
|
||||
if (fuse::GetSocType() == fuse::SocType_Erista) {
|
||||
RequestAllPadsPowerDown(APBDEV_PMC_IO_DPD3_REQ, EristaDpd3Mask);
|
||||
} else {
|
||||
RequestAllPadsPowerDown(APBDEV_PMC_IO_DPD3_REQ, MarikoDpd3Mask);
|
||||
}
|
||||
|
||||
/* Request that DPD4 pads power down. */
|
||||
constexpr u32 Dpd4Mask = 0x0FFF1FFF;
|
||||
RequestAllPadsPowerDown(APBDEV_PMC_IO_DPD4_REQ, Dpd4Mask);
|
||||
}
|
||||
|
||||
}
|
26
exosphere/program/sc7fw/source/sc7fw_dram.hpp
Normal file
26
exosphere/program/sc7fw/source/sc7fw_dram.hpp
Normal file
|
@ -0,0 +1,26 @@
|
|||
/*
|
||||
* 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::sc7fw {
|
||||
|
||||
void SaveEmcFsp();
|
||||
void EnableSdramSelfRefresh();
|
||||
void EnableEmcAllSegmentsRefresh();
|
||||
void EnableDdrDeepPowerDown();
|
||||
|
||||
}
|
22
exosphere/program/sc7fw/source/sc7fw_exception_vectors.s
Normal file
22
exosphere/program/sc7fw/source/sc7fw_exception_vectors.s
Normal file
|
@ -0,0 +1,22 @@
|
|||
/*
|
||||
* 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/>.
|
||||
*/
|
||||
|
||||
.section .vectors, "ax", %progbits
|
||||
.align 3
|
||||
.global reset
|
||||
reset:
|
||||
b _start
|
||||
b _ZN3ams5sc7fw16ExceptionHandlerEv
|
116
exosphere/program/sc7fw/source/sc7fw_main.cpp
Normal file
116
exosphere/program/sc7fw/source/sc7fw_main.cpp
Normal file
|
@ -0,0 +1,116 @@
|
|||
/*
|
||||
* 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 "sc7fw_util.hpp"
|
||||
#include "sc7fw_dram.hpp"
|
||||
|
||||
namespace ams::sc7fw {
|
||||
|
||||
namespace {
|
||||
|
||||
constexpr inline const uintptr_t PMC = secmon::MemoryRegionPhysicalDevicePmc.GetAddress();
|
||||
|
||||
void DisableCrail() {
|
||||
/* Wait for CRAIL to be off. */
|
||||
while (!reg::HasValue(PMC + APBDEV_PMC_PWRGATE_STATUS, PMC_REG_BITS_ENUM(PWRGATE_STATUS_CRAIL, OFF))) { /* ... */ }
|
||||
|
||||
/* Set CRAIL to be clamped. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_SET_SW_CLAMP, PMC_REG_BITS_VALUE(SET_SW_CLAMP_CRAIL, 1));
|
||||
|
||||
/* Wait for CRAIL to be clamped. */
|
||||
while (!reg::HasValue(PMC + APBDEV_PMC_CLAMP_STATUS, PMC_REG_BITS_ENUM(CLAMP_STATUS_CRAIL, ENABLE))) { /* ... */ }
|
||||
|
||||
/* Spin loop for a short while, to allow time for the clamp to take effect. */
|
||||
sc7fw::SpinLoop(10);
|
||||
|
||||
/* Initialize i2c-5. */
|
||||
i2c::Initialize(i2c::Port_5);
|
||||
|
||||
/* Disable the voltage to CPU. */
|
||||
pmic::DisableVddCpu(fuse::GetRegulator());
|
||||
|
||||
/* Wait 700 microseconds to ensure voltage is disabled. */
|
||||
util::WaitMicroSeconds(700);
|
||||
}
|
||||
|
||||
void DisableAllInterrupts() {
|
||||
/* Disable all interrupts for bpmp in all interrupt controllers. */
|
||||
reg::Write(PRI_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
reg::Write(SEC_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
reg::Write(TRI_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
reg::Write(QUAD_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
reg::Write(PENTA_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
reg::Write(HEXA_ICTLR(ICTLR_COP_IER_CLR), ~0u);
|
||||
}
|
||||
|
||||
void EnterSc7() {
|
||||
/* Disable read buffering and write buffering in the BPMP cache. */
|
||||
reg::ReadWrite(AVP_CACHE_ADDRESS(AVP_CACHE_CONFIG), AVP_CACHE_REG_BITS_ENUM(DISABLE_WB, TRUE),
|
||||
AVP_CACHE_REG_BITS_ENUM(DISABLE_RB, TRUE));
|
||||
|
||||
/* Ensure the CPU Rail is turned off. */
|
||||
DisableCrail();
|
||||
|
||||
/* Disable all interrupts. */
|
||||
DisableAllInterrupts();
|
||||
|
||||
/* Save the EMC FSP */
|
||||
SaveEmcFsp();
|
||||
|
||||
/* Enable self-refresh for DRAM */
|
||||
EnableSdramSelfRefresh();
|
||||
|
||||
/* Enable refresh for all EMC devices. */
|
||||
EnableEmcAllSegmentsRefresh();
|
||||
|
||||
/* Enable deep power-down for ddr. */
|
||||
EnableDdrDeepPowerDown();
|
||||
|
||||
/* Enable pad sampling during deep sleep. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_DPD_SAMPLE, PMC_REG_BITS_ENUM(DPD_SAMPLE_ON, ENABLE));
|
||||
reg::Read(PMC + APBDEV_PMC_DPD_SAMPLE);
|
||||
|
||||
/* Wait a while for pad sampling to be enabled. */
|
||||
sc7fw::SpinLoop(0x128);
|
||||
|
||||
/* Enter deep sleep. */
|
||||
reg::ReadWrite(PMC + APBDEV_PMC_DPD_ENABLE, PMC_REG_BITS_ENUM(DPD_ENABLE_ON, ENABLE));
|
||||
|
||||
/* Wait forever until we're asleep. */
|
||||
while (true) { /* ... */ }
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void Main() {
|
||||
EnterSc7();
|
||||
}
|
||||
|
||||
NORETURN void ExceptionHandler() {
|
||||
/* Write enable to MAIN_RESET. */
|
||||
reg::Write(PMC + APBDEV_PMC_CNTRL, PMC_REG_BITS_ENUM(CNTRL_MAIN_RESET, ENABLE));
|
||||
while (true) { /* ... */ }
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
namespace ams::diag {
|
||||
|
||||
void AbortImpl() {
|
||||
sc7fw::ExceptionHandler();
|
||||
}
|
||||
|
||||
}
|
35
exosphere/program/sc7fw/source/sc7fw_start.s
Normal file
35
exosphere/program/sc7fw/source/sc7fw_start.s
Normal file
|
@ -0,0 +1,35 @@
|
|||
/*
|
||||
* 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/>.
|
||||
*/
|
||||
|
||||
.section .text.start, "ax", %progbits
|
||||
.align 3
|
||||
.global _start
|
||||
_start:
|
||||
/* Set CPSR_cf and CPSR_cf. */
|
||||
msr cpsr_f, #0xC0
|
||||
msr cpsr_cf, #0xD3
|
||||
|
||||
/* Set the stack pointer. */
|
||||
ldr sp, =0x40005000
|
||||
|
||||
/* Set our link register to the exception handler. */
|
||||
ldr lr, =_ZN3ams5sc7fw16ExceptionHandlerEv
|
||||
|
||||
/* Invoke main. */
|
||||
bl _ZN3ams5sc7fw4MainEv
|
||||
|
||||
/* Infinite loop. */
|
||||
1: b 1b
|
23
exosphere/program/sc7fw/source/sc7fw_util.hpp
Normal file
23
exosphere/program/sc7fw/source/sc7fw_util.hpp
Normal file
|
@ -0,0 +1,23 @@
|
|||
/*
|
||||
* 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::sc7fw {
|
||||
|
||||
void SpinLoop(unsigned int num);
|
||||
|
||||
}
|
30
exosphere/program/sc7fw/source/sc7fw_util_asm.s
Normal file
30
exosphere/program/sc7fw/source/sc7fw_util_asm.s
Normal file
|
@ -0,0 +1,30 @@
|
|||
/*
|
||||
* 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/>.
|
||||
*/
|
||||
|
||||
.section .text._ZN3ams5sc7fw8SpinLoopEj, "ax", %progbits
|
||||
.global _ZN3ams5sc7fw8SpinLoopEj
|
||||
.thumb_func
|
||||
.syntax unified
|
||||
_ZN3ams5sc7fw8SpinLoopEj:
|
||||
1:
|
||||
/* Subtract one from the count. */
|
||||
subs r0, r0, #1
|
||||
|
||||
/* If we aren't at zero, continue looping. */
|
||||
bgt 1b
|
||||
|
||||
/* Return. */
|
||||
bx lr
|
Loading…
Add table
Add a link
Reference in a new issue