mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-03 11:11:14 +00:00
kern: Implement QueryIoMapping logic for < 8.0.0
This commit is contained in:
parent
ff022115ca
commit
d9e9fbe3c2
1 changed files with 40 additions and 25 deletions
|
@ -32,7 +32,7 @@ namespace ams::kern::svc {
|
||||||
/* Check whether the address is aligned. */
|
/* Check whether the address is aligned. */
|
||||||
const bool aligned = util::IsAligned(phys_addr, PageSize);
|
const bool aligned = util::IsAligned(phys_addr, PageSize);
|
||||||
|
|
||||||
if (aligned) {
|
auto QueryIoMappingFromPageTable = [&] ALWAYS_INLINE_LAMBDA (uint64_t phys_addr, size_t size) -> Result {
|
||||||
/* The size must be non-zero. */
|
/* The size must be non-zero. */
|
||||||
R_UNLESS(size > 0, svc::ResultInvalidSize());
|
R_UNLESS(size > 0, svc::ResultInvalidSize());
|
||||||
|
|
||||||
|
@ -44,33 +44,48 @@ namespace ams::kern::svc {
|
||||||
|
|
||||||
/* Use the size as the found size. */
|
/* Use the size as the found size. */
|
||||||
found_size = size;
|
found_size = size;
|
||||||
|
|
||||||
|
return ResultSuccess();
|
||||||
|
};
|
||||||
|
|
||||||
|
if (aligned) {
|
||||||
|
/* Query the input. */
|
||||||
|
R_TRY(QueryIoMappingFromPageTable(phys_addr, size));
|
||||||
} else {
|
} else {
|
||||||
/* TODO: Older kernel ABI compatibility. */
|
if (kern::GetTargetFirmware() < TargetFirmware_8_0_0 && phys_addr >= PageSize) {
|
||||||
/* Newer kernel only allows unaligned addresses when they're special enum members. */
|
/* Query the aligned-down page. */
|
||||||
R_UNLESS(phys_addr < PageSize, svc::ResultNotFound());
|
const size_t offset = phys_addr & (PageSize - 1);
|
||||||
|
R_TRY(QueryIoMappingFromPageTable(phys_addr - offset, size + offset));
|
||||||
|
|
||||||
/* Try to find the memory region. */
|
/* Adjust the output address. */
|
||||||
const KMemoryRegion *region;
|
found_address += offset;
|
||||||
switch (static_cast<ams::svc::MemoryRegionType>(phys_addr)) {
|
} else {
|
||||||
case ams::svc::MemoryRegionType_KernelTraceBuffer:
|
/* Newer kernel only allows unaligned addresses when they're special enum members. */
|
||||||
region = KMemoryLayout::TryGetKernelTraceBufferRegion();
|
R_UNLESS(phys_addr < PageSize, svc::ResultNotFound());
|
||||||
break;
|
|
||||||
case ams::svc::MemoryRegionType_OnMemoryBootImage:
|
/* Try to find the memory region. */
|
||||||
region = KMemoryLayout::TryGetOnMemoryBootImageRegion();
|
const KMemoryRegion *region;
|
||||||
break;
|
switch (static_cast<ams::svc::MemoryRegionType>(phys_addr)) {
|
||||||
case ams::svc::MemoryRegionType_DTB:
|
case ams::svc::MemoryRegionType_KernelTraceBuffer:
|
||||||
region = KMemoryLayout::TryGetDTBRegion();
|
region = KMemoryLayout::TryGetKernelTraceBufferRegion();
|
||||||
break;
|
break;
|
||||||
default:
|
case ams::svc::MemoryRegionType_OnMemoryBootImage:
|
||||||
region = nullptr;
|
region = KMemoryLayout::TryGetOnMemoryBootImageRegion();
|
||||||
break;
|
break;
|
||||||
|
case ams::svc::MemoryRegionType_DTB:
|
||||||
|
region = KMemoryLayout::TryGetDTBRegion();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
region = nullptr;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure that we found the region. */
|
||||||
|
R_UNLESS(region != nullptr, svc::ResultNotFound());
|
||||||
|
|
||||||
|
R_TRY(pt.QueryStaticMapping(std::addressof(found_address), region->GetAddress(), region->GetSize()));
|
||||||
|
found_size = region->GetSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Ensure that we found the region. */
|
|
||||||
R_UNLESS(region != nullptr, svc::ResultNotFound());
|
|
||||||
|
|
||||||
R_TRY(pt.QueryStaticMapping(std::addressof(found_address), region->GetAddress(), region->GetSize()));
|
|
||||||
found_size = region->GetSize();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* We succeeded. */
|
/* We succeeded. */
|
||||||
|
|
Loading…
Reference in a new issue