/* * 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 . */ #if defined(ATMOSPHERE_IS_STRATOSPHERE) #include #else #include #endif namespace ams::dd { uintptr_t QueryIoMapping(dd::PhysicalAddress phys_addr, size_t size) { #if defined(ATMOSPHERE_IS_EXOSPHERE) #if defined(ATMOSPHERE_ARCH_ARM64) /* TODO: Secure Monitor translation? */ AMS_UNUSED(size); return static_cast(phys_addr); #elif defined(ATMOSPHERE_ARCH_ARM) /* TODO: BPMP translation? */ AMS_UNUSED(size); return static_cast(phys_addr); #else #error "Unknown architecture for ams::dd::QueryIoMapping (EXOSPHERE)!" #endif #elif defined(ATMOSPHERE_IS_MESOSPHERE) /* TODO: Kernel address translation? */ AMS_UNUSED(size); return static_cast(phys_addr); #elif defined(ATMOSPHERE_IS_STRATOSPHERE) svc::Address virt_addr = 0; const dd::PhysicalAddress aligned_addr = util::AlignDown(phys_addr, os::MemoryPageSize); const size_t offset = phys_addr - aligned_addr; const size_t aligned_size = size + offset; if (hos::GetVersion() >= hos::Version_10_0_0) { svc::Size region_size = 0; R_TRY_CATCH(svc::QueryIoMapping(&virt_addr, ®ion_size, aligned_addr, aligned_size)) { /* Official software handles this by returning 0. */ R_CATCH(svc::ResultNotFound) { return 0; } } R_END_TRY_CATCH_WITH_ABORT_UNLESS; AMS_ASSERT(region_size >= aligned_size); } else { R_TRY_CATCH(svc::LegacyQueryIoMapping(&virt_addr, aligned_addr, aligned_size)) { /* Official software handles this by returning 0. */ R_CATCH(svc::ResultNotFound) { return 0; } } R_END_TRY_CATCH_WITH_ABORT_UNLESS; } return static_cast(virt_addr) + offset; #else #error "Unknown execution context for ams::dd::QueryIoMapping!" #endif } u32 ReadIoRegister(dd::PhysicalAddress phys_addr) { #if defined(ATMOSPHERE_IS_EXOSPHERE) || defined(ATMOSPHERE_IS_MESOSPHERE) return reg::Read(dd::QueryIoMapping(phys_addr, sizeof(u32))); #elif defined(ATMOSPHERE_IS_STRATOSPHERE) u32 val; R_ABORT_UNLESS(svc::ReadWriteRegister(std::addressof(val), phys_addr, 0, 0)); return val; #else #error "Unknown execution context for ams::dd::ReadIoRegister!" #endif } void WriteIoRegister(dd::PhysicalAddress phys_addr, u32 value) { #if defined(ATMOSPHERE_IS_EXOSPHERE) || defined(ATMOSPHERE_IS_MESOSPHERE) reg::Write(dd::QueryIoMapping(phys_addr, sizeof(u32)), value); #elif defined(ATMOSPHERE_IS_STRATOSPHERE) u32 out_val; R_ABORT_UNLESS(svc::ReadWriteRegister(std::addressof(out_val), phys_addr, 0xFFFFFFFF, value)); AMS_UNUSED(out_val); #else #error "Unknown execution context for ams::dd::WriteIoRegister!" #endif } }