/* * Copyright (c) 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 namespace ams::kern { Result KTransferMemory::Initialize(KProcessAddress addr, size_t size, ams::svc::MemoryPermission own_perm) { MESOSPHERE_ASSERT_THIS(); /* Set members. */ m_owner = GetCurrentProcessPointer(); /* Get the owner page table. */ auto &page_table = m_owner->GetPageTable(); /* Construct the page group, guarding to make sure our state is valid on exit. */ auto pg_guard = util::ConstructAtGuarded(m_page_group, page_table.GetBlockInfoManager()); /* Lock the memory. */ R_TRY(page_table.LockForTransferMemory(GetPointer(m_page_group), addr, size, ConvertToKMemoryPermission(own_perm))); /* Set remaining tracking members. */ m_owner->Open(); m_owner_perm = own_perm; m_address = addr; m_is_initialized = true; m_is_mapped = false; /* We succeeded. */ pg_guard.Cancel(); return ResultSuccess(); } void KTransferMemory::Finalize() { MESOSPHERE_ASSERT_THIS(); /* Unlock. */ if (!m_is_mapped) { const size_t size = GetReference(m_page_group).GetNumPages() * PageSize; MESOSPHERE_R_ABORT_UNLESS(m_owner->GetPageTable().UnlockForTransferMemory(m_address, size, GetReference(m_page_group))); } /* Close the page group. */ GetReference(m_page_group).Close(); GetReference(m_page_group).Finalize(); } void KTransferMemory::PostDestroy(uintptr_t arg) { KProcess *owner = reinterpret_cast(arg); owner->ReleaseResource(ams::svc::LimitableResource_TransferMemoryCountMax, 1); owner->Close(); } Result KTransferMemory::Map(KProcessAddress address, size_t size, ams::svc::MemoryPermission map_perm) { MESOSPHERE_ASSERT_THIS(); /* Validate the size. */ R_UNLESS(GetReference(m_page_group).GetNumPages() == util::DivideUp(size, PageSize), svc::ResultInvalidSize()); /* Validate the permission. */ R_UNLESS(m_owner_perm == map_perm, svc::ResultInvalidState()); /* Lock ourselves. */ KScopedLightLock lk(m_lock); /* Ensure we're not already mapped. */ R_UNLESS(!m_is_mapped, svc::ResultInvalidState()); /* Map the memory. */ const KMemoryState state = (m_owner_perm == ams::svc::MemoryPermission_None) ? KMemoryState_Transfered : KMemoryState_SharedTransfered; R_TRY(GetCurrentProcess().GetPageTable().MapPageGroup(address, GetReference(m_page_group), state, KMemoryPermission_UserReadWrite)); /* Mark ourselves as mapped. */ m_is_mapped = true; return ResultSuccess(); } Result KTransferMemory::Unmap(KProcessAddress address, size_t size) { MESOSPHERE_ASSERT_THIS(); /* Validate the size. */ R_UNLESS(GetReference(m_page_group).GetNumPages() == util::DivideUp(size, PageSize), svc::ResultInvalidSize()); /* Lock ourselves. */ KScopedLightLock lk(m_lock); /* Unmap the memory. */ const KMemoryState state = (m_owner_perm == ams::svc::MemoryPermission_None) ? KMemoryState_Transfered : KMemoryState_SharedTransfered; R_TRY(GetCurrentProcess().GetPageTable().UnmapPageGroup(address, GetReference(m_page_group), state)); /* Mark ourselves as unmapped. */ MESOSPHERE_ASSERT(m_is_mapped); m_is_mapped = false; return ResultSuccess(); } }